ttyrecv.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. /*====================================================================*
  2. *
  3. * Copyright (c) 2013 Qualcomm Atheros, Inc.
  4. *
  5. * All rights reserved.
  6. *
  7. *====================================================================*/
  8. /*====================================================================*
  9. *
  10. * ttyrecv.c -
  11. *
  12. * Contributor(s):
  13. * Nathaniel Houghton <nhoughto@qca.qualcomm.com>
  14. *
  15. *--------------------------------------------------------------------*/
  16. /*====================================================================*
  17. * system header files;
  18. *--------------------------------------------------------------------*/
  19. #include <sys/time.h>
  20. #include <sys/types.h>
  21. #include <fcntl.h>
  22. #include <limits.h>
  23. #include <stdio.h>
  24. #include <stdlib.h>
  25. #include <string.h>
  26. #include <unistd.h>
  27. #include <termios.h>
  28. /*====================================================================*
  29. * custom header files;
  30. *--------------------------------------------------------------------*/
  31. #include "../tools/error.h"
  32. #include "../tools/putoptv.h"
  33. #include "../tools/getoptv.h"
  34. #include "../tools/number.h"
  35. #include "../tools/types.h"
  36. #include "../tools/flags.h"
  37. #include "../serial/serial.h"
  38. /*====================================================================*
  39. * custom source files;
  40. *--------------------------------------------------------------------*/
  41. #ifndef MAKEFILE
  42. #include "../tools/getoptv.c"
  43. #include "../tools/putoptv.c"
  44. #include "../tools/version.c"
  45. #include "../tools/error.c"
  46. #include "../tools/uintspec.c"
  47. #include "../tools/todigit.c"
  48. #endif
  49. #ifndef MAKEFILE
  50. #include "../serial/baudrate.c"
  51. #endif
  52. /*====================================================================*
  53. * program constants;
  54. *--------------------------------------------------------------------*/
  55. #define SERIAL_PORT "/dev/ttyS0"
  56. #define TTYRECV_VERBOSE (1 << 0)
  57. /*====================================================================*
  58. *
  59. * double ttyrecv (int ifd, int ofd, size_t time, size_t chunk_size, flag_t flags) ;
  60. *
  61. *--------------------------------------------------------------------*/
  62. static double ttyrecv (int ifd, int ofd, size_t time, size_t chunk_size, flag_t flags)
  63. {
  64. char * buf;
  65. char * p;
  66. ssize_t r;
  67. ssize_t w;
  68. size_t bytes_read;
  69. struct timeval tv_start,
  70. tv_now,
  71. tv_result;
  72. struct timeval tv_timeout;
  73. double bytes_sec;
  74. fd_set rfd;
  75. buf = malloc (chunk_size);
  76. if (buf == NULL)
  77. {
  78. error (1, errno, "could not allocate memory");
  79. }
  80. tcflush (ifd, TCIFLUSH);
  81. FD_ZERO (& rfd);
  82. FD_SET (ifd, & rfd);
  83. tv_timeout.tv_sec = 5;
  84. tv_timeout.tv_usec = 0;
  85. if (select (ifd + 1, & rfd, NULL, NULL, & tv_timeout) != 1)
  86. {
  87. error (1, errno, "timed out waiting for data");
  88. }
  89. if (gettimeofday (& tv_start, NULL) == - 1)
  90. {
  91. error (1, errno, "could not get time");
  92. }
  93. if (_anyset (flags, TTYRECV_VERBOSE))
  94. {
  95. fprintf (stderr, "Started receive timer.\n");
  96. }
  97. bytes_read = 0;
  98. do
  99. {
  100. FD_ZERO (& rfd);
  101. FD_SET (ifd, & rfd);
  102. tv_timeout.tv_sec = 1;
  103. tv_timeout.tv_usec = 0;
  104. if (select (ifd + 1, & rfd, NULL, NULL, & tv_timeout) == 1)
  105. {
  106. r = read (ifd, buf, chunk_size);
  107. if (r == - 1)
  108. {
  109. error (1, 0, "could not read");
  110. }
  111. bytes_read += r;
  112. if (ofd != - 1)
  113. {
  114. p = buf;
  115. while (r)
  116. {
  117. w = write (ofd, p, r);
  118. if (w == - 1)
  119. {
  120. error (1, errno, "could not write");
  121. }
  122. p += w;
  123. r -= w;
  124. }
  125. }
  126. }
  127. if (gettimeofday (& tv_now, NULL) == - 1)
  128. {
  129. error (1, errno, "could not get time");
  130. }
  131. timersub (& tv_now, & tv_start, & tv_result);
  132. }
  133. while (tv_result.tv_sec < (signed) (time));
  134. bytes_sec = bytes_read / (tv_result.tv_sec + tv_result.tv_usec / 1000000.0);
  135. free (buf);
  136. return (bytes_sec * 8 / 1000.0);
  137. }
  138. /*====================================================================*
  139. *
  140. * int main (int argc, char const * argv []);
  141. *
  142. *--------------------------------------------------------------------*/
  143. int main (int argc, char const * argv [])
  144. {
  145. static char const * optv [] =
  146. {
  147. "cl:rs:t:qv",
  148. "",
  149. "Serial Line Rate Tester",
  150. "c\tconsume received data (do not output to stdout)",
  151. "l f\tserial port is (f) [" SERIAL_PORT "]",
  152. "r\tprint the receive data rate to stdout",
  153. "s n\tport speed [ 115200 ]",
  154. "t n\treceive for (n) seconds [ 10 ]",
  155. "q\tquiet mode",
  156. "v\tverbose mode",
  157. (char const *) (0)
  158. };
  159. struct termios termios;
  160. char * line = SERIAL_PORT;
  161. double rate = 0;
  162. speed_t speed = B115200;
  163. size_t time = 10;
  164. size_t chunk_size = 256;
  165. flag_t flags = 0;
  166. signed consume = 0;
  167. signed rflag = 0;
  168. signed fd;
  169. signed c;
  170. optind = 1;
  171. while (~ (c = getoptv (argc, argv, optv)))
  172. {
  173. switch ((char) (c))
  174. {
  175. case 'c':
  176. consume = 1;
  177. break;
  178. case 'r':
  179. rflag = 1;
  180. break;
  181. case 'l':
  182. line = optarg;
  183. break;
  184. case 's':
  185. if (baudrate (uintspec (optarg, 0, UINT_MAX), & speed))
  186. {
  187. error (1, 0, "could not set baud rate");
  188. }
  189. break;
  190. case 't':
  191. time = uintspec (optarg, 0, SIZE_MAX);
  192. break;
  193. case 'v':
  194. _setbits (flags, TTYRECV_VERBOSE);
  195. break;
  196. default:
  197. break;
  198. }
  199. }
  200. argc -= optind;
  201. argv += optind;
  202. fd = open (line, O_RDWR | O_NONBLOCK | O_NOCTTY);
  203. if (fd == - 1)
  204. {
  205. error (1, errno, "could not open %s", line);
  206. }
  207. if (fcntl(fd, F_SETFL, 0) == -1)
  208. {
  209. error (1, errno, "failed to set tty flags");
  210. }
  211. if (tcgetattr (fd, & termios) == - 1)
  212. {
  213. error (1, errno, "could not get tty attributes");
  214. }
  215. cfmakeraw (& termios);
  216. termios.c_cflag = CS8 | CREAD | CLOCAL;
  217. if (cfsetspeed (& termios, speed) == - 1)
  218. {
  219. error (1, errno, "could not set tty speed");
  220. }
  221. if (tcsetattr (fd, TCSANOW, & termios) == - 1)
  222. {
  223. error (1, errno, "could not set tty attributes");
  224. }
  225. if (! consume)
  226. {
  227. rate = ttyrecv (fd, STDOUT_FILENO, time, chunk_size, flags);
  228. }
  229. else
  230. {
  231. rate = ttyrecv (fd, - 1, time, chunk_size, flags);
  232. }
  233. if (rflag)
  234. {
  235. fprintf (stderr, "%.02f Kbps\n", rate);
  236. }
  237. close (fd);
  238. return (0);
  239. }