ptsctl.c 11 KB

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