serial.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * serial.c - Qualcomm Atheros Serial Line Command Buffer Management;
  11. *
  12. * serial.h
  13. *
  14. * this module contains a serial line command buffer and functions
  15. * to encode and decode it in different formats and send or receive
  16. * it over the serial line;
  17. *
  18. * Contributor(s):
  19. * Charles Maier <cmaier@qca.qualcomm.com>
  20. *
  21. *--------------------------------------------------------------------*/
  22. #ifndef SERIAL_SOURCE
  23. #define SERIAL_SOURCE
  24. /*====================================================================*
  25. * system header files;
  26. *--------------------------------------------------------------------*/
  27. #include <unistd.h>
  28. #include <stdlib.h>
  29. #include <stdint.h>
  30. #include <memory.h>
  31. #include <errno.h>
  32. #if defined (WIN32)
  33. #include <Windows.h>
  34. #endif
  35. /*====================================================================*
  36. * custom header files;
  37. *--------------------------------------------------------------------*/
  38. #include "../serial/serial.h"
  39. #include "../tools/number.h"
  40. #include "../tools/types.h"
  41. #include "../tools/flags.h"
  42. #include "../tools/error.h"
  43. /*====================================================================*
  44. * private variables;
  45. *--------------------------------------------------------------------*/
  46. struct command command;
  47. /*====================================================================*
  48. *
  49. * void clearcommand ();
  50. *
  51. * serial.h
  52. *
  53. * erase the current command by writing 0s;
  54. *
  55. *--------------------------------------------------------------------*/
  56. void clearcommand ()
  57. {
  58. extern struct command command;
  59. memset (& command, 0, sizeof (command));
  60. return;
  61. }
  62. /*====================================================================*
  63. *
  64. * void sendcommand (struct _file_ * port, flag_t flags);
  65. *
  66. * serial.h
  67. *
  68. * echo then send the command;
  69. *
  70. *--------------------------------------------------------------------*/
  71. void sendcommand (struct _file_ * port, flag_t flags)
  72. {
  73. extern struct command command;
  74. if (_anyset (flags, UART_VERBOSE))
  75. {
  76. write (STDERR_FILENO, command.buffer, command.length);
  77. write (STDERR_FILENO, "\n", sizeof (char));
  78. }
  79. if (write (port->file, command.buffer, command.length) != (signed) (command.length))
  80. {
  81. error (1, errno, "Can't write to %s", port->name);
  82. }
  83. clearcommand ();
  84. return;
  85. }
  86. /*====================================================================*
  87. *
  88. * void readcommand (struct _file_ * port, flag_t flags);
  89. *
  90. * serial.h
  91. *
  92. * read response serial line and log the response;
  93. *
  94. *--------------------------------------------------------------------*/
  95. void readcommand (struct _file_ * port, flag_t flags)
  96. {
  97. extern struct command command;
  98. #if defined (WIN32)
  99. PAUSE (250);
  100. memset (& command, 0, sizeof (command));
  101. command.length = read (port->file, command.buffer, sizeof (command.buffer));
  102. if (command.length < 0)
  103. {
  104. error (1, errno, "Bad response from %s", port->name);
  105. }
  106. if (command.length == 0)
  107. {
  108. error (1, errno, "No response from %s", port->name);
  109. }
  110. #else
  111. struct timeval tv;
  112. fd_set rfd;
  113. ssize_t tmp;
  114. memset (& command, 0, sizeof (command));
  115. while (! strchr (command.buffer, '\r'))
  116. {
  117. tv.tv_sec = 1;
  118. tv.tv_usec = 0;
  119. FD_ZERO (& rfd);
  120. FD_SET (port->file, & rfd);
  121. if (select (port->file + 1, & rfd, NULL, NULL, & tv) != 1)
  122. {
  123. error (1, errno, "Read timeout");
  124. }
  125. tmp = read (port->file, command.buffer + command.length, sizeof (command.buffer) - command.length - 1);
  126. if (tmp < 0)
  127. {
  128. error (1, errno, "Could not read %s", port->name);
  129. }
  130. command.length += tmp;
  131. command.buffer [command.length] = '\0';
  132. }
  133. #endif
  134. if (_anyset (flags, UART_VERBOSE))
  135. {
  136. write (STDERR_FILENO, command.buffer, command.length);
  137. write (STDERR_FILENO, "\n", sizeof (char));
  138. }
  139. if (! memcmp (command.buffer, "ERROR", 5))
  140. {
  141. error (1, ECANCELED, "Device refused request");
  142. }
  143. return;
  144. }
  145. /*====================================================================*
  146. *
  147. * void insert (char c);
  148. *
  149. * serial.h
  150. *
  151. * insert a character into the command buffer at the current buffer
  152. * position then increment the buffer position pointer;
  153. *
  154. *--------------------------------------------------------------------*/
  155. void insert (char c)
  156. {
  157. extern struct command command;
  158. if (command.length < sizeof (command.buffer))
  159. {
  160. command.buffer [command.length++] = c;
  161. }
  162. return;
  163. }
  164. /*====================================================================*
  165. *
  166. * unsigned readframe (signed fd, void * memory, size_t extent);
  167. *
  168. * serial.h
  169. *
  170. * read a file and convert hexadecimal octets to binary bytes then
  171. * store them in consecutive memory locations up to a given length;
  172. * return the actual number of bytes stored;
  173. *
  174. * digits may be consecutive or separated by white space consisting
  175. * of spaces, tabs, linefeeds, carriage returns, formfeeds or other
  176. * characters such as punctuation; script-style comments are treated
  177. * as white space;
  178. *
  179. *--------------------------------------------------------------------*/
  180. static signed fdgetc (signed fd)
  181. {
  182. char c;
  183. return ((read (fd, & c, sizeof (c)) == sizeof (c))? c: EOF);
  184. }
  185. size_t readframe (signed fd, void * memory, size_t extent)
  186. {
  187. unsigned digits = 0;
  188. uint8_t * origin = (uint8_t *) (memory);
  189. uint8_t * offset = (uint8_t *) (memory);
  190. signed c = EOF;
  191. while ((extent) && ((c = fdgetc (fd)) != EOF) && (c != ';'))
  192. {
  193. if (isspace (c))
  194. {
  195. continue;
  196. }
  197. if (c == '#')
  198. {
  199. do
  200. {
  201. c = fdgetc (fd);
  202. }
  203. while ((c != '\n') && (c != EOF));
  204. continue;
  205. }
  206. if (c == '/')
  207. {
  208. c = fdgetc (fd);
  209. if (c == '/')
  210. {
  211. do
  212. {
  213. c = fdgetc (fd);
  214. }
  215. while ((c != '\n') && (c != EOF));
  216. continue;
  217. }
  218. if (c == '*')
  219. {
  220. while ((c != '/') && (c != EOF))
  221. {
  222. while ((c != '*') && (c != EOF))
  223. {
  224. c = fdgetc (fd);
  225. }
  226. c = fdgetc (fd);
  227. }
  228. continue;
  229. }
  230. continue;
  231. }
  232. if (isxdigit (c))
  233. {
  234. * offset = c;
  235. offset++;
  236. digits++;
  237. extent--;
  238. continue;
  239. }
  240. error (1, ENOTSUP, "Illegal hex digit '%c' (0x%02X) in source", c, c);
  241. }
  242. if (digits & 1)
  243. {
  244. error (1, ENOTSUP, "Odd number of hex digits (%d) in source", digits);
  245. }
  246. return (offset - origin);
  247. }
  248. /*====================================================================*
  249. *
  250. * void decode (void const * memory, size_t extent);
  251. *
  252. * serial.h
  253. *
  254. * copy a memory region into command buffer at the current position
  255. * and increment the buffer position pointer; convert bytes to hex
  256. * octets;
  257. *
  258. *--------------------------------------------------------------------*/
  259. void decode (void const * memory, size_t extent)
  260. {
  261. extern struct command command;
  262. register byte * binary = (byte *) (memory);
  263. while ((command.length < sizeof (command.buffer)) && (extent--))
  264. {
  265. insert (DIGITS_HEX [(* binary >> 4) & 0x0F]);
  266. insert (DIGITS_HEX [(* binary >> 0) & 0x0F]);
  267. binary++;
  268. }
  269. return;
  270. }
  271. /*====================================================================*
  272. *
  273. * void encode (void * memory, size_t extent);
  274. *
  275. * serial.h
  276. *
  277. * encode a memory region from the current command buffer position
  278. * and increment the command buffer position pointer;
  279. *
  280. *--------------------------------------------------------------------*/
  281. void encode (void * memory, size_t extent)
  282. {
  283. extern struct command command;
  284. register byte * binary = (byte *) (memory);
  285. unsigned digit;
  286. while ((command.offset < command.length) && (extent--))
  287. {
  288. * binary = 0;
  289. if ((digit = todigit (command.buffer [command.offset++])) > 0x0F)
  290. {
  291. command.buffer [command.offset] = (char) (0);
  292. error (1, EINVAL, "[%s]1", command.buffer);
  293. }
  294. * binary |= digit << 4;
  295. if ((digit = todigit (command.buffer [command.offset++])) > 0x0F)
  296. {
  297. command.buffer [command.offset] = (char) (0);
  298. error (1, EINVAL, "[%s]2", command.buffer);
  299. }
  300. * binary |= digit;
  301. binary++;
  302. }
  303. return;
  304. }
  305. /*====================================================================*
  306. *
  307. * void string (char * string);
  308. *
  309. * serial.h
  310. *
  311. * extract the contents of a quoted string string from the command
  312. * buffer; it assumes that the current char is a quote character;
  313. *
  314. * copy command buffer characters to an external string; start a the
  315. * current buffer position and continue until the buffer exhausts or
  316. * a closing quote is encountered; NUL terminate the string;
  317. *
  318. *--------------------------------------------------------------------*/
  319. void string (char * string)
  320. {
  321. extern struct command command;
  322. while ((command.offset < command.length) && (command.buffer [command.offset] != '\"'))
  323. {
  324. * string++ = command.buffer [command.offset++];
  325. }
  326. * string = (char) (0);
  327. return;
  328. }
  329. /*====================================================================*
  330. *
  331. * uint64_t hextoint (unsigned bytes);
  332. *
  333. * serial.h
  334. *
  335. * this function is used to extract a hexadecimal integer string as
  336. * an integer of specified length; an error occurs of the string is
  337. * to long for the specified integer size in bytes;
  338. *
  339. *--------------------------------------------------------------------*/
  340. uint64_t hextoint (unsigned bytes)
  341. {
  342. extern struct command command;
  343. uint64_t limit = - 1;
  344. uint64_t value = 0;
  345. unsigned radix = 16;
  346. unsigned digit = 0;
  347. if (bytes < sizeof (limit))
  348. {
  349. limit <<= (bytes << 3);
  350. limit = ~ limit;
  351. }
  352. while ((digit = todigit (command.buffer [command.offset])) < radix)
  353. {
  354. value *= radix;
  355. value += digit;
  356. command.offset++;
  357. if (value > limit)
  358. {
  359. command.buffer [command.offset] = (char) (0);
  360. error (1, EINVAL, "[%s] exceeds %d bits", command.buffer, (bytes << 3));
  361. }
  362. }
  363. return (value);
  364. }
  365. /*====================================================================*
  366. *
  367. * void mustbe (char c);
  368. *
  369. * serial.h
  370. *
  371. * test the character at the current buffer position; advance the
  372. * buffer position pointer and return true on match; terminate the
  373. * program on mismatch or exhausted buffer;
  374. *
  375. *--------------------------------------------------------------------*/
  376. void mustbe (char c)
  377. {
  378. extern struct command command;
  379. if (command.offset >= command.length)
  380. {
  381. command.buffer [command.offset] = (char) (0);
  382. error (1, EINVAL, "[%s]: overflow", command.buffer);
  383. }
  384. if (command.buffer [command.offset++] != (c))
  385. {
  386. command.buffer [command.offset] = (char) (0);
  387. error (1, EINVAL, "[%s]: expecting 0x%02X", command.buffer, c);
  388. }
  389. return;
  390. }
  391. /*====================================================================*
  392. *
  393. *--------------------------------------------------------------------*/
  394. #endif