int6kdetect.c 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * int6kdetect.c
  11. *
  12. * Contributor(s):
  13. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  14. *
  15. *--------------------------------------------------------------------*/
  16. /*====================================================================*
  17. * system header files;
  18. *--------------------------------------------------------------------*/
  19. #include <fcntl.h>
  20. #include <limits.h>
  21. #include <stdio.h>
  22. #include <stdlib.h>
  23. #include <string.h>
  24. #include <unistd.h>
  25. #ifdef WIN32
  26. #include <windows.h>
  27. #else
  28. #include <termios.h>
  29. #endif
  30. /*====================================================================*
  31. * custom header files;
  32. *--------------------------------------------------------------------*/
  33. #include "../tools/error.h"
  34. #include "../tools/files.h"
  35. #include "../tools/flags.h"
  36. #include "../tools/putoptv.h"
  37. #include "../tools/getoptv.h"
  38. #include "../serial/serial.h"
  39. /*====================================================================*
  40. * custom source files;
  41. *--------------------------------------------------------------------*/
  42. #ifndef MAKEFILE
  43. #include "../tools/getoptv.c"
  44. #include "../tools/putoptv.c"
  45. #include "../tools/version.c"
  46. #include "../tools/error.c"
  47. #endif
  48. #ifndef MAKEFILE
  49. #include "../serial/baudrate.c"
  50. #endif
  51. /*====================================================================*
  52. * program constants;
  53. *--------------------------------------------------------------------*/
  54. #define SERIAL_PORT "/dev/ttyS0"
  55. #define INT6KDETECT_QUIET (1 << 0)
  56. #define INT6KDETECT_CMD_MODE (1 << 1)
  57. struct serial
  58. {
  59. #ifdef WIN32
  60. HANDLE h;
  61. #else
  62. int fd;
  63. #endif
  64. };
  65. #define UART_NOPARITY 0
  66. #define UART_EVENPARITY 1
  67. #define UART_ODDPARITY 2
  68. struct serial_mode
  69. {
  70. int baud_rate;
  71. int parity;
  72. int data_bits;
  73. int stop_bits;
  74. };
  75. ssize_t read_serial (struct serial * s, void * buf, size_t nbytes)
  76. {
  77. #ifdef WIN32
  78. DWORD read;
  79. BOOL r;
  80. r = ReadFile (s->h, buf, (DWORD) nbytes, & read, NULL);
  81. if (r) return read;
  82. else return - 1;
  83. #else
  84. return (read (s->fd, buf, nbytes));
  85. #endif
  86. }
  87. ssize_t write_serial (struct serial * s, void * buf, size_t nbytes)
  88. {
  89. #ifdef WIN32
  90. DWORD written;
  91. BOOL r;
  92. r = WriteFile (s->h, buf, (DWORD) nbytes, & written, NULL);
  93. if (r) return written;
  94. else return - 1;
  95. #else
  96. return (write (s->fd, buf, nbytes));
  97. #endif
  98. }
  99. int open_serial (char const * file, struct serial * s)
  100. {
  101. #ifdef WIN32
  102. s->h = CreateFile (file, GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);
  103. if (s->h == INVALID_HANDLE_VALUE)
  104. {
  105. return (- 1);
  106. }
  107. #else
  108. if ((s->fd = open (file, O_RDWR)) == - 1)
  109. {
  110. return (- 1);
  111. }
  112. #endif
  113. return (0);
  114. }
  115. int close_serial (struct serial * s)
  116. {
  117. #ifdef WIN32
  118. if (CloseHandle (s->h)) return 0;
  119. else return - 1;
  120. #else
  121. return (close (s->fd));
  122. #endif
  123. }
  124. int set_serial (struct serial * s, struct serial_mode * serial_mode)
  125. {
  126. #ifdef WIN32
  127. COMMTIMEOUTS timeouts;
  128. DCB dcbSerial;
  129. memset (& dcbSerial, 0, sizeof (dcbSerial));
  130. dcbSerial.DCBlength = sizeof (dcbSerial);
  131. if (! GetCommState (s->h, & dcbSerial))
  132. {
  133. return (- 1);
  134. }
  135. dcbSerial.BaudRate = serial_mode->baud_rate;
  136. dcbSerial.ByteSize = serial_mode->data_bits;
  137. switch (serial_mode->stop_bits)
  138. {
  139. case 1:
  140. dcbSerial.StopBits = ONESTOPBIT;
  141. break;
  142. case 2:
  143. dcbSerial.StopBits = TWOSTOPBITS;
  144. break;
  145. default:
  146. error (1, 0, "invalid stop bit setting");
  147. }
  148. switch (serial_mode->parity)
  149. {
  150. case UART_ODDPARITY:
  151. dcbSerial.Parity = ODDPARITY;
  152. dcbSerial.fParity = TRUE;
  153. break;
  154. case UART_EVENPARITY:
  155. dcbSerial.Parity = EVENPARITY;
  156. dcbSerial.fParity = TRUE;
  157. break;
  158. case UART_NOPARITY:
  159. dcbSerial.Parity = NOPARITY;
  160. dcbSerial.fParity = FALSE;
  161. break;
  162. default:
  163. error (1, 0, "invalid parity serial_mode");
  164. }
  165. if (! SetCommState (s->h, & dcbSerial))
  166. {
  167. error (0, 0, "could not set serial port settings");
  168. return (- 1);
  169. }
  170. timeouts.ReadIntervalTimeout = 0;
  171. timeouts.ReadTotalTimeoutConstant = 10;
  172. timeouts.ReadTotalTimeoutMultiplier = 0;
  173. timeouts.WriteTotalTimeoutConstant = 10;
  174. timeouts.WriteTotalTimeoutMultiplier = 10;
  175. if (! SetCommTimeouts (s->h, & timeouts))
  176. {
  177. return (- 1);
  178. }
  179. #else
  180. struct termios termios;
  181. speed_t speed;
  182. tcgetattr (s->fd, & termios);
  183. cfmakeraw (& termios);
  184. termios.c_cflag &= ~ CSIZE;
  185. switch (serial_mode->data_bits)
  186. {
  187. case 8:
  188. termios.c_cflag |= CS8;
  189. break;
  190. case 7:
  191. termios.c_cflag |= CS7;
  192. break;
  193. case 6:
  194. termios.c_cflag |= CS6;
  195. break;
  196. case 5:
  197. termios.c_cflag |= CS5;
  198. break;
  199. default:
  200. error (1, 0, "invalid serial byte size");
  201. }
  202. switch (serial_mode->stop_bits)
  203. {
  204. case 2:
  205. termios.c_cflag |= CSTOPB;
  206. break;
  207. case 1:
  208. termios.c_cflag &= ~ CSTOPB;
  209. break;
  210. default:
  211. error (1, 0, "invalid number of stop bits");
  212. }
  213. switch (serial_mode->parity)
  214. {
  215. case UART_ODDPARITY:
  216. termios.c_cflag |= PARENB;
  217. termios.c_cflag |= PARODD;
  218. break;
  219. case UART_EVENPARITY:
  220. termios.c_cflag |= PARENB;
  221. termios.c_cflag &= ~ PARODD;
  222. break;
  223. case UART_NOPARITY:
  224. termios.c_cflag &= ~ PARENB;
  225. break;
  226. default:
  227. error (1, 0, "invalid parity serial_mode");
  228. }
  229. if (baudrate (serial_mode->baud_rate, & speed) == - 1)
  230. {
  231. error (0, 0, "warning: unsupported baud rate: %d", serial_mode->baud_rate);
  232. return (- 1);
  233. }
  234. if (cfsetspeed (& termios, speed) == - 1) error (1, 0, "could not set serial baud rate");
  235. termios.c_cc [VTIME] = 1;
  236. termios.c_cc [VMIN] = 0;
  237. if (tcsetattr (s->fd, TCSANOW, & termios) == - 1) error (1, 0, "could not set serial attributes");
  238. #endif
  239. return (0);
  240. }
  241. int at_cmd (struct serial * s)
  242. {
  243. char buf [32];
  244. ssize_t r;
  245. if (write_serial (s, "AT\r", 3) == - 1) error (1, 0, "could not write");
  246. memset (buf, 0, sizeof (buf));
  247. r = read_serial (s, buf, sizeof (buf) - 1);
  248. if (r < 0) return - 1;
  249. else if (r == 0) return - 1;
  250. if (! strcmp (buf, "OK\r")) return 0;
  251. return (- 1);
  252. }
  253. void wakeup (struct serial * s)
  254. {
  255. sleep (1);
  256. if (write_serial (s, "+++", 3) == - 1) error (1, 0, "could not write");
  257. sleep (1);
  258. }
  259. void dump_serial_mode (struct serial_mode * serial_mode)
  260. {
  261. printf ("baud_rate = %d\n", serial_mode->baud_rate);
  262. printf ("stop_bits = %d\n", serial_mode->stop_bits);
  263. printf ("data_bits = %d\n", serial_mode->data_bits);
  264. printf ("parity = %d\n", serial_mode->parity);
  265. }
  266. int try_serial_mode (struct serial * s, struct serial_mode * serial_mode, flag_t flags)
  267. {
  268. if (set_serial (s, serial_mode) == - 1)
  269. {
  270. error (0, 0, "could not set serial_mode");
  271. return (- 1);
  272. }
  273. if (! _anyset (flags, INT6KDETECT_CMD_MODE)) wakeup (s);
  274. at_cmd (s);
  275. return (at_cmd (s));
  276. }
  277. int detect (struct serial * s, struct serial_mode * serial_mode, flag_t flags)
  278. {
  279. static int rate [] =
  280. {
  281. 115200,
  282. 9600,
  283. 460800,
  284. 230400,
  285. 57600,
  286. 38400,
  287. 19200,
  288. 4800,
  289. 2400,
  290. 600,
  291. 300,
  292. 50
  293. };
  294. static int parity [] =
  295. {
  296. UART_NOPARITY,
  297. UART_EVENPARITY,
  298. UART_ODDPARITY
  299. };
  300. size_t i;
  301. size_t j;
  302. unsigned current;
  303. unsigned total;
  304. total = 2 * 2 * 3 * (sizeof (rate) / sizeof (int));
  305. current = 0;
  306. for (serial_mode->stop_bits = 1; serial_mode->stop_bits <= 2; ++ serial_mode->stop_bits)
  307. {
  308. for (serial_mode->data_bits = 8; serial_mode->data_bits >= 7; -- serial_mode->data_bits)
  309. {
  310. for (i = 0; i < sizeof (parity) / sizeof (int); ++ i)
  311. {
  312. serial_mode->parity = parity [i];
  313. for (j = 0; j < sizeof (rate) / sizeof (int); ++ j)
  314. {
  315. serial_mode->baud_rate = rate [j];
  316. ++ current;
  317. if (! _anyset (flags, INT6KDETECT_QUIET))
  318. {
  319. printf ("\rTesting mode: %03d/%03d (%.01f%%)...", current, total, current * 100.0 / total);
  320. fflush (stdout);
  321. }
  322. if (! try_serial_mode (s, serial_mode, flags))
  323. {
  324. if (! _anyset (flags, INT6KDETECT_QUIET)) printf ("\n");
  325. return (0);
  326. }
  327. }
  328. }
  329. }
  330. }
  331. if (! _anyset (flags, INT6KDETECT_QUIET)) printf ("\n");
  332. return (- 1);
  333. }
  334. int main (int argc, char const * argv [])
  335. {
  336. static char const * optv [] =
  337. {
  338. "cl:qv",
  339. "",
  340. "Qualcomm Atheros UART Device Detector",
  341. "c\tassume device is in command mode",
  342. "l f\tserial port is (f) [" SERIAL_PORT "]",
  343. "q\tquiet mode",
  344. "v\tverbose mode",
  345. (char const *) (0)
  346. };
  347. signed c;
  348. char const * line = SERIAL_PORT;
  349. struct serial serial;
  350. struct serial_mode serial_mode;
  351. flag_t flags = 0;
  352. optind = 1;
  353. while (~ (c = getoptv (argc, argv, optv)))
  354. {
  355. switch ((char) (c))
  356. {
  357. case 'c':
  358. _setbits (flags, INT6KDETECT_CMD_MODE);
  359. break;
  360. case 'l':
  361. line = optarg;
  362. break;
  363. case 'q':
  364. _setbits (flags, INT6KDETECT_QUIET);
  365. break;
  366. default:
  367. break;
  368. }
  369. }
  370. argc -= optind;
  371. argv += optind;
  372. if (open_serial (line, & serial) == - 1) error (1, errno, "could not open %s", line);
  373. if (detect (& serial, & serial_mode, flags) == - 1) error (1, 0, "could not detect device");
  374. printf ("Detected the following serial mode:\n");
  375. dump_serial_mode (& serial_mode);
  376. close_serial (& serial);
  377. return (0);
  378. }