ttysig.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * ttysig.c - Serial Line Signal Controller;
  11. *
  12. * This program is for Linux only;
  13. *
  14. * Contributor(s):
  15. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  16. *
  17. *--------------------------------------------------------------------*/
  18. /*====================================================================*
  19. * kernel header files;
  20. *--------------------------------------------------------------------*/
  21. #include <sys/ioctl.h>
  22. /*====================================================================*
  23. * system header files;
  24. *--------------------------------------------------------------------*/
  25. #include <ctype.h>
  26. #include <errno.h>
  27. #include <fcntl.h>
  28. #include <termios.h>
  29. #include <stdlib.h>
  30. #include <string.h>
  31. #include <unistd.h>
  32. /*====================================================================*
  33. * custom header files;
  34. *--------------------------------------------------------------------*/
  35. #include "../tools/getoptv.h"
  36. #include "../tools/putoptv.h"
  37. #include "../tools/version.h"
  38. #include "../tools/types.h"
  39. #include "../tools/flags.h"
  40. #include "../tools/number.h"
  41. #include "../tools/error.h"
  42. /*====================================================================*
  43. * custom source files;
  44. *--------------------------------------------------------------------*/
  45. #ifndef MAKEFILE
  46. #include "../tools/getoptv.c"
  47. #include "../tools/putoptv.c"
  48. #include "../tools/version.c"
  49. #include "../tools/uintspec.c"
  50. #include "../tools/todigit.c"
  51. #include "../tools/error.c"
  52. #endif
  53. /*====================================================================*
  54. * constants;
  55. *--------------------------------------------------------------------*/
  56. #define SERIAL_PORT "/dev/ttyS0"
  57. #define TTYSIG_SET_DTR (1 << 0)
  58. #define TTYSIG_SET_RTS (1 << 1)
  59. #define TTYSIG_READ_STATUS (1 << 2)
  60. #define TTYSIG_INTERACTIVE (1 << 3)
  61. #define TTYSIG_NOPROMPT (1 << 4)
  62. void set_status (int fd, int flag, int value)
  63. {
  64. int status;
  65. if (ioctl (fd, TIOCMGET, & status) == - 1) error (1, errno, "TIOCMGET failed");
  66. if (value) status |= flag;
  67. else status &= ~ flag;
  68. if (ioctl (fd, TIOCMSET, & status) == - 1) error (1, errno, "TIOCMSET failed");
  69. }
  70. void print_status (int fd)
  71. {
  72. int status;
  73. if (ioctl (fd, TIOCMGET, & status) == - 1) error (1, errno, "TIOCMGET failed");
  74. printf ("--> DTR: %s\n", (status & TIOCM_DTR)? "+V": "-V");
  75. printf ("--> RTS: %s\n", (status & TIOCM_RTS)? "+V": "-V");
  76. printf ("<-- CTS: %s\n", (status & TIOCM_CTS)? "+V": "-V");
  77. printf ("<-- DSR: %s\n", (status & TIOCM_DSR)? "+V": "-V");
  78. printf ("<-- DCD: %s\n", (status & TIOCM_CD)? "+V": "-V");
  79. printf ("<-- RI : %s\n", (status & TIOCM_RI)? "+V": "-V");
  80. }
  81. void comment (void)
  82. {
  83. int c;
  84. while ((c = getchar ()) != EOF)
  85. {
  86. if (c == '\n')
  87. {
  88. ungetc (c, stdin);
  89. break;
  90. }
  91. }
  92. }
  93. int number (char * buf, int * val)
  94. {
  95. char * p;
  96. while (isspace ((unsigned char)* buf)) ++ buf;
  97. if (! isdigit ((unsigned char)* buf))
  98. {
  99. error (0, 0, "\"%s\" is not a number", buf);
  100. return (- 1);
  101. }
  102. * val = atoi (buf);
  103. p = buf;
  104. while (isdigit ((unsigned char)* buf)) ++ buf;
  105. if (* buf != '\0')
  106. {
  107. error (0, 0, "\"%s\" is not a number", p);
  108. return (- 1);
  109. }
  110. return (0);
  111. }
  112. void interactive (int fd, flag_t flags)
  113. {
  114. char buf [32];
  115. int i,
  116. c;
  117. int value;
  118. char * p;
  119. for (; ; )
  120. {
  121. if (! _anyset (flags, TTYSIG_NOPROMPT))
  122. {
  123. printf ("command (D #, R #, e, r, s, q): ");
  124. fflush (stdout);
  125. }
  126. i = 0;
  127. while ((c = getchar ()) != EOF)
  128. {
  129. if (c == '#')
  130. {
  131. comment ();
  132. continue;
  133. }
  134. if (c == '\n') break;
  135. if (i == sizeof (buf) - 1) error (1, 0, "input too large");
  136. buf [i++] = c;
  137. }
  138. if (c == EOF) return;
  139. if (i == 0) continue;
  140. buf [i] = '\0';
  141. switch (buf [0])
  142. {
  143. case 'D':
  144. if (number (buf + 1, & value)) break;
  145. set_status (fd, TIOCM_DTR, value);
  146. break;
  147. case 'e':
  148. p = buf + 1;
  149. if (* p == ' ') ++ p;
  150. printf ("%s\n", p);
  151. fflush (stdout);
  152. break;
  153. case 'R':
  154. if (number (buf + 1, & value)) break;
  155. set_status (fd, TIOCM_RTS, value);
  156. break;
  157. case 'r':
  158. print_status (fd);
  159. break;
  160. case 's':
  161. if (number (buf + 1, & value)) break;
  162. sleep (value);
  163. break;
  164. case 'q':
  165. case 'Q':
  166. return;
  167. break;
  168. case '\0':
  169. break;
  170. default:
  171. error (0, 0, "invalid command");
  172. }
  173. }
  174. }
  175. int main (int argc, char const ** argv)
  176. {
  177. int fd;
  178. static char const * optv [] =
  179. {
  180. "s:D:IR:rqv",
  181. "[ttysig script filename]",
  182. "Serial Line Signal Controller",
  183. "D n\tset DTR (0 = -V, 1 = +V) at startup",
  184. "I\tInteractive mode",
  185. "R n\tset RTS (0 = -V, 1 = +V) at startup",
  186. "r\tread current RTS/DTR values at startup",
  187. "s f\tserial port is (f) [" SERIAL_PORT "]",
  188. "q\tquiet mode",
  189. "v\tverbose mode",
  190. (char const *) (0)
  191. };
  192. signed c;
  193. optind = 1;
  194. flag_t flags = 0;
  195. uint8_t dtr_value;
  196. uint8_t rts_value;
  197. char * device = SERIAL_PORT;
  198. int input = - 1;
  199. while (~ (c = getoptv (argc, argv, optv)))
  200. {
  201. switch ((char) (c))
  202. {
  203. case 'D':
  204. dtr_value = uintspec (optarg, 0, 1);
  205. _setbits (flags, TTYSIG_SET_DTR);
  206. break;
  207. case 'I':
  208. _setbits (flags, TTYSIG_INTERACTIVE);
  209. break;
  210. case 'R':
  211. rts_value = uintspec (optarg, 0, 1);
  212. _setbits (flags, TTYSIG_SET_RTS);
  213. break;
  214. case 'r':
  215. _setbits (flags, TTYSIG_READ_STATUS);
  216. break;
  217. case 's':
  218. device = optarg;
  219. break;
  220. default:
  221. break;
  222. }
  223. }
  224. argc -= optind;
  225. argv += optind;
  226. if (argc == 1)
  227. {
  228. input = open (* argv, O_RDONLY);
  229. if (input == - 1) error (1, errno, "%s", * argv);
  230. if (dup2 (input, STDIN_FILENO) == - 1) error (1, errno, "%s", * argv);
  231. _setbits (flags, TTYSIG_INTERACTIVE | TTYSIG_NOPROMPT);
  232. }
  233. else if (argc > 0) error (1, 0, "Invalid arguments");
  234. fd = open (device, O_RDONLY);
  235. if (fd == - 1) error (1, errno, "could not open %s", device);
  236. if (_anyset (flags, TTYSIG_SET_DTR)) set_status (fd, TIOCM_DTR, dtr_value);
  237. if (_anyset (flags, TTYSIG_SET_RTS)) set_status (fd, TIOCM_RTS, rts_value);
  238. if (_anyset (flags, TTYSIG_READ_STATUS)) print_status (fd);
  239. if (_anyset (flags, TTYSIG_INTERACTIVE)) interactive (fd, flags);
  240. close (fd);
  241. return (0);
  242. }