weeder.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * weeder.c - Weeder Solid State Relay Module Controller;
  11. *
  12. * Contributor(s):
  13. * Charles Maier <cmaier@qca.qualcomm.com>
  14. * Nathaniel Houghton <nhoughto@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/error.h"
  48. /*====================================================================*
  49. * custom source files;
  50. *--------------------------------------------------------------------*/
  51. #ifndef MAKEFILE
  52. #include "../tools/getoptv.c"
  53. #include "../tools/putoptv.c"
  54. #include "../tools/version.c"
  55. #include "../tools/uintspec.c"
  56. #include "../tools/synonym.c"
  57. #include "../tools/todigit.c"
  58. #include "../tools/error.c"
  59. #endif
  60. /*====================================================================*
  61. * program constants;
  62. *--------------------------------------------------------------------*/
  63. #define WEEDER_UNITS "BA"
  64. #define WEEDER_LEDS 5
  65. #define WEEDER_BITS 7
  66. #define WEEDER_WAIT 25
  67. #define WEEDER_ECHO 0
  68. #define WEEDER_MODE 1
  69. #define WEEDER_BUFFER_LENGTH 10
  70. #define WEEDER_STRING_LENGTH 15
  71. #ifdef WIN32
  72. # define WEEDER_PORT "com1:"
  73. #else
  74. # define WEEDER_PORT "/dev/ttyS0"
  75. #endif
  76. #define WEEDER_SILENCE (1 << 0)
  77. #define WEEDER_VERBOSE (1 << 1)
  78. #define WEEDER_DISPLAY (1 << 2)
  79. #define WEEDER_NEWLINE (1 << 2)
  80. /*====================================================================*
  81. * program variables;
  82. *--------------------------------------------------------------------*/
  83. static struct _term_ const modes [] =
  84. {
  85. {
  86. "off",
  87. "0"
  88. },
  89. {
  90. "on",
  91. "1"
  92. }
  93. };
  94. static char buffer [WEEDER_BUFFER_LENGTH];
  95. static char string [WEEDER_STRING_LENGTH];
  96. static signed length = 0;
  97. static signed offset = 0;
  98. /*====================================================================*
  99. *
  100. * void function1 (struct _file_ * port, char const * units, unsigned wait, unsigned echo);
  101. *
  102. * send echo command to Weeder Solid State Relay modules B then A;
  103. * Standard Atheros relay modules were wired in reverse order for
  104. * some reason;
  105. *
  106. *--------------------------------------------------------------------*/
  107. static void function1 (struct _file_ * port, char const * units, unsigned wait, unsigned echo)
  108. {
  109. extern char buffer [];
  110. extern signed length;
  111. while (* units)
  112. {
  113. length = 0;
  114. buffer [length++] = * units++;
  115. buffer [length++] = 'X';
  116. buffer [length++] = '0' + (echo & 1);
  117. buffer [length++] = '\r';
  118. if (write (port->file, buffer, length) != length)
  119. {
  120. error (1, errno, FILE_CANTSAVE, port->name);
  121. }
  122. SLEEP (wait);
  123. }
  124. return;
  125. }
  126. /*====================================================================*
  127. *
  128. * void function2 (struct _file_ * port, char const * units, unsigned wait, unsigned mode, unsigned data);
  129. *
  130. * send write command to Weeder Solid State Relay modules B then A
  131. * because Qualcomm Atheros relay modules are wired in reverse order
  132. * for some reason;
  133. *
  134. *--------------------------------------------------------------------*/
  135. static void function2 (struct _file_ * port, char const * units, unsigned wait, unsigned mode, unsigned data)
  136. {
  137. extern char buffer [WEEDER_BUFFER_LENGTH];
  138. extern signed length;
  139. length = 0;
  140. buffer [length++] = * units++;
  141. buffer [length++] = 'W';
  142. buffer [length++] = '0' + (mode & 1);
  143. buffer [length++] = '0';
  144. buffer [length++] = '0';
  145. while (length < WEEDER_BITS)
  146. {
  147. buffer [length++] = '0' + (data & 1);
  148. data >>= 1;
  149. }
  150. buffer [length++] = '\r';
  151. if (write (port->file, buffer, length) != length)
  152. {
  153. error (1, errno, FILE_CANTSAVE, port->name);
  154. }
  155. SLEEP (wait);
  156. length = 0;
  157. buffer [length++] = * units++;
  158. buffer [length++] = 'W';
  159. while (length < WEEDER_BITS)
  160. {
  161. buffer [length++] = '0' + (data & 1);
  162. data >>= 1;
  163. }
  164. buffer [length++] = '\r';
  165. if (write (port->file, buffer, length) != length)
  166. {
  167. error (1, errno, FILE_CANTSAVE, port->name);
  168. }
  169. SLEEP (wait);
  170. return;
  171. }
  172. /*====================================================================*
  173. *
  174. * void function3 (struct _file_ * port, char const * units, unsigned wait);
  175. *
  176. * read weeder solid state controller and display settings on the
  177. * console as attenuation;
  178. *
  179. *--------------------------------------------------------------------*/
  180. static void function3 (struct _file_ * port, char const * units, unsigned wait)
  181. {
  182. extern char buffer [WEEDER_BUFFER_LENGTH];
  183. extern char string [WEEDER_STRING_LENGTH];
  184. extern signed length;
  185. extern signed offset;
  186. unsigned number = 0;
  187. memset (string, 0, sizeof (string));
  188. for (offset = 0; * units; offset += WEEDER_LEDS)
  189. {
  190. length = 0;
  191. buffer [length++] = * units++;
  192. buffer [length++] = 'R';
  193. buffer [length++] = '\r';
  194. if (write (port->file, buffer, length) != length)
  195. {
  196. error (1, errno, FILE_CANTSAVE, port->name);
  197. }
  198. SLEEP (wait);
  199. memset (buffer, 0, sizeof (buffer));
  200. if (read (port->file, buffer, WEEDER_LEDS + 2) == - 1)
  201. {
  202. error (1, errno, FILE_CANTREAD, port->name);
  203. }
  204. SLEEP (wait);
  205. memcpy (& string [offset], & buffer [1], WEEDER_LEDS);
  206. }
  207. while (offset-- > 3)
  208. {
  209. number <<= 1;
  210. number |= string [offset] - '0';
  211. }
  212. printf ("%d\n", number);
  213. return;
  214. }
  215. /*====================================================================*
  216. *
  217. * int main (int argc, char const * argv []);
  218. *
  219. *--------------------------------------------------------------------*/
  220. int main (int argc, char const * argv [])
  221. {
  222. static char const * optv [] =
  223. {
  224. "e:m:o:p:iqrvw:",
  225. "",
  226. "Weeder Solid State Relay Module Controller",
  227. "e n\techo is (n) [" LITERAL (WEEDER_ECHO) "]",
  228. "m n\tmode is (n) [" LITERAL (WEEDER_MODE) "]",
  229. "o s\tunit order is (s) [" WEEDER_UNITS "]",
  230. "p f\tport is (f) [" WEEDER_PORT "]",
  231. "q\tquiet mode",
  232. "r\tread attenuator value",
  233. "v\tverbose mode",
  234. "w n\twait (n) millseconds [" LITERAL (WEEDER_WAIT) "]",
  235. (char const *) (0)
  236. };
  237. struct _file_ port =
  238. {
  239. - 1,
  240. WEEDER_PORT
  241. };
  242. #if defined (WIN32)
  243. HANDLE hSerial;
  244. DCB dcbSerial =
  245. {
  246. 0
  247. };
  248. #else
  249. struct termios termios;
  250. #endif
  251. char const * units = WEEDER_UNITS;
  252. unsigned wait = WEEDER_WAIT;
  253. unsigned echo = WEEDER_ECHO;
  254. unsigned mode = WEEDER_MODE;
  255. unsigned data = 0;
  256. flag_t flags = (flag_t) (0);
  257. signed c;
  258. optind = 1;
  259. if (getenv ("WEEDER"))
  260. {
  261. port.name = strdup (getenv ("WEEDER"));
  262. }
  263. while (~ (c = getoptv (argc, argv, optv)))
  264. {
  265. switch (c)
  266. {
  267. case 'e':
  268. echo = (unsigned) (uintspec (synonym (optarg, modes, SIZEOF (modes)), 0, 1));
  269. break;
  270. case 'm':
  271. mode = (unsigned) (uintspec (synonym (optarg, modes, SIZEOF (modes)), 0, 1));
  272. break;
  273. case 'n':
  274. _setbits (flags, WEEDER_NEWLINE);
  275. break;
  276. case 'o':
  277. units = optarg;
  278. break;
  279. case 'p':
  280. port.name = optarg;
  281. break;
  282. case 'w':
  283. wait = (unsigned) (uintspec (optarg, 5, 100));
  284. break;
  285. case 'q':
  286. _setbits (flags, WEEDER_SILENCE);
  287. break;
  288. case 'r':
  289. _setbits (flags, WEEDER_DISPLAY);
  290. break;
  291. case 'v':
  292. _setbits (flags, WEEDER_VERBOSE);
  293. break;
  294. default:
  295. break;
  296. }
  297. }
  298. argc -= optind;
  299. argv += optind;
  300. if ((argc) && (* argv))
  301. {
  302. data = (unsigned) (uintspec (* argv, 0, 0x7F));
  303. }
  304. #if defined (WIN32)
  305. hSerial = CreateFile (port.name, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
  306. if (hSerial == INVALID_HANDLE_VALUE)
  307. {
  308. error (1, errno, FILE_CANTOPEN, port.name);
  309. }
  310. dcbSerial.DCBlength = sizeof (dcbSerial);
  311. if (! GetCommState (hSerial, & dcbSerial))
  312. {
  313. error (1, 0, FILE_CANTREAD " state", port.name);
  314. }
  315. dcbSerial.BaudRate = CBR_9600;
  316. dcbSerial.ByteSize = 8;
  317. dcbSerial.StopBits = ONESTOPBIT;
  318. dcbSerial.Parity = NOPARITY;
  319. if (! SetCommState (hSerial, & dcbSerial))
  320. {
  321. error (1, 0, FILE_CANTSAVE, port.name);
  322. }
  323. CloseHandle (hSerial);
  324. if ((port.file = open (port.name, O_BINARY | O_RDWR)) == - 1)
  325. {
  326. error (1, errno, FILE_CANTOPEN " state", port.name);
  327. }
  328. #else
  329. if ((port.file = open (port.name, O_RDWR | O_NOCTTY | O_NDELAY)) == - 1)
  330. {
  331. error (1, 0, FILE_CANTOPEN, port.name);
  332. }
  333. tcgetattr (port.file, & termios);
  334. termios.c_cflag = CS8;
  335. cfsetospeed (& termios, B9600);
  336. tcsetattr (port.file, TCSANOW, & termios);
  337. #endif
  338. function1 (& port, units, wait, echo);
  339. if ((argc) && (* argv))
  340. {
  341. function2 (& port, units, wait, mode, data);
  342. }
  343. if (_anyset (flags, WEEDER_DISPLAY))
  344. {
  345. function3 (& port, units, wait);
  346. }
  347. close (port.file);
  348. exit (0);
  349. }