mirror of https://github.com/python/cpython.git
1033 lines
22 KiB
C
1033 lines
22 KiB
C
/* termiosmodule.c -- POSIX terminal I/O module implementation. */
|
|
|
|
#include "Python.h"
|
|
|
|
/* Apparently, on SGI, termios.h won't define CTRL if _XOPEN_SOURCE
|
|
is defined, so we define it here. */
|
|
#if defined(__sgi)
|
|
#define CTRL(c) ((c)&037)
|
|
#endif
|
|
|
|
#if defined(__sun)
|
|
/* We could do better. Check issue-32660 */
|
|
#include <sys/filio.h>
|
|
#include <sys/sockio.h>
|
|
#endif
|
|
|
|
#include <termios.h>
|
|
#include <sys/ioctl.h>
|
|
|
|
/* HP-UX requires that this be included to pick up MDCD, MCTS, MDSR,
|
|
* MDTR, MRI, and MRTS (apparently used internally by some things
|
|
* defined as macros; these are not used here directly).
|
|
*/
|
|
#ifdef HAVE_SYS_MODEM_H
|
|
#include <sys/modem.h>
|
|
#endif
|
|
/* HP-UX requires that this be included to pick up TIOCGPGRP and friends */
|
|
#ifdef HAVE_SYS_BSDTTY_H
|
|
#include <sys/bsdtty.h>
|
|
#endif
|
|
|
|
PyDoc_STRVAR(termios__doc__,
|
|
"This module provides an interface to the Posix calls for tty I/O control.\n\
|
|
For a complete description of these calls, see the Posix or Unix manual\n\
|
|
pages. It is only available for those Unix versions that support Posix\n\
|
|
termios style tty I/O control.\n\
|
|
\n\
|
|
All functions in this module take a file descriptor fd as their first\n\
|
|
argument. This can be an integer file descriptor, such as returned by\n\
|
|
sys.stdin.fileno(), or a file object, such as sys.stdin itself.");
|
|
|
|
typedef struct {
|
|
PyObject *TermiosError;
|
|
} termiosmodulestate;
|
|
#define modulestate(o) ((termiosmodulestate *)PyModule_GetState(o))
|
|
#define modulestate_global modulestate(PyState_FindModule(&termiosmodule))
|
|
|
|
static int fdconv(PyObject* obj, void* p)
|
|
{
|
|
int fd;
|
|
|
|
fd = PyObject_AsFileDescriptor(obj);
|
|
if (fd >= 0) {
|
|
*(int*)p = fd;
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static struct PyModuleDef termiosmodule;
|
|
|
|
PyDoc_STRVAR(termios_tcgetattr__doc__,
|
|
"tcgetattr(fd) -> list_of_attrs\n\
|
|
\n\
|
|
Get the tty attributes for file descriptor fd, as follows:\n\
|
|
[iflag, oflag, cflag, lflag, ispeed, ospeed, cc] where cc is a list\n\
|
|
of the tty special characters (each a string of length 1, except the items\n\
|
|
with indices VMIN and VTIME, which are integers when these fields are\n\
|
|
defined). The interpretation of the flags and the speeds as well as the\n\
|
|
indexing in the cc array must be done using the symbolic constants defined\n\
|
|
in this module.");
|
|
|
|
static PyObject *
|
|
termios_tcgetattr(PyObject *self, PyObject *args)
|
|
{
|
|
int fd;
|
|
struct termios mode;
|
|
PyObject *cc;
|
|
speed_t ispeed, ospeed;
|
|
PyObject *v;
|
|
int i;
|
|
char ch;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&:tcgetattr",
|
|
fdconv, (void*)&fd))
|
|
return NULL;
|
|
|
|
if (tcgetattr(fd, &mode) == -1)
|
|
return PyErr_SetFromErrno(modulestate_global->TermiosError);
|
|
|
|
ispeed = cfgetispeed(&mode);
|
|
ospeed = cfgetospeed(&mode);
|
|
|
|
cc = PyList_New(NCCS);
|
|
if (cc == NULL)
|
|
return NULL;
|
|
for (i = 0; i < NCCS; i++) {
|
|
ch = (char)mode.c_cc[i];
|
|
v = PyBytes_FromStringAndSize(&ch, 1);
|
|
if (v == NULL)
|
|
goto err;
|
|
PyList_SetItem(cc, i, v);
|
|
}
|
|
|
|
/* Convert the MIN and TIME slots to integer. On some systems, the
|
|
MIN and TIME slots are the same as the EOF and EOL slots. So we
|
|
only do this in noncanonical input mode. */
|
|
if ((mode.c_lflag & ICANON) == 0) {
|
|
v = PyLong_FromLong((long)mode.c_cc[VMIN]);
|
|
if (v == NULL)
|
|
goto err;
|
|
PyList_SetItem(cc, VMIN, v);
|
|
v = PyLong_FromLong((long)mode.c_cc[VTIME]);
|
|
if (v == NULL)
|
|
goto err;
|
|
PyList_SetItem(cc, VTIME, v);
|
|
}
|
|
|
|
if (!(v = PyList_New(7)))
|
|
goto err;
|
|
|
|
PyList_SetItem(v, 0, PyLong_FromLong((long)mode.c_iflag));
|
|
PyList_SetItem(v, 1, PyLong_FromLong((long)mode.c_oflag));
|
|
PyList_SetItem(v, 2, PyLong_FromLong((long)mode.c_cflag));
|
|
PyList_SetItem(v, 3, PyLong_FromLong((long)mode.c_lflag));
|
|
PyList_SetItem(v, 4, PyLong_FromLong((long)ispeed));
|
|
PyList_SetItem(v, 5, PyLong_FromLong((long)ospeed));
|
|
if (PyErr_Occurred()) {
|
|
Py_DECREF(v);
|
|
goto err;
|
|
}
|
|
PyList_SetItem(v, 6, cc);
|
|
return v;
|
|
err:
|
|
Py_DECREF(cc);
|
|
return NULL;
|
|
}
|
|
|
|
PyDoc_STRVAR(termios_tcsetattr__doc__,
|
|
"tcsetattr(fd, when, attributes) -> None\n\
|
|
\n\
|
|
Set the tty attributes for file descriptor fd.\n\
|
|
The attributes to be set are taken from the attributes argument, which\n\
|
|
is a list like the one returned by tcgetattr(). The when argument\n\
|
|
determines when the attributes are changed: termios.TCSANOW to\n\
|
|
change immediately, termios.TCSADRAIN to change after transmitting all\n\
|
|
queued output, or termios.TCSAFLUSH to change after transmitting all\n\
|
|
queued output and discarding all queued input. ");
|
|
|
|
static PyObject *
|
|
termios_tcsetattr(PyObject *self, PyObject *args)
|
|
{
|
|
int fd, when;
|
|
struct termios mode;
|
|
speed_t ispeed, ospeed;
|
|
PyObject *term, *cc, *v;
|
|
int i;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&iO:tcsetattr",
|
|
fdconv, &fd, &when, &term))
|
|
return NULL;
|
|
if (!PyList_Check(term) || PyList_Size(term) != 7) {
|
|
PyErr_SetString(PyExc_TypeError,
|
|
"tcsetattr, arg 3: must be 7 element list");
|
|
return NULL;
|
|
}
|
|
|
|
/* Get the old mode, in case there are any hidden fields... */
|
|
termiosmodulestate *state = modulestate_global;
|
|
if (tcgetattr(fd, &mode) == -1)
|
|
return PyErr_SetFromErrno(state->TermiosError);
|
|
mode.c_iflag = (tcflag_t) PyLong_AsLong(PyList_GetItem(term, 0));
|
|
mode.c_oflag = (tcflag_t) PyLong_AsLong(PyList_GetItem(term, 1));
|
|
mode.c_cflag = (tcflag_t) PyLong_AsLong(PyList_GetItem(term, 2));
|
|
mode.c_lflag = (tcflag_t) PyLong_AsLong(PyList_GetItem(term, 3));
|
|
ispeed = (speed_t) PyLong_AsLong(PyList_GetItem(term, 4));
|
|
ospeed = (speed_t) PyLong_AsLong(PyList_GetItem(term, 5));
|
|
cc = PyList_GetItem(term, 6);
|
|
if (PyErr_Occurred())
|
|
return NULL;
|
|
|
|
if (!PyList_Check(cc) || PyList_Size(cc) != NCCS) {
|
|
PyErr_Format(PyExc_TypeError,
|
|
"tcsetattr: attributes[6] must be %d element list",
|
|
NCCS);
|
|
return NULL;
|
|
}
|
|
|
|
for (i = 0; i < NCCS; i++) {
|
|
v = PyList_GetItem(cc, i);
|
|
|
|
if (PyBytes_Check(v) && PyBytes_Size(v) == 1)
|
|
mode.c_cc[i] = (cc_t) * PyBytes_AsString(v);
|
|
else if (PyLong_Check(v))
|
|
mode.c_cc[i] = (cc_t) PyLong_AsLong(v);
|
|
else {
|
|
PyErr_SetString(PyExc_TypeError,
|
|
"tcsetattr: elements of attributes must be characters or integers");
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
if (cfsetispeed(&mode, (speed_t) ispeed) == -1)
|
|
return PyErr_SetFromErrno(state->TermiosError);
|
|
if (cfsetospeed(&mode, (speed_t) ospeed) == -1)
|
|
return PyErr_SetFromErrno(state->TermiosError);
|
|
if (tcsetattr(fd, when, &mode) == -1)
|
|
return PyErr_SetFromErrno(state->TermiosError);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyDoc_STRVAR(termios_tcsendbreak__doc__,
|
|
"tcsendbreak(fd, duration) -> None\n\
|
|
\n\
|
|
Send a break on file descriptor fd.\n\
|
|
A zero duration sends a break for 0.25-0.5 seconds; a nonzero duration\n\
|
|
has a system dependent meaning.");
|
|
|
|
static PyObject *
|
|
termios_tcsendbreak(PyObject *self, PyObject *args)
|
|
{
|
|
int fd, duration;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&i:tcsendbreak",
|
|
fdconv, &fd, &duration))
|
|
return NULL;
|
|
if (tcsendbreak(fd, duration) == -1)
|
|
return PyErr_SetFromErrno(modulestate_global->TermiosError);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyDoc_STRVAR(termios_tcdrain__doc__,
|
|
"tcdrain(fd) -> None\n\
|
|
\n\
|
|
Wait until all output written to file descriptor fd has been transmitted.");
|
|
|
|
static PyObject *
|
|
termios_tcdrain(PyObject *self, PyObject *args)
|
|
{
|
|
int fd;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&:tcdrain",
|
|
fdconv, &fd))
|
|
return NULL;
|
|
if (tcdrain(fd) == -1)
|
|
return PyErr_SetFromErrno(modulestate_global->TermiosError);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyDoc_STRVAR(termios_tcflush__doc__,
|
|
"tcflush(fd, queue) -> None\n\
|
|
\n\
|
|
Discard queued data on file descriptor fd.\n\
|
|
The queue selector specifies which queue: termios.TCIFLUSH for the input\n\
|
|
queue, termios.TCOFLUSH for the output queue, or termios.TCIOFLUSH for\n\
|
|
both queues. ");
|
|
|
|
static PyObject *
|
|
termios_tcflush(PyObject *self, PyObject *args)
|
|
{
|
|
int fd, queue;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&i:tcflush",
|
|
fdconv, &fd, &queue))
|
|
return NULL;
|
|
if (tcflush(fd, queue) == -1)
|
|
return PyErr_SetFromErrno(modulestate_global->TermiosError);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
PyDoc_STRVAR(termios_tcflow__doc__,
|
|
"tcflow(fd, action) -> None\n\
|
|
\n\
|
|
Suspend or resume input or output on file descriptor fd.\n\
|
|
The action argument can be termios.TCOOFF to suspend output,\n\
|
|
termios.TCOON to restart output, termios.TCIOFF to suspend input,\n\
|
|
or termios.TCION to restart input.");
|
|
|
|
static PyObject *
|
|
termios_tcflow(PyObject *self, PyObject *args)
|
|
{
|
|
int fd, action;
|
|
|
|
if (!PyArg_ParseTuple(args, "O&i:tcflow",
|
|
fdconv, &fd, &action))
|
|
return NULL;
|
|
if (tcflow(fd, action) == -1)
|
|
return PyErr_SetFromErrno(modulestate_global->TermiosError);
|
|
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
static PyMethodDef termios_methods[] =
|
|
{
|
|
{"tcgetattr", termios_tcgetattr,
|
|
METH_VARARGS, termios_tcgetattr__doc__},
|
|
{"tcsetattr", termios_tcsetattr,
|
|
METH_VARARGS, termios_tcsetattr__doc__},
|
|
{"tcsendbreak", termios_tcsendbreak,
|
|
METH_VARARGS, termios_tcsendbreak__doc__},
|
|
{"tcdrain", termios_tcdrain,
|
|
METH_VARARGS, termios_tcdrain__doc__},
|
|
{"tcflush", termios_tcflush,
|
|
METH_VARARGS, termios_tcflush__doc__},
|
|
{"tcflow", termios_tcflow,
|
|
METH_VARARGS, termios_tcflow__doc__},
|
|
{NULL, NULL}
|
|
};
|
|
|
|
|
|
#if defined(VSWTCH) && !defined(VSWTC)
|
|
#define VSWTC VSWTCH
|
|
#endif
|
|
|
|
#if defined(VSWTC) && !defined(VSWTCH)
|
|
#define VSWTCH VSWTC
|
|
#endif
|
|
|
|
static struct constant {
|
|
char *name;
|
|
long value;
|
|
} termios_constants[] = {
|
|
/* cfgetospeed(), cfsetospeed() constants */
|
|
{"B0", B0},
|
|
{"B50", B50},
|
|
{"B75", B75},
|
|
{"B110", B110},
|
|
{"B134", B134},
|
|
{"B150", B150},
|
|
{"B200", B200},
|
|
{"B300", B300},
|
|
{"B600", B600},
|
|
{"B1200", B1200},
|
|
{"B1800", B1800},
|
|
{"B2400", B2400},
|
|
{"B4800", B4800},
|
|
{"B9600", B9600},
|
|
{"B19200", B19200},
|
|
{"B38400", B38400},
|
|
#ifdef B57600
|
|
{"B57600", B57600},
|
|
#endif
|
|
#ifdef B115200
|
|
{"B115200", B115200},
|
|
#endif
|
|
#ifdef B230400
|
|
{"B230400", B230400},
|
|
#endif
|
|
#ifdef B460800
|
|
{"B460800", B460800},
|
|
#endif
|
|
#ifdef B500000
|
|
{"B500000", B500000},
|
|
#endif
|
|
#ifdef B576000
|
|
{"B576000", B576000},
|
|
#endif
|
|
#ifdef B921600
|
|
{"B921600", B921600},
|
|
#endif
|
|
#ifdef B1000000
|
|
{"B1000000", B1000000},
|
|
#endif
|
|
#ifdef B1152000
|
|
{"B1152000", B1152000},
|
|
#endif
|
|
#ifdef B1500000
|
|
{"B1500000", B1500000},
|
|
#endif
|
|
#ifdef B2000000
|
|
{"B2000000", B2000000},
|
|
#endif
|
|
#ifdef B2500000
|
|
{"B2500000", B2500000},
|
|
#endif
|
|
#ifdef B3000000
|
|
{"B3000000", B3000000},
|
|
#endif
|
|
#ifdef B3500000
|
|
{"B3500000", B3500000},
|
|
#endif
|
|
#ifdef B4000000
|
|
{"B4000000", B4000000},
|
|
#endif
|
|
|
|
#ifdef CBAUDEX
|
|
{"CBAUDEX", CBAUDEX},
|
|
#endif
|
|
|
|
/* tcsetattr() constants */
|
|
{"TCSANOW", TCSANOW},
|
|
{"TCSADRAIN", TCSADRAIN},
|
|
{"TCSAFLUSH", TCSAFLUSH},
|
|
#ifdef TCSASOFT
|
|
{"TCSASOFT", TCSASOFT},
|
|
#endif
|
|
|
|
/* tcflush() constants */
|
|
{"TCIFLUSH", TCIFLUSH},
|
|
{"TCOFLUSH", TCOFLUSH},
|
|
{"TCIOFLUSH", TCIOFLUSH},
|
|
|
|
/* tcflow() constants */
|
|
{"TCOOFF", TCOOFF},
|
|
{"TCOON", TCOON},
|
|
{"TCIOFF", TCIOFF},
|
|
{"TCION", TCION},
|
|
|
|
/* struct termios.c_iflag constants */
|
|
{"IGNBRK", IGNBRK},
|
|
{"BRKINT", BRKINT},
|
|
{"IGNPAR", IGNPAR},
|
|
{"PARMRK", PARMRK},
|
|
{"INPCK", INPCK},
|
|
{"ISTRIP", ISTRIP},
|
|
{"INLCR", INLCR},
|
|
{"IGNCR", IGNCR},
|
|
{"ICRNL", ICRNL},
|
|
#ifdef IUCLC
|
|
{"IUCLC", IUCLC},
|
|
#endif
|
|
{"IXON", IXON},
|
|
{"IXANY", IXANY},
|
|
{"IXOFF", IXOFF},
|
|
#ifdef IMAXBEL
|
|
{"IMAXBEL", IMAXBEL},
|
|
#endif
|
|
|
|
/* struct termios.c_oflag constants */
|
|
{"OPOST", OPOST},
|
|
#ifdef OLCUC
|
|
{"OLCUC", OLCUC},
|
|
#endif
|
|
#ifdef ONLCR
|
|
{"ONLCR", ONLCR},
|
|
#endif
|
|
#ifdef OCRNL
|
|
{"OCRNL", OCRNL},
|
|
#endif
|
|
#ifdef ONOCR
|
|
{"ONOCR", ONOCR},
|
|
#endif
|
|
#ifdef ONLRET
|
|
{"ONLRET", ONLRET},
|
|
#endif
|
|
#ifdef OFILL
|
|
{"OFILL", OFILL},
|
|
#endif
|
|
#ifdef OFDEL
|
|
{"OFDEL", OFDEL},
|
|
#endif
|
|
#ifdef NLDLY
|
|
{"NLDLY", NLDLY},
|
|
#endif
|
|
#ifdef CRDLY
|
|
{"CRDLY", CRDLY},
|
|
#endif
|
|
#ifdef TABDLY
|
|
{"TABDLY", TABDLY},
|
|
#endif
|
|
#ifdef BSDLY
|
|
{"BSDLY", BSDLY},
|
|
#endif
|
|
#ifdef VTDLY
|
|
{"VTDLY", VTDLY},
|
|
#endif
|
|
#ifdef FFDLY
|
|
{"FFDLY", FFDLY},
|
|
#endif
|
|
|
|
/* struct termios.c_oflag-related values (delay mask) */
|
|
#ifdef NL0
|
|
{"NL0", NL0},
|
|
#endif
|
|
#ifdef NL1
|
|
{"NL1", NL1},
|
|
#endif
|
|
#ifdef CR0
|
|
{"CR0", CR0},
|
|
#endif
|
|
#ifdef CR1
|
|
{"CR1", CR1},
|
|
#endif
|
|
#ifdef CR2
|
|
{"CR2", CR2},
|
|
#endif
|
|
#ifdef CR3
|
|
{"CR3", CR3},
|
|
#endif
|
|
#ifdef TAB0
|
|
{"TAB0", TAB0},
|
|
#endif
|
|
#ifdef TAB1
|
|
{"TAB1", TAB1},
|
|
#endif
|
|
#ifdef TAB2
|
|
{"TAB2", TAB2},
|
|
#endif
|
|
#ifdef TAB3
|
|
{"TAB3", TAB3},
|
|
#endif
|
|
#ifdef XTABS
|
|
{"XTABS", XTABS},
|
|
#endif
|
|
#ifdef BS0
|
|
{"BS0", BS0},
|
|
#endif
|
|
#ifdef BS1
|
|
{"BS1", BS1},
|
|
#endif
|
|
#ifdef VT0
|
|
{"VT0", VT0},
|
|
#endif
|
|
#ifdef VT1
|
|
{"VT1", VT1},
|
|
#endif
|
|
#ifdef FF0
|
|
{"FF0", FF0},
|
|
#endif
|
|
#ifdef FF1
|
|
{"FF1", FF1},
|
|
#endif
|
|
|
|
/* struct termios.c_cflag constants */
|
|
{"CSIZE", CSIZE},
|
|
{"CSTOPB", CSTOPB},
|
|
{"CREAD", CREAD},
|
|
{"PARENB", PARENB},
|
|
{"PARODD", PARODD},
|
|
{"HUPCL", HUPCL},
|
|
{"CLOCAL", CLOCAL},
|
|
#ifdef CIBAUD
|
|
{"CIBAUD", CIBAUD},
|
|
#endif
|
|
#ifdef CRTSCTS
|
|
{"CRTSCTS", (long)CRTSCTS},
|
|
#endif
|
|
|
|
/* struct termios.c_cflag-related values (character size) */
|
|
{"CS5", CS5},
|
|
{"CS6", CS6},
|
|
{"CS7", CS7},
|
|
{"CS8", CS8},
|
|
|
|
/* struct termios.c_lflag constants */
|
|
{"ISIG", ISIG},
|
|
{"ICANON", ICANON},
|
|
#ifdef XCASE
|
|
{"XCASE", XCASE},
|
|
#endif
|
|
{"ECHO", ECHO},
|
|
{"ECHOE", ECHOE},
|
|
{"ECHOK", ECHOK},
|
|
{"ECHONL", ECHONL},
|
|
#ifdef ECHOCTL
|
|
{"ECHOCTL", ECHOCTL},
|
|
#endif
|
|
#ifdef ECHOPRT
|
|
{"ECHOPRT", ECHOPRT},
|
|
#endif
|
|
#ifdef ECHOKE
|
|
{"ECHOKE", ECHOKE},
|
|
#endif
|
|
#ifdef FLUSHO
|
|
{"FLUSHO", FLUSHO},
|
|
#endif
|
|
{"NOFLSH", NOFLSH},
|
|
{"TOSTOP", TOSTOP},
|
|
#ifdef PENDIN
|
|
{"PENDIN", PENDIN},
|
|
#endif
|
|
{"IEXTEN", IEXTEN},
|
|
|
|
/* indexes into the control chars array returned by tcgetattr() */
|
|
{"VINTR", VINTR},
|
|
{"VQUIT", VQUIT},
|
|
{"VERASE", VERASE},
|
|
{"VKILL", VKILL},
|
|
{"VEOF", VEOF},
|
|
{"VTIME", VTIME},
|
|
{"VMIN", VMIN},
|
|
#ifdef VSWTC
|
|
/* The #defines above ensure that if either is defined, both are,
|
|
* but both may be omitted by the system headers. ;-( */
|
|
{"VSWTC", VSWTC},
|
|
{"VSWTCH", VSWTCH},
|
|
#endif
|
|
{"VSTART", VSTART},
|
|
{"VSTOP", VSTOP},
|
|
{"VSUSP", VSUSP},
|
|
{"VEOL", VEOL},
|
|
#ifdef VREPRINT
|
|
{"VREPRINT", VREPRINT},
|
|
#endif
|
|
#ifdef VDISCARD
|
|
{"VDISCARD", VDISCARD},
|
|
#endif
|
|
#ifdef VWERASE
|
|
{"VWERASE", VWERASE},
|
|
#endif
|
|
#ifdef VLNEXT
|
|
{"VLNEXT", VLNEXT},
|
|
#endif
|
|
#ifdef VEOL2
|
|
{"VEOL2", VEOL2},
|
|
#endif
|
|
|
|
|
|
#ifdef B460800
|
|
{"B460800", B460800},
|
|
#endif
|
|
#ifdef B500000
|
|
{"B500000", B500000},
|
|
#endif
|
|
#ifdef B576000
|
|
{ "B576000", B576000},
|
|
#endif
|
|
#ifdef B921600
|
|
{ "B921600", B921600},
|
|
#endif
|
|
#ifdef B1000000
|
|
{ "B1000000", B1000000},
|
|
#endif
|
|
#ifdef B1152000
|
|
{ "B1152000", B1152000},
|
|
#endif
|
|
#ifdef B1500000
|
|
{ "B1500000", B1500000},
|
|
#endif
|
|
#ifdef B2000000
|
|
{ "B2000000", B2000000},
|
|
#endif
|
|
#ifdef B2500000
|
|
{ "B2500000", B2500000},
|
|
#endif
|
|
#ifdef B3000000
|
|
{ "B3000000", B3000000},
|
|
#endif
|
|
#ifdef B3500000
|
|
{ "B3500000", B3500000},
|
|
#endif
|
|
#ifdef B4000000
|
|
{ "B4000000", B4000000},
|
|
#endif
|
|
#ifdef CBAUD
|
|
{"CBAUD", CBAUD},
|
|
#endif
|
|
#ifdef CDEL
|
|
{"CDEL", CDEL},
|
|
#endif
|
|
#ifdef CDSUSP
|
|
{"CDSUSP", CDSUSP},
|
|
#endif
|
|
#ifdef CEOF
|
|
{"CEOF", CEOF},
|
|
#endif
|
|
#ifdef CEOL
|
|
{"CEOL", CEOL},
|
|
#endif
|
|
#ifdef CEOL2
|
|
{"CEOL2", CEOL2},
|
|
#endif
|
|
#ifdef CEOT
|
|
{"CEOT", CEOT},
|
|
#endif
|
|
#ifdef CERASE
|
|
{"CERASE", CERASE},
|
|
#endif
|
|
#ifdef CESC
|
|
{"CESC", CESC},
|
|
#endif
|
|
#ifdef CFLUSH
|
|
{"CFLUSH", CFLUSH},
|
|
#endif
|
|
#ifdef CINTR
|
|
{"CINTR", CINTR},
|
|
#endif
|
|
#ifdef CKILL
|
|
{"CKILL", CKILL},
|
|
#endif
|
|
#ifdef CLNEXT
|
|
{"CLNEXT", CLNEXT},
|
|
#endif
|
|
#ifdef CNUL
|
|
{"CNUL", CNUL},
|
|
#endif
|
|
#ifdef COMMON
|
|
{"COMMON", COMMON},
|
|
#endif
|
|
#ifdef CQUIT
|
|
{"CQUIT", CQUIT},
|
|
#endif
|
|
#ifdef CRPRNT
|
|
{"CRPRNT", CRPRNT},
|
|
#endif
|
|
#ifdef CSTART
|
|
{"CSTART", CSTART},
|
|
#endif
|
|
#ifdef CSTOP
|
|
{"CSTOP", CSTOP},
|
|
#endif
|
|
#ifdef CSUSP
|
|
{"CSUSP", CSUSP},
|
|
#endif
|
|
#ifdef CSWTCH
|
|
{"CSWTCH", CSWTCH},
|
|
#endif
|
|
#ifdef CWERASE
|
|
{"CWERASE", CWERASE},
|
|
#endif
|
|
#ifdef EXTA
|
|
{"EXTA", EXTA},
|
|
#endif
|
|
#ifdef EXTB
|
|
{"EXTB", EXTB},
|
|
#endif
|
|
#ifdef FIOASYNC
|
|
{"FIOASYNC", FIOASYNC},
|
|
#endif
|
|
#ifdef FIOCLEX
|
|
{"FIOCLEX", FIOCLEX},
|
|
#endif
|
|
#ifdef FIONBIO
|
|
{"FIONBIO", FIONBIO},
|
|
#endif
|
|
#ifdef FIONCLEX
|
|
{"FIONCLEX", FIONCLEX},
|
|
#endif
|
|
#ifdef FIONREAD
|
|
{"FIONREAD", FIONREAD},
|
|
#endif
|
|
#ifdef IBSHIFT
|
|
{"IBSHIFT", IBSHIFT},
|
|
#endif
|
|
#ifdef INIT_C_CC
|
|
{"INIT_C_CC", INIT_C_CC},
|
|
#endif
|
|
#ifdef IOCSIZE_MASK
|
|
{"IOCSIZE_MASK", IOCSIZE_MASK},
|
|
#endif
|
|
#ifdef IOCSIZE_SHIFT
|
|
{"IOCSIZE_SHIFT", IOCSIZE_SHIFT},
|
|
#endif
|
|
#ifdef NCC
|
|
{"NCC", NCC},
|
|
#endif
|
|
#ifdef NCCS
|
|
{"NCCS", NCCS},
|
|
#endif
|
|
#ifdef NSWTCH
|
|
{"NSWTCH", NSWTCH},
|
|
#endif
|
|
#ifdef N_MOUSE
|
|
{"N_MOUSE", N_MOUSE},
|
|
#endif
|
|
#ifdef N_PPP
|
|
{"N_PPP", N_PPP},
|
|
#endif
|
|
#ifdef N_SLIP
|
|
{"N_SLIP", N_SLIP},
|
|
#endif
|
|
#ifdef N_STRIP
|
|
{"N_STRIP", N_STRIP},
|
|
#endif
|
|
#ifdef N_TTY
|
|
{"N_TTY", N_TTY},
|
|
#endif
|
|
#ifdef TCFLSH
|
|
{"TCFLSH", TCFLSH},
|
|
#endif
|
|
#ifdef TCGETA
|
|
{"TCGETA", TCGETA},
|
|
#endif
|
|
#ifdef TCGETS
|
|
{"TCGETS", TCGETS},
|
|
#endif
|
|
#ifdef TCSBRK
|
|
{"TCSBRK", TCSBRK},
|
|
#endif
|
|
#ifdef TCSBRKP
|
|
{"TCSBRKP", TCSBRKP},
|
|
#endif
|
|
#ifdef TCSETA
|
|
{"TCSETA", TCSETA},
|
|
#endif
|
|
#ifdef TCSETAF
|
|
{"TCSETAF", TCSETAF},
|
|
#endif
|
|
#ifdef TCSETAW
|
|
{"TCSETAW", TCSETAW},
|
|
#endif
|
|
#ifdef TCSETS
|
|
{"TCSETS", TCSETS},
|
|
#endif
|
|
#ifdef TCSETSF
|
|
{"TCSETSF", TCSETSF},
|
|
#endif
|
|
#ifdef TCSETSW
|
|
{"TCSETSW", TCSETSW},
|
|
#endif
|
|
#ifdef TCXONC
|
|
{"TCXONC", TCXONC},
|
|
#endif
|
|
#ifdef TIOCCONS
|
|
{"TIOCCONS", TIOCCONS},
|
|
#endif
|
|
#ifdef TIOCEXCL
|
|
{"TIOCEXCL", TIOCEXCL},
|
|
#endif
|
|
#ifdef TIOCGETD
|
|
{"TIOCGETD", TIOCGETD},
|
|
#endif
|
|
#ifdef TIOCGICOUNT
|
|
{"TIOCGICOUNT", TIOCGICOUNT},
|
|
#endif
|
|
#ifdef TIOCGLCKTRMIOS
|
|
{"TIOCGLCKTRMIOS", TIOCGLCKTRMIOS},
|
|
#endif
|
|
#ifdef TIOCGPGRP
|
|
{"TIOCGPGRP", TIOCGPGRP},
|
|
#endif
|
|
#ifdef TIOCGSERIAL
|
|
{"TIOCGSERIAL", TIOCGSERIAL},
|
|
#endif
|
|
#ifdef TIOCGSOFTCAR
|
|
{"TIOCGSOFTCAR", TIOCGSOFTCAR},
|
|
#endif
|
|
#ifdef TIOCGWINSZ
|
|
{"TIOCGWINSZ", TIOCGWINSZ},
|
|
#endif
|
|
#ifdef TIOCINQ
|
|
{"TIOCINQ", TIOCINQ},
|
|
#endif
|
|
#ifdef TIOCLINUX
|
|
{"TIOCLINUX", TIOCLINUX},
|
|
#endif
|
|
#ifdef TIOCMBIC
|
|
{"TIOCMBIC", TIOCMBIC},
|
|
#endif
|
|
#ifdef TIOCMBIS
|
|
{"TIOCMBIS", TIOCMBIS},
|
|
#endif
|
|
#ifdef TIOCMGET
|
|
{"TIOCMGET", TIOCMGET},
|
|
#endif
|
|
#ifdef TIOCMIWAIT
|
|
{"TIOCMIWAIT", TIOCMIWAIT},
|
|
#endif
|
|
#ifdef TIOCMSET
|
|
{"TIOCMSET", TIOCMSET},
|
|
#endif
|
|
#ifdef TIOCM_CAR
|
|
{"TIOCM_CAR", TIOCM_CAR},
|
|
#endif
|
|
#ifdef TIOCM_CD
|
|
{"TIOCM_CD", TIOCM_CD},
|
|
#endif
|
|
#ifdef TIOCM_CTS
|
|
{"TIOCM_CTS", TIOCM_CTS},
|
|
#endif
|
|
#ifdef TIOCM_DSR
|
|
{"TIOCM_DSR", TIOCM_DSR},
|
|
#endif
|
|
#ifdef TIOCM_DTR
|
|
{"TIOCM_DTR", TIOCM_DTR},
|
|
#endif
|
|
#ifdef TIOCM_LE
|
|
{"TIOCM_LE", TIOCM_LE},
|
|
#endif
|
|
#ifdef TIOCM_RI
|
|
{"TIOCM_RI", TIOCM_RI},
|
|
#endif
|
|
#ifdef TIOCM_RNG
|
|
{"TIOCM_RNG", TIOCM_RNG},
|
|
#endif
|
|
#ifdef TIOCM_RTS
|
|
{"TIOCM_RTS", TIOCM_RTS},
|
|
#endif
|
|
#ifdef TIOCM_SR
|
|
{"TIOCM_SR", TIOCM_SR},
|
|
#endif
|
|
#ifdef TIOCM_ST
|
|
{"TIOCM_ST", TIOCM_ST},
|
|
#endif
|
|
#ifdef TIOCNOTTY
|
|
{"TIOCNOTTY", TIOCNOTTY},
|
|
#endif
|
|
#ifdef TIOCNXCL
|
|
{"TIOCNXCL", TIOCNXCL},
|
|
#endif
|
|
#ifdef TIOCOUTQ
|
|
{"TIOCOUTQ", TIOCOUTQ},
|
|
#endif
|
|
#ifdef TIOCPKT
|
|
{"TIOCPKT", TIOCPKT},
|
|
#endif
|
|
#ifdef TIOCPKT_DATA
|
|
{"TIOCPKT_DATA", TIOCPKT_DATA},
|
|
#endif
|
|
#ifdef TIOCPKT_DOSTOP
|
|
{"TIOCPKT_DOSTOP", TIOCPKT_DOSTOP},
|
|
#endif
|
|
#ifdef TIOCPKT_FLUSHREAD
|
|
{"TIOCPKT_FLUSHREAD", TIOCPKT_FLUSHREAD},
|
|
#endif
|
|
#ifdef TIOCPKT_FLUSHWRITE
|
|
{"TIOCPKT_FLUSHWRITE", TIOCPKT_FLUSHWRITE},
|
|
#endif
|
|
#ifdef TIOCPKT_NOSTOP
|
|
{"TIOCPKT_NOSTOP", TIOCPKT_NOSTOP},
|
|
#endif
|
|
#ifdef TIOCPKT_START
|
|
{"TIOCPKT_START", TIOCPKT_START},
|
|
#endif
|
|
#ifdef TIOCPKT_STOP
|
|
{"TIOCPKT_STOP", TIOCPKT_STOP},
|
|
#endif
|
|
#ifdef TIOCSCTTY
|
|
{"TIOCSCTTY", TIOCSCTTY},
|
|
#endif
|
|
#ifdef TIOCSERCONFIG
|
|
{"TIOCSERCONFIG", TIOCSERCONFIG},
|
|
#endif
|
|
#ifdef TIOCSERGETLSR
|
|
{"TIOCSERGETLSR", TIOCSERGETLSR},
|
|
#endif
|
|
#ifdef TIOCSERGETMULTI
|
|
{"TIOCSERGETMULTI", TIOCSERGETMULTI},
|
|
#endif
|
|
#ifdef TIOCSERGSTRUCT
|
|
{"TIOCSERGSTRUCT", TIOCSERGSTRUCT},
|
|
#endif
|
|
#ifdef TIOCSERGWILD
|
|
{"TIOCSERGWILD", TIOCSERGWILD},
|
|
#endif
|
|
#ifdef TIOCSERSETMULTI
|
|
{"TIOCSERSETMULTI", TIOCSERSETMULTI},
|
|
#endif
|
|
#ifdef TIOCSERSWILD
|
|
{"TIOCSERSWILD", TIOCSERSWILD},
|
|
#endif
|
|
#ifdef TIOCSER_TEMT
|
|
{"TIOCSER_TEMT", TIOCSER_TEMT},
|
|
#endif
|
|
#ifdef TIOCSETD
|
|
{"TIOCSETD", TIOCSETD},
|
|
#endif
|
|
#ifdef TIOCSLCKTRMIOS
|
|
{"TIOCSLCKTRMIOS", TIOCSLCKTRMIOS},
|
|
#endif
|
|
#ifdef TIOCSPGRP
|
|
{"TIOCSPGRP", TIOCSPGRP},
|
|
#endif
|
|
#ifdef TIOCSSERIAL
|
|
{"TIOCSSERIAL", TIOCSSERIAL},
|
|
#endif
|
|
#ifdef TIOCSSOFTCAR
|
|
{"TIOCSSOFTCAR", TIOCSSOFTCAR},
|
|
#endif
|
|
#ifdef TIOCSTI
|
|
{"TIOCSTI", TIOCSTI},
|
|
#endif
|
|
#ifdef TIOCSWINSZ
|
|
{"TIOCSWINSZ", TIOCSWINSZ},
|
|
#endif
|
|
#ifdef TIOCTTYGSTRUCT
|
|
{"TIOCTTYGSTRUCT", TIOCTTYGSTRUCT},
|
|
#endif
|
|
|
|
/* sentinel */
|
|
{NULL, 0}
|
|
};
|
|
|
|
static int termiosmodule_traverse(PyObject *m, visitproc visit, void *arg) {
|
|
Py_VISIT(modulestate(m)->TermiosError);
|
|
return 0;
|
|
}
|
|
|
|
static int termiosmodule_clear(PyObject *m) {
|
|
Py_CLEAR(modulestate(m)->TermiosError);
|
|
return 0;
|
|
}
|
|
|
|
static void termiosmodule_free(void *m) {
|
|
termiosmodule_clear((PyObject *)m);
|
|
}
|
|
|
|
static struct PyModuleDef termiosmodule = {
|
|
PyModuleDef_HEAD_INIT,
|
|
"termios",
|
|
termios__doc__,
|
|
sizeof(termiosmodulestate),
|
|
termios_methods,
|
|
NULL,
|
|
termiosmodule_traverse,
|
|
termiosmodule_clear,
|
|
termiosmodule_free,
|
|
};
|
|
|
|
PyMODINIT_FUNC
|
|
PyInit_termios(void)
|
|
{
|
|
PyObject *m;
|
|
struct constant *constant = termios_constants;
|
|
|
|
if ((m = PyState_FindModule(&termiosmodule)) != NULL) {
|
|
Py_INCREF(m);
|
|
return m;
|
|
}
|
|
|
|
if ((m = PyModule_Create(&termiosmodule)) == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
termiosmodulestate *state = PyModule_GetState(m);
|
|
state->TermiosError = PyErr_NewException("termios.error", NULL, NULL);
|
|
if (state->TermiosError == NULL) {
|
|
return NULL;
|
|
}
|
|
Py_INCREF(state->TermiosError);
|
|
PyModule_AddObject(m, "error", state->TermiosError);
|
|
|
|
while (constant->name != NULL) {
|
|
PyModule_AddIntConstant(m, constant->name, constant->value);
|
|
++constant;
|
|
}
|
|
return m;
|
|
}
|