relay.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * ptsctl.c - PTS Module Controller;
  11. *
  12. * Contributor(s):
  13. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  14. * Charles Maier <cmaier@qca.qualcomm.com>
  15. * Mathieu Olivari <mathieu@qca.qualcomm.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. /*====================================================================*
  19. * system header files;
  20. *--------------------------------------------------------------------*/
  21. #include <unistd.h>
  22. #include <stdlib.h>
  23. #include <string.h>
  24. #include <ctype.h>
  25. #if defined (__linux__)
  26. # include <termios.h>
  27. #elif defined (__APPLE__)
  28. # include <termios.h>
  29. #elif defined (__OpenBSD__)
  30. # include <termios.h>
  31. #elif defined (WIN32)
  32. # include <windows.h>
  33. #else
  34. #error "Unknown Environment"
  35. #endif
  36. /*====================================================================*
  37. * custom header files;
  38. *--------------------------------------------------------------------*/
  39. #include "../tools/getoptv.h"
  40. #include "../tools/putoptv.h"
  41. #include "../tools/version.h"
  42. #include "../tools/number.h"
  43. #include "../tools/symbol.h"
  44. #include "../tools/timer.h"
  45. #include "../tools/files.h"
  46. #include "../tools/flags.h"
  47. #include "../tools/timer.h"
  48. #include "../tools/error.h"
  49. //#include "../plc/plc.h"
  50. //#include "../plc/pts.h"
  51. /*====================================================================*
  52. * custom source files;
  53. *--------------------------------------------------------------------*/
  54. #ifndef MAKEFILE
  55. #include "../tools/getoptv.c"
  56. #include "../tools/putoptv.c"
  57. #include "../tools/version.c"
  58. #include "../tools/uintspec.c"
  59. #include "../tools/synonym.c"
  60. #include "../tools/todigit.c"
  61. #include "../tools/error.c"
  62. #endif
  63. /*====================================================================*
  64. * program constants;
  65. *--------------------------------------------------------------------*/
  66. #define PTSCTL_DEBUG 0
  67. #define PTSCTL_UNITS "CBA"
  68. #define PTSCTL_LEDS 5
  69. #define PTSCTL_BITS 7
  70. #define PTSCTL_WAIT 50
  71. #define PTSCTL_ECHO 0
  72. #define PTSCTL_MODE 1
  73. #define PTSCTL_LINE_ATTN 127
  74. #define PTSCTL_GRND_ATTN 127
  75. #define PTSCTL_BUFFER_SIZE 10
  76. #define PTSCTL_STRING_SIZE 15
  77. #ifdef WIN32
  78. # define PTSCTL_PORT "com1:"
  79. #else
  80. # define PTSCTL_PORT "/dev/ttyS0"
  81. #endif
  82. #define PTSCTL_SILENCE (1 << 0)
  83. #define PTSCTL_VERBOSE (1 << 1)
  84. #define PTSCTL_CHANGE (1 << 2)
  85. #define PTSCTL_DISPLAY (1 << 3)
  86. #define PTSCTL_ITERATE (1 << 4)
  87. /*====================================================================*
  88. * program variables;
  89. *--------------------------------------------------------------------*/
  90. #if 0
  91. static const struct _term_ modes [] =
  92. {
  93. {
  94. "off",
  95. "0"
  96. },
  97. {
  98. "on",
  99. "1"
  100. }
  101. };
  102. #endif
  103. static char buffer [PTSCTL_BUFFER_SIZE];
  104. static char string [PTSCTL_STRING_SIZE];
  105. static signed length = 0;
  106. static signed offset = 0;
  107. /*====================================================================*
  108. *
  109. * void cycle (char * string, unsigned offset, unsigned length);
  110. *
  111. * rotate a number of consecutive characters starting at a given
  112. * offset within a string; this is used to shift the character,
  113. * that represents the power on/off bit, out of the way during
  114. * data conversions from binary to ASCII and ASCII to binary;
  115. *
  116. *--------------------------------------------------------------------*/
  117. static void cycle (char * string, unsigned offset, unsigned length)
  118. {
  119. signed c = string [offset];
  120. memcpy (& string [offset], & string [offset + 1], length);
  121. string [offset + length] = c;
  122. return;
  123. }
  124. /*====================================================================*
  125. *
  126. * void function1 (struct _file_ * port, char const * units, unsigned wait, unsigned echo);
  127. *
  128. * send echo command to Weeder Solid State Relay modules in an order
  129. * specified by units;
  130. *
  131. *--------------------------------------------------------------------*/
  132. static void function1 (struct _file_ * port, char const * units, unsigned wait, unsigned echo)
  133. {
  134. extern char buffer [PTSCTL_BUFFER_SIZE];
  135. extern signed length;
  136. while (* units)
  137. {
  138. length = 0;
  139. buffer [length++] = * units++;
  140. buffer [length++] = 'X';
  141. buffer [length++] = '0' + (echo & 1);
  142. buffer [length++] = '\r';
  143. if (write (port->file, buffer, length) != length)
  144. {
  145. error (1, errno, FILE_CANTSAVE, port->name);
  146. }
  147. SLEEP (wait);
  148. }
  149. return;
  150. }
  151. /*====================================================================*
  152. *
  153. * void function2 (struct _file_ * port, char const * units, unsigned wait, unsigned data);
  154. *
  155. * send write command to Weeder Solid State Relay modules in an
  156. * order specified by units;
  157. *
  158. *--------------------------------------------------------------------*/
  159. static void function2 (struct _file_ * port, char const * units, unsigned wait, unsigned data)
  160. {
  161. extern char buffer [PTSCTL_BUFFER_SIZE];
  162. extern char string [PTSCTL_STRING_SIZE];
  163. extern signed length;
  164. extern signed offset;
  165. memset (string, 0, sizeof (string));
  166. memset (buffer, 0, sizeof (buffer));
  167. for (offset = 0; offset < (signed) (sizeof (string)); offset++)
  168. {
  169. string [offset] = '0' + (data & 1);
  170. data >>= 1;
  171. }
  172. cycle (string, 0, 5);
  173. for (offset = 0; * units; offset += PTSCTL_LEDS)
  174. {
  175. length = 0;
  176. buffer [length++] = * units++;
  177. buffer [length++] = 'W';
  178. memcpy (& buffer [length], & string [offset], PTSCTL_LEDS);
  179. length += PTSCTL_LEDS;
  180. buffer [length++] = '\r';
  181. if (write (port->file, buffer, length) != length)
  182. {
  183. error (1, errno, FILE_CANTSAVE, port->name);
  184. }
  185. SLEEP (wait);
  186. }
  187. return;
  188. }
  189. /*====================================================================*
  190. *
  191. * void function3 (struct _file_ * port, char const * units, unsigned wait);
  192. *
  193. * read weeder solid state modules and display settings on the
  194. * console as attenuation;
  195. *
  196. *--------------------------------------------------------------------*/
  197. static void function3 (struct _file_ * port, char const * units, unsigned wait)
  198. {
  199. extern char buffer [PTSCTL_BUFFER_SIZE];
  200. extern char string [PTSCTL_STRING_SIZE];
  201. extern signed length;
  202. extern signed offset;
  203. signed value1 = 0;
  204. signed value2 = 0;
  205. memset (string, 0, sizeof (string));
  206. for (offset = 0; * units; offset += PTSCTL_LEDS)
  207. {
  208. length = 0;
  209. buffer [length++] = * units++;
  210. buffer [length++] = 'R';
  211. buffer [length++] = '\r';
  212. if (write (port->file, buffer, length) != length)
  213. {
  214. error (1, errno, FILE_CANTSAVE, port->name);
  215. }
  216. SLEEP (wait);
  217. memset (buffer, 0, sizeof (buffer));
  218. if (read (port->file, buffer, PTSCTL_LEDS + 2) == - 1)
  219. {
  220. error (1, errno, FILE_CANTREAD, port->name);
  221. }
  222. memcpy (& string [offset], & buffer [1], PTSCTL_LEDS);
  223. SLEEP (wait);
  224. }
  225. cycle (string, PTSCTL_LEDS, 2);
  226. while (-- offset > PTSCTL_BITS)
  227. {
  228. value1 <<= 1;
  229. value1 |= string [offset] - '0';
  230. }
  231. while (offset-- > 0)
  232. {
  233. value2 <<= 1;
  234. value2 |= string [offset] - '0';
  235. }
  236. if ((value1 >= 0) && (value2 >= 0))
  237. {
  238. printf ("%d %d\n", value1, value2);
  239. }
  240. return;
  241. }
  242. /*====================================================================*
  243. *
  244. * void function4 (struct _file_ * port, char const * units, unsigned wait);
  245. *
  246. * sequence through all attenuator settings at one second intervals;
  247. * this function can be used to debug program additions and changes;
  248. *
  249. *--------------------------------------------------------------------*/
  250. static void function4 (struct _file_ * port, char const * units, unsigned wait)
  251. {
  252. signed value;
  253. for (value = 0; value < 128; value++)
  254. {
  255. function2 (port, units, wait, (value << 8) | (value << 1) | 1);
  256. function3 (port, units, wait);
  257. SLEEP (wait);
  258. }
  259. return;
  260. }
  261. /*====================================================================*
  262. *
  263. * int main (int argc, char const * argv []);
  264. *
  265. *--------------------------------------------------------------------*/
  266. int relay (unsigned atn_grd, unsigned atn_line, char const * name )
  267. {
  268. #if 0
  269. static char const * optv [] =
  270. {
  271. "f:g:n:p:iqrvw:z",
  272. "",
  273. "PTS Module Controller",
  274. "f f\tport is (f) [" PTSCTL_PORT "]",
  275. "g n\tline ground attenuation is (n) [" LITERAL (PTSCTL_GRND_ATTN) "]",
  276. "n n\tline neutral attenuation is (n) [" LITERAL (PTSCTL_LINE_ATTN) "]",
  277. "p n\tpower is (n) [" LITERAL (PTSCTL_MODE) "]",
  278. "q\tquiet mode",
  279. "r\tread and display attenuator settings",
  280. "v\tverbose mode",
  281. "w n\twait (n) millseconds [" LITERAL (PTSCTL_WAIT) "]",
  282. (char const *) (0)
  283. };
  284. #endif
  285. struct _file_ port =
  286. {
  287. - 1,
  288. PTSCTL_PORT
  289. };
  290. #if defined (WIN32)
  291. HANDLE hSerial;
  292. DCB dcbSerial =
  293. {
  294. 0
  295. };
  296. #else
  297. struct termios termios;
  298. #endif
  299. char const * units = PTSCTL_UNITS;
  300. unsigned wait = PTSCTL_WAIT;
  301. unsigned mode = PTSCTL_MODE;
  302. unsigned echo = PTSCTL_ECHO;
  303. unsigned line = PTSCTL_LINE_ATTN;
  304. unsigned grnd = PTSCTL_GRND_ATTN;
  305. unsigned data = 0;
  306. flag_t flags = (flag_t) (0);
  307. //signed c;
  308. //int i;
  309. optind = 1;
  310. if (getenv ("PTSCTL"))
  311. {
  312. port.name = strdup (getenv ("PTSCTL"));
  313. }
  314. #if 0
  315. fprintf(stderr, " in relay_driver argc is %d", argc );
  316. for( i=0; i< argc; i++ )
  317. fprintf(stderr, " in ptsctl argv is %s", argv[i] );
  318. while (~ (c = getoptv (argc, argv, optv)))
  319. {
  320. fprintf(stderr, " in relay_driver c is %c", c );
  321. switch (c)
  322. {
  323. case 'f':
  324. port.name = optarg;
  325. break;
  326. case 'g':
  327. _setbits (flags, PTSCTL_CHANGE);
  328. grnd = (unsigned) (uintspec (optarg, 0, 0x7F));
  329. break;
  330. case 'n':
  331. _setbits (flags, PTSCTL_CHANGE);
  332. line = (unsigned) (uintspec (optarg, 0, 0x7F));
  333. break;
  334. case 'p':
  335. _setbits (flags, PTSCTL_CHANGE);
  336. mode = (unsigned) (uintspec (synonym (optarg, modes, SIZEOF (modes)), 0, 1));
  337. break;
  338. case 'w':
  339. wait = (unsigned) (uintspec (optarg, 5, 100));
  340. break;
  341. case 'q':
  342. _setbits (flags, PTSCTL_SILENCE);
  343. break;
  344. case 'r':
  345. _setbits (flags, PTSCTL_DISPLAY);
  346. break;
  347. case 'v':
  348. _setbits (flags, PTSCTL_VERBOSE);
  349. break;
  350. case 'z':
  351. _setbits (flags, PTSCTL_ITERATE);
  352. break;
  353. default:
  354. break;
  355. }
  356. }
  357. argc -= optind;
  358. argv += optind;
  359. if (argc)
  360. {
  361. error (1, ENOTSUP, ERROR_TOOMANY);
  362. }
  363. #endif
  364. port.name = name;
  365. line = atn_line;
  366. grnd = atn_grd;
  367. _setbits (flags, PTSCTL_CHANGE);
  368. // see what we have
  369. //fprintf(stderr, "line atn=%d dB\n", atn_line );
  370. //fprintf(stderr, "ground atn=%d dB\n", atn_grd );
  371. //fprintf(stderr, "selected com port=%s\n", name );
  372. #if defined (WIN32)
  373. hSerial = CreateFile (port.name, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
  374. if (hSerial == INVALID_HANDLE_VALUE)
  375. {
  376. error (1, errno, FILE_CANTOPEN, port.name);
  377. }
  378. dcbSerial.DCBlength = sizeof (dcbSerial);
  379. if (! GetCommState (hSerial, & dcbSerial))
  380. {
  381. error (1, 0, FILE_CANTREAD " state", port.name);
  382. }
  383. dcbSerial.BaudRate = CBR_9600;
  384. dcbSerial.ByteSize = 8;
  385. dcbSerial.StopBits = ONESTOPBIT;
  386. dcbSerial.Parity = NOPARITY;
  387. if (! SetCommState (hSerial, & dcbSerial))
  388. {
  389. error (1, 0, FILE_CANTSAVE " state", port.name);
  390. }
  391. CloseHandle (hSerial);
  392. if ((port.file = open (port.name, O_BINARY | O_RDWR)) == - 1)
  393. {
  394. error (1, errno, FILE_CANTOPEN, port.name);
  395. }
  396. #else
  397. if ((port.file = open (port.name, O_RDWR | O_NOCTTY | O_NDELAY)) == - 1)
  398. {
  399. error (1, 0, FILE_CANTOPEN, port.name);
  400. }
  401. tcgetattr (port.file, & termios);
  402. termios.c_cflag = CS8;
  403. cfsetospeed (& termios, B9600);
  404. tcsetattr (port.file, TCSANOW, & termios);
  405. #endif
  406. function1 (& port, units, wait, echo);
  407. if (_anyset (flags, PTSCTL_CHANGE))
  408. {
  409. data = line << 8 | grnd << 1 | mode;
  410. function2 (& port, units, wait, data);
  411. }
  412. if (_anyset (flags, PTSCTL_DISPLAY))
  413. {
  414. function3 (& port, units, wait);
  415. }
  416. if (_anyset (flags, PTSCTL_ITERATE))
  417. {
  418. function4 (& port, units, wait);
  419. }
  420. close (port.file);
  421. return (0);
  422. }