123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317 |
- /* Copyright (C) 1992, 1993, 1996, 1997, 1998 Free Software Foundation, Inc.
- This file is part of the GNU C Library.
- The GNU C Library is free software; you can redistribute it and/or
- modify it under the terms of the GNU Library General Public License as
- published by the Free Software Foundation; either version 2 of the
- License, or (at your option) any later version.
- The GNU C Library is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- Library General Public License for more details.
- You should have received a copy of the GNU Library General Public
- License along with the GNU C Library; see the file COPYING.LIB. If not,
- write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- Boston, MA 02111-1307, USA.
-
- About the only thing remaining here fromthe original Linux-8086 C library
- version by Robert de Bath <robert@mayday.compulink.co.uk>, is the general
- layout. All else has been recently stolen from GNU libc, since that was
- much more current.
- */
- #include <errno.h>
- #include <stddef.h>
- #include <sys/ioctl.h>
- #include <sys/types.h>
- #include <unistd.h>
- #include <termios.h>
- #ifdef L_isatty
- /* Return 1 if FD is a terminal, 0 if not. */
- int isatty(int fd)
- {
- struct termios term;
- return (tcgetattr (fd, &term) == 0);
- }
- #endif
- #ifdef L_tcdrain
- /* Wait for pending output to be written on FD. */
- int __libc_tcdrain (int fd)
- {
- /* With an argument of 1, TCSBRK waits for the output to drain. */
- return ioctl(fd, TCSBRK, 1);
- }
- weak_alias(__libc_tcdrain, tcdrain)
- #endif
- #ifdef L_tcflow
- /* Suspend or restart transmission on FD. */
- int tcflow ( int fd, int action)
- {
- return ioctl(fd, TCXONC, action);
- }
- #endif
- #ifdef L_tcflush
- /* Flush pending data on FD. */
- int tcflush ( int fd, int queue_selector)
- {
- return ioctl(fd, TCFLSH, queue_selector);
- }
- #endif
- #ifdef L_tcsendbreak
- /* Send zero bits on FD. */
- int tcsendbreak( int fd, int duration)
- {
- /* The break lasts 0.25 to 0.5 seconds if DURATION is zero,
- and an implementation-defined period if DURATION is nonzero.
- We define a positive DURATION to be number of milliseconds to break. */
- if (duration <= 0)
- return ioctl(fd, TCSBRK, 0);
- #ifdef TCSBRKP
- /* Probably Linux-specific: a positive third TCSBRKP ioctl argument is
- defined to be the number of 100ms units to break. */
- return ioctl(fd, TCSBRKP, (duration + 99) / 100);
- #else
- /* ioctl can't send a break of any other duration for us.
- This could be changed to use trickery (e.g. lower speed and
- send a '\0') to send the break, but for now just return an error. */
- __set_errno (EINVAL);
- return -1;
- #endif
- }
- #endif
- #ifdef L_tcsetpgrp
- /* Set the foreground process group ID of FD set PGRP_ID. */
- int tcsetpgrp ( int fd, pid_t pgrp_id)
- {
- return ioctl (fd, TIOCSPGRP, &pgrp_id);
- }
- #endif
- #ifdef L_tcgetpgrp
- /* Return the foreground process group ID of FD. */
- pid_t tcgetpgrp ( int fd)
- {
- int pgrp;
- if (ioctl (fd, TIOCGPGRP, &pgrp) < 0)
- return (pid_t) -1;
- return (pid_t) pgrp;
- }
- #endif
- /* This is a gross hack around a kernel bug. If the cfsetispeed functions is
- * called with the SPEED argument set to zero this means use the same speed as
- * for output. But we don't have independent input and output speeds and
- * therefore cannot record this.
- *
- * We use an unused bit in the `c_iflag' field to keep track of this use of
- * `cfsetispeed'. The value here must correspond to the one used in
- * `tcsetattr.c'. */
- #define IBAUD0 020000000000
- #ifdef L_cfgetospeed
- /* Return the output baud rate stored in *TERMIOS_P. */
- speed_t cfgetospeed ( const struct termios *termios_p)
- {
- return termios_p->c_cflag & (CBAUD | CBAUDEX);
- }
- #endif
- #ifdef L_cfgetispeed
- /* Return the input baud rate stored in *TERMIOS_P.
- * Although for Linux there is no difference between input and output
- * speed, the numerical 0 is a special case for the input baud rate. It
- * should set the input baud rate to the output baud rate. */
- speed_t cfgetispeed (const struct termios *termios_p)
- {
- return ((termios_p->c_iflag & IBAUD0)
- ? 0 : termios_p->c_cflag & (CBAUD | CBAUDEX));
- }
- #endif
- #ifdef L_cfsetospeed
- /* Set the output baud rate stored in *TERMIOS_P to SPEED. */
- int cfsetospeed (struct termios *termios_p, speed_t speed)
- {
- if ((speed & ~CBAUD) != 0
- && (speed < B57600 || speed > B460800))
- {
- __set_errno(EINVAL);
- return -1;
- }
- termios_p->c_cflag &= ~(CBAUD | CBAUDEX);
- termios_p->c_cflag |= speed;
- return 0;
- }
- #endif
- #ifdef L_cfsetispeed
- /* Set the input baud rate stored in *TERMIOS_P to SPEED.
- * Although for Linux there is no difference between input and output
- * speed, the numerical 0 is a special case for the input baud rate. It
- * should set the input baud rate to the output baud rate. */
- int cfsetispeed ( struct termios *termios_p, speed_t speed)
- {
- if ((speed & ~CBAUD) != 0
- && (speed < B57600 || speed > B460800))
- {
- __set_errno(EINVAL);
- return -1;
- }
- if (speed == 0)
- termios_p->c_iflag |= IBAUD0;
- else
- {
- termios_p->c_iflag &= ~IBAUD0;
- termios_p->c_cflag &= ~(CBAUD | CBAUDEX);
- termios_p->c_cflag |= speed;
- }
- return 0;
- }
- #endif
- #ifdef L_cfsetspeed
- struct speed_struct
- {
- speed_t value;
- speed_t internal;
- };
- static const struct speed_struct speeds[] =
- {
- #ifdef B0
- { 0, B0 },
- #endif
- #ifdef B50
- { 50, B50 },
- #endif
- #ifdef B75
- { 75, B75 },
- #endif
- #ifdef B110
- { 110, B110 },
- #endif
- #ifdef B134
- { 134, B134 },
- #endif
- #ifdef B150
- { 150, B150 },
- #endif
- #ifdef B200
- { 200, B200 },
- #endif
- #ifdef B300
- { 300, B300 },
- #endif
- #ifdef B600
- { 600, B600 },
- #endif
- #ifdef B1200
- { 1200, B1200 },
- #endif
- #ifdef B1200
- { 1200, B1200 },
- #endif
- #ifdef B1800
- { 1800, B1800 },
- #endif
- #ifdef B2400
- { 2400, B2400 },
- #endif
- #ifdef B4800
- { 4800, B4800 },
- #endif
- #ifdef B9600
- { 9600, B9600 },
- #endif
- #ifdef B19200
- { 19200, B19200 },
- #endif
- #ifdef B38400
- { 38400, B38400 },
- #endif
- #ifdef B57600
- { 57600, B57600 },
- #endif
- #ifdef B76800
- { 76800, B76800 },
- #endif
- #ifdef B115200
- { 115200, B115200 },
- #endif
- #ifdef B153600
- { 153600, B153600 },
- #endif
- #ifdef B230400
- { 230400, B230400 },
- #endif
- #ifdef B307200
- { 307200, B307200 },
- #endif
- #ifdef B460800
- { 460800, B460800 },
- #endif
- };
- /* Set both the input and output baud rates stored in *TERMIOS_P to SPEED. */
- int cfsetspeed (struct termios *termios_p, speed_t speed)
- {
- size_t cnt;
- for (cnt = 0; cnt < sizeof (speeds) / sizeof (speeds[0]); ++cnt)
- if (speed == speeds[cnt].internal)
- {
- cfsetispeed (termios_p, speed);
- cfsetospeed (termios_p, speed);
- return 0;
- }
- else if (speed == speeds[cnt].value)
- {
- cfsetispeed (termios_p, speeds[cnt].internal);
- cfsetospeed (termios_p, speeds[cnt].internal);
- return 0;
- }
- __set_errno (EINVAL);
- return -1;
- }
- #endif
- #ifdef L_cfmakeraw
- /* Copyright (C) 1992, 1996, 1997 Free Software Foundation, Inc.
- This file is part of the GNU C Library.
- */
- #include <termios.h>
- /* Set *T to indicate raw mode. */
- void
- cfmakeraw (struct termios *t)
- {
- t->c_iflag &= ~(IGNBRK|BRKINT|PARMRK|ISTRIP|INLCR|IGNCR|ICRNL|IXON);
- t->c_oflag &= ~OPOST;
- t->c_lflag &= ~(ECHO|ECHONL|ICANON|ISIG|IEXTEN);
- t->c_cflag &= ~(CSIZE|PARENB);
- t->c_cflag |= CS8;
- t->c_cc[VMIN] = 1; /* read returns when one char is available. */
- t->c_cc[VTIME] = 0;
- }
- #endif
|