X-Git-Url: http://sigrok.org/gitweb/?a=blobdiff_plain;f=linux_termios.c;h=e23d187b0dac63210c203edfa3b177ab6d44980a;hb=962143bc5c3a9a6e130cc1273ab465238c178d70;hp=874889aba4641c7ed268923f1eeb32d4908b5d47;hpb=40978c2b4182d09b197a0218cc107e49509747ad;p=libserialport.git diff --git a/linux_termios.c b/linux_termios.c index 874889a..e23d187 100644 --- a/linux_termios.c +++ b/linux_termios.c @@ -17,39 +17,57 @@ * along with this program. If not, see . */ +/* + * At the time of writing, glibc does not support the Linux kernel interfaces + * for setting non-standard baud rates and flow control. We therefore have to + * prepare the correct ioctls ourselves, for which we need the declarations in + * linux/termios.h. + * + * We can't include linux/termios.h in serialport.c however, because its + * contents conflict with the termios.h provided by glibc. So this file exists + * to isolate the bits of code which use the kernel termios declarations. + * + * The details vary by architecture. Some architectures have c_ispeed/c_ospeed + * in struct termios, accessed with TCSETS/TCGETS. Others have these fields in + * struct termios2, accessed with TCSETS2/TCGETS2. Some architectures have the + * TCSETX/TCGETX ioctls used with struct termiox, others do not. + */ + +#include #include #include "linux_termios.h" -int get_termios_get_ioctl(void) +SP_PRIV unsigned long get_termios_get_ioctl(void) { -#ifdef HAVE_TERMIOS2 +#ifdef HAVE_STRUCT_TERMIOS2 return TCGETS2; #else return TCGETS; #endif } -int get_termios_set_ioctl(void) +SP_PRIV unsigned long get_termios_set_ioctl(void) { -#ifdef HAVE_TERMIOS2 +#ifdef HAVE_STRUCT_TERMIOS2 return TCSETS2; #else return TCSETS; #endif } -int get_termios_size(void) +SP_PRIV size_t get_termios_size(void) { -#ifdef HAVE_TERMIOS2 +#ifdef HAVE_STRUCT_TERMIOS2 return sizeof(struct termios2); #else return sizeof(struct termios); #endif } -int get_termios_speed(void *data) +#if (defined(HAVE_TERMIOS_SPEED) || defined(HAVE_TERMIOS2_SPEED)) && defined(HAVE_DECL_BOTHER) +SP_PRIV int get_termios_speed(void *data) { -#ifdef HAVE_TERMIOS2 +#ifdef HAVE_STRUCT_TERMIOS2 struct termios2 *term = (struct termios2 *) data; #else struct termios *term = (struct termios *) data; @@ -60,9 +78,9 @@ int get_termios_speed(void *data) return term->c_ispeed; } -void set_termios_speed(void *data, int speed) +SP_PRIV void set_termios_speed(void *data, int speed) { -#ifdef HAVE_TERMIOS2 +#ifdef HAVE_STRUCT_TERMIOS2 struct termios2 *term = (struct termios2 *) data; #else struct termios *term = (struct termios *) data; @@ -71,43 +89,40 @@ void set_termios_speed(void *data, int speed) term->c_cflag |= BOTHER; term->c_ispeed = term->c_ospeed = speed; } +#endif -#ifdef HAVE_TERMIOX -int get_termiox_size(void) +#ifdef HAVE_STRUCT_TERMIOX +SP_PRIV size_t get_termiox_size(void) { return sizeof(struct termiox); } -int get_termiox_flow(void *data) +SP_PRIV int get_termiox_flow(void *data, int *rts, int *cts, int *dtr, int *dsr) { struct termiox *termx = (struct termiox *) data; - int flags = 0; - if (termx->x_cflag & RTSXOFF) - flags |= RTS_FLOW; - if (termx->x_cflag & CTSXON) - flags |= CTS_FLOW; - if (termx->x_cflag & DTRXOFF) - flags |= DTR_FLOW; - if (termx->x_cflag & DSRXON) - flags |= DSR_FLOW; + + *rts = (termx->x_cflag & RTSXOFF); + *cts = (termx->x_cflag & CTSXON); + *dtr = (termx->x_cflag & DTRXOFF); + *dsr = (termx->x_cflag & DSRXON); return flags; } -void set_termiox_flow(void *data, int flags) +SP_PRIV void set_termiox_flow(void *data, int rts, int cts, int dtr, int dsr) { struct termiox *termx = (struct termiox *) data; termx->x_cflag &= ~(RTSXOFF | CTSXON | DTRXOFF | DSRXON); - if (flags & RTS_FLOW) + if (rts) termx->x_cflag |= RTSXOFF; - if (flags & CTS_FLOW) + if (cts) termx->x_cflag |= CTSXON; - if (flags & DTR_FLOW) + if (dtr) termx->x_cflag |= DTRXOFF; - if (flags & DSR_FLOW) + if (dsr) termx->x_cflag |= DSRXON; } #endif