termios.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327
  1. /* Copyright (C) 1992, 1993, 1996, 1997, 1998 Free Software Foundation, Inc.
  2. This file is part of the GNU C Library.
  3. The GNU C Library is free software; you can redistribute it and/or
  4. modify it under the terms of the GNU Library General Public License as
  5. published by the Free Software Foundation; either version 2 of the
  6. License, or (at your option) any later version.
  7. The GNU C Library is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Library General Public License for more details.
  11. You should have received a copy of the GNU Library General Public
  12. License along with the GNU C Library; see the file COPYING.LIB. If not,
  13. write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  14. Boston, MA 02111-1307, USA.
  15. About the only thing remaining here fromthe original Linux-8086 C library
  16. version by Robert de Bath <robert@mayday.compulink.co.uk>, is the general
  17. layout. All else has been recently stolen from GNU libc, since that was
  18. much more current.
  19. */
  20. #define tcgetattr __tcgetattr
  21. #include <errno.h>
  22. #include <stddef.h>
  23. #include <sys/ioctl.h>
  24. #include <sys/types.h>
  25. #include <unistd.h>
  26. #include <termios.h>
  27. #ifdef L_isatty
  28. /* Return 1 if FD is a terminal, 0 if not. */
  29. int attribute_hidden __isatty(int fd)
  30. {
  31. struct termios term;
  32. return (tcgetattr (fd, &term) == 0);
  33. }
  34. strong_alias(__isatty,isatty)
  35. #endif
  36. #ifdef L_tcdrain
  37. /* Wait for pending output to be written on FD. */
  38. int __libc_tcdrain (int fd)
  39. {
  40. /* With an argument of 1, TCSBRK waits for the output to drain. */
  41. return __ioctl(fd, TCSBRK, 1);
  42. }
  43. weak_alias(__libc_tcdrain, tcdrain)
  44. #endif
  45. #ifdef L_tcflow
  46. /* Suspend or restart transmission on FD. */
  47. int tcflow ( int fd, int action)
  48. {
  49. return __ioctl(fd, TCXONC, action);
  50. }
  51. #endif
  52. #ifdef L_tcflush
  53. /* Flush pending data on FD. */
  54. int tcflush ( int fd, int queue_selector)
  55. {
  56. return __ioctl(fd, TCFLSH, queue_selector);
  57. }
  58. #endif
  59. #ifdef L_tcsendbreak
  60. /* Send zero bits on FD. */
  61. int tcsendbreak( int fd, int duration)
  62. {
  63. /* The break lasts 0.25 to 0.5 seconds if DURATION is zero,
  64. and an implementation-defined period if DURATION is nonzero.
  65. We define a positive DURATION to be number of milliseconds to break. */
  66. if (duration <= 0)
  67. return __ioctl(fd, TCSBRK, 0);
  68. #ifdef TCSBRKP
  69. /* Probably Linux-specific: a positive third TCSBRKP ioctl argument is
  70. defined to be the number of 100ms units to break. */
  71. return __ioctl(fd, TCSBRKP, (duration + 99) / 100);
  72. #else
  73. /* ioctl can't send a break of any other duration for us.
  74. This could be changed to use trickery (e.g. lower speed and
  75. send a '\0') to send the break, but for now just return an error. */
  76. __set_errno (EINVAL);
  77. return -1;
  78. #endif
  79. }
  80. #endif
  81. #ifdef L_tcsetpgrp
  82. /* Set the foreground process group ID of FD set PGRP_ID. */
  83. int tcsetpgrp ( int fd, pid_t pgrp_id)
  84. {
  85. return __ioctl (fd, TIOCSPGRP, &pgrp_id);
  86. }
  87. #endif
  88. #ifdef L_tcgetpgrp
  89. /* Return the foreground process group ID of FD. */
  90. pid_t attribute_hidden __tcgetpgrp ( int fd)
  91. {
  92. int pgrp;
  93. if (__ioctl (fd, TIOCGPGRP, &pgrp) < 0)
  94. return (pid_t) -1;
  95. return (pid_t) pgrp;
  96. }
  97. strong_alias(__tcgetpgrp,tcgetpgrp)
  98. #endif
  99. /* This is a gross hack around a kernel bug. If the cfsetispeed functions is
  100. * called with the SPEED argument set to zero this means use the same speed as
  101. * for output. But we don't have independent input and output speeds and
  102. * therefore cannot record this.
  103. *
  104. * We use an unused bit in the `c_iflag' field to keep track of this use of
  105. * `cfsetispeed'. The value here must correspond to the one used in
  106. * `tcsetattr.c'. */
  107. #define IBAUD0 020000000000
  108. #ifdef L_cfgetospeed
  109. /* Return the output baud rate stored in *TERMIOS_P. */
  110. speed_t cfgetospeed ( const struct termios *termios_p)
  111. {
  112. return termios_p->c_cflag & (CBAUD | CBAUDEX);
  113. }
  114. #endif
  115. #ifdef L_cfgetispeed
  116. /* Return the input baud rate stored in *TERMIOS_P.
  117. * Although for Linux there is no difference between input and output
  118. * speed, the numerical 0 is a special case for the input baud rate. It
  119. * should set the input baud rate to the output baud rate. */
  120. speed_t cfgetispeed (const struct termios *termios_p)
  121. {
  122. return ((termios_p->c_iflag & IBAUD0)
  123. ? 0 : termios_p->c_cflag & (CBAUD | CBAUDEX));
  124. }
  125. #endif
  126. #ifdef L_cfsetospeed
  127. /* Set the output baud rate stored in *TERMIOS_P to SPEED. */
  128. int attribute_hidden __cfsetospeed (struct termios *termios_p, speed_t speed)
  129. {
  130. if ((speed & ~CBAUD) != 0
  131. && (speed < B57600 || speed > B460800))
  132. {
  133. __set_errno(EINVAL);
  134. return -1;
  135. }
  136. termios_p->c_cflag &= ~(CBAUD | CBAUDEX);
  137. termios_p->c_cflag |= speed;
  138. return 0;
  139. }
  140. strong_alias(__cfsetospeed,cfsetospeed)
  141. #endif
  142. #ifdef L_cfsetispeed
  143. /* Set the input baud rate stored in *TERMIOS_P to SPEED.
  144. * Although for Linux there is no difference between input and output
  145. * speed, the numerical 0 is a special case for the input baud rate. It
  146. * should set the input baud rate to the output baud rate. */
  147. int attribute_hidden __cfsetispeed ( struct termios *termios_p, speed_t speed)
  148. {
  149. if ((speed & ~CBAUD) != 0
  150. && (speed < B57600 || speed > B460800))
  151. {
  152. __set_errno(EINVAL);
  153. return -1;
  154. }
  155. if (speed == 0)
  156. termios_p->c_iflag |= IBAUD0;
  157. else
  158. {
  159. termios_p->c_iflag &= ~IBAUD0;
  160. termios_p->c_cflag &= ~(CBAUD | CBAUDEX);
  161. termios_p->c_cflag |= speed;
  162. }
  163. return 0;
  164. }
  165. strong_alias(__cfsetispeed,cfsetispeed)
  166. #endif
  167. #ifdef L_cfsetspeed
  168. extern int __cfsetospeed (struct termios *__termios_p, speed_t __speed) __THROW attribute_hidden;
  169. extern int __cfsetispeed (struct termios *__termios_p, speed_t __speed) __THROW attribute_hidden;
  170. struct speed_struct
  171. {
  172. speed_t value;
  173. speed_t internal;
  174. };
  175. static const struct speed_struct speeds[] =
  176. {
  177. #ifdef B0
  178. { 0, B0 },
  179. #endif
  180. #ifdef B50
  181. { 50, B50 },
  182. #endif
  183. #ifdef B75
  184. { 75, B75 },
  185. #endif
  186. #ifdef B110
  187. { 110, B110 },
  188. #endif
  189. #ifdef B134
  190. { 134, B134 },
  191. #endif
  192. #ifdef B150
  193. { 150, B150 },
  194. #endif
  195. #ifdef B200
  196. { 200, B200 },
  197. #endif
  198. #ifdef B300
  199. { 300, B300 },
  200. #endif
  201. #ifdef B600
  202. { 600, B600 },
  203. #endif
  204. #ifdef B1200
  205. { 1200, B1200 },
  206. #endif
  207. #ifdef B1200
  208. { 1200, B1200 },
  209. #endif
  210. #ifdef B1800
  211. { 1800, B1800 },
  212. #endif
  213. #ifdef B2400
  214. { 2400, B2400 },
  215. #endif
  216. #ifdef B4800
  217. { 4800, B4800 },
  218. #endif
  219. #ifdef B9600
  220. { 9600, B9600 },
  221. #endif
  222. #ifdef B19200
  223. { 19200, B19200 },
  224. #endif
  225. #ifdef B38400
  226. { 38400, B38400 },
  227. #endif
  228. #ifdef B57600
  229. { 57600, B57600 },
  230. #endif
  231. #ifdef B76800
  232. { 76800, B76800 },
  233. #endif
  234. #ifdef B115200
  235. { 115200, B115200 },
  236. #endif
  237. #ifdef B153600
  238. { 153600, B153600 },
  239. #endif
  240. #ifdef B230400
  241. { 230400, B230400 },
  242. #endif
  243. #ifdef B307200
  244. { 307200, B307200 },
  245. #endif
  246. #ifdef B460800
  247. { 460800, B460800 },
  248. #endif
  249. };
  250. /* Set both the input and output baud rates stored in *TERMIOS_P to SPEED. */
  251. int cfsetspeed (struct termios *termios_p, speed_t speed)
  252. {
  253. size_t cnt;
  254. for (cnt = 0; cnt < sizeof (speeds) / sizeof (speeds[0]); ++cnt)
  255. if (speed == speeds[cnt].internal)
  256. {
  257. __cfsetispeed (termios_p, speed);
  258. __cfsetospeed (termios_p, speed);
  259. return 0;
  260. }
  261. else if (speed == speeds[cnt].value)
  262. {
  263. __cfsetispeed (termios_p, speeds[cnt].internal);
  264. __cfsetospeed (termios_p, speeds[cnt].internal);
  265. return 0;
  266. }
  267. __set_errno (EINVAL);
  268. return -1;
  269. }
  270. #endif
  271. #ifdef L_cfmakeraw
  272. /* Copyright (C) 1992, 1996, 1997 Free Software Foundation, Inc.
  273. This file is part of the GNU C Library.
  274. */
  275. #include <termios.h>
  276. /* Set *T to indicate raw mode. */
  277. void
  278. cfmakeraw (struct termios *t)
  279. {
  280. t->c_iflag &= ~(IGNBRK|BRKINT|PARMRK|ISTRIP|INLCR|IGNCR|ICRNL|IXON);
  281. t->c_oflag &= ~OPOST;
  282. t->c_lflag &= ~(ECHO|ECHONL|ICANON|ISIG|IEXTEN);
  283. t->c_cflag &= ~(CSIZE|PARENB);
  284. t->c_cflag |= CS8;
  285. t->c_cc[VMIN] = 1; /* read returns when one char is available. */
  286. t->c_cc[VTIME] = 0;
  287. }
  288. #endif