1
0
mirror of git://sigrok.org/libserialport synced 2023-08-10 21:13:24 +03:00

Refactor port configuration code.

This commit is contained in:
Martin Ling 2013-11-18 19:52:37 +00:00
parent 067417af4f
commit e33ab9aaf2

View File

@ -52,10 +52,6 @@ struct sp_port_data {
#else #else
struct termios term; struct termios term;
int controlbits; int controlbits;
int rts;
int cts;
int dtr;
int dsr;
#endif #endif
}; };
@ -96,17 +92,8 @@ const struct std_baudrate std_baudrates[] = {
#define NUM_STD_BAUDRATES ARRAY_SIZE(std_baudrates) #define NUM_STD_BAUDRATES ARRAY_SIZE(std_baudrates)
/* Helper functions for configuring ports. */ /* Helper functions for configuring ports. */
static int start_config(struct sp_port *port, struct sp_port_data *data); static int get_config(struct sp_port *port, struct sp_port_data *data, struct sp_port_config *config);
static int set_baudrate(struct sp_port_data *data, int baudrate); static int set_config(struct sp_port *port, struct sp_port_data *data, struct sp_port_config *config);
static int set_bits(struct sp_port_data *data, int bits);
static int set_parity(struct sp_port_data *data, int parity);
static int set_stopbits(struct sp_port_data *data, int stopbits);
static int set_rts(struct sp_port_data *data, int rts);
static int set_cts(struct sp_port_data *data, int cts);
static int set_dtr(struct sp_port_data *data, int dtr);
static int set_dsr(struct sp_port_data *data, int dsr);
static int set_xon_xoff(struct sp_port_data *data, int xon_xoff);
static int apply_config(struct sp_port *port, struct sp_port_data *data);
int sp_get_port_by_name(const char *portname, struct sp_port **port_ptr) int sp_get_port_by_name(const char *portname, struct sp_port **port_ptr)
{ {
@ -450,6 +437,8 @@ int sp_open(struct sp_port *port, int flags)
#else #else
int flags_local = 0; int flags_local = 0;
struct sp_port_data data; struct sp_port_data data;
struct sp_port_config config;
int ret;
/* Map 'flags' to the OS-specific settings. */ /* Map 'flags' to the OS-specific settings. */
if (flags & SP_MODE_RDWR) if (flags & SP_MODE_RDWR)
@ -462,7 +451,13 @@ int sp_open(struct sp_port *port, int flags)
if ((port->fd = open(port->name, flags_local)) < 0) if ((port->fd = open(port->name, flags_local)) < 0)
return SP_ERR_FAIL; return SP_ERR_FAIL;
start_config(port, &data); ret = get_config(port, &data, &config);
if (ret < 0)
{
sp_close(port);
return ret;
}
/* Turn off all serial port cooking. */ /* Turn off all serial port cooking. */
data.term.c_iflag &= ~(ISTRIP | INLCR | ICRNL); data.term.c_iflag &= ~(ISTRIP | INLCR | ICRNL);
@ -476,7 +471,13 @@ int sp_open(struct sp_port *port, int flags)
/* Ignore modem status lines; enable receiver */ /* Ignore modem status lines; enable receiver */
data.term.c_cflag |= (CLOCAL | CREAD); data.term.c_cflag |= (CLOCAL | CREAD);
apply_config(port, &data); ret = set_config(port, &data, &config);
if (ret < 0)
{
sp_close(port);
return ret;
}
#endif #endif
return SP_OK; return SP_OK;
@ -564,344 +565,16 @@ int sp_read(struct sp_port *port, void *buf, size_t count)
#endif #endif
} }
static int start_config(struct sp_port *port, struct sp_port_data *data) static int get_config(struct sp_port *port, struct sp_port_data *data, struct sp_port_config *config)
{ {
CHECK_PORT(); unsigned int i;
#ifdef _WIN32 #ifdef _WIN32
if (!GetCommState(port->hdl, &data->dcb)) if (!GetCommState(port->hdl, &data->dcb))
return SP_ERR_FAIL; return SP_ERR_FAIL;
#else
if (tcgetattr(port->fd, &data->term) < 0)
return SP_ERR_FAIL;
if (ioctl(port->fd, TIOCMGET, &data->controlbits) < 0)
return SP_ERR_FAIL;
if (data->term.c_cflag & CRTSCTS) {
data->rts = SP_RTS_FLOW_CONTROL;
data->cts = SP_CTS_FLOW_CONTROL;
} else {
data->rts = (data->controlbits & TIOCM_RTS) ? SP_RTS_ON : SP_RTS_OFF;
data->cts = SP_CTS_IGNORE;
}
data->dtr = (data->controlbits & TIOCM_DTR) ? SP_DTR_ON : SP_DTR_OFF;
data->dsr = SP_DSR_IGNORE;
#endif
return SP_OK;
}
static int set_baudrate(struct sp_port_data *data, int baudrate)
{
unsigned int i;
for (i = 0; i < NUM_STD_BAUDRATES; i++) { for (i = 0; i < NUM_STD_BAUDRATES; i++) {
if (baudrate == std_baudrates[i].value) { if (data->dcb.BaudRate == std_baudrates[i].index) {
#ifdef _WIN32
data->dcb.BaudRate = std_baudrates[i].index;
#else
if (cfsetospeed(&data->term, std_baudrates[i].index) < 0)
return SP_ERR_FAIL;
if (cfsetispeed(&data->term, std_baudrates[i].index) < 0)
return SP_ERR_FAIL;
#endif
break;
}
}
if (i == NUM_STD_BAUDRATES)
return SP_ERR_ARG;
return SP_OK;
}
static int set_bits(struct sp_port_data *data, int bits)
{
#ifdef _WIN32
data->dcb.ByteSize = bits;
#else
data->term.c_cflag &= ~CSIZE;
switch (bits) {
case 8:
data->term.c_cflag |= CS8;
break;
case 7:
data->term.c_cflag |= CS7;
break;
case 6:
data->term.c_cflag |= CS6;
break;
default:
return SP_ERR_ARG;
}
#endif
return SP_OK;
}
static int set_parity(struct sp_port_data *data, int parity)
{
#ifdef _WIN32
switch (parity) {
/* Note: There's also SPACEPARITY, MARKPARITY (unneeded so far). */
case SP_PARITY_NONE:
data->dcb.Parity = NOPARITY;
break;
case SP_PARITY_EVEN:
data->dcb.Parity = EVENPARITY;
break;
case SP_PARITY_ODD:
data->dcb.Parity = ODDPARITY;
break;
default:
return SP_ERR_ARG;
}
#else
data->term.c_iflag &= ~IGNPAR;
data->term.c_cflag &= ~(PARENB | PARODD);
switch (parity) {
case SP_PARITY_NONE:
data->term.c_iflag |= IGNPAR;
break;
case SP_PARITY_EVEN:
data->term.c_cflag |= PARENB;
break;
case SP_PARITY_ODD:
data->term.c_cflag |= PARENB | PARODD;
break;
default:
return SP_ERR_ARG;
}
#endif
return SP_OK;
}
static int set_stopbits(struct sp_port_data *data, int stopbits)
{
#ifdef _WIN32
switch (stopbits) {
/* Note: There's also ONE5STOPBITS == 1.5 (unneeded so far). */
case 1:
data->dcb.StopBits = ONESTOPBIT;
break;
case 2:
data->dcb.StopBits = TWOSTOPBITS;
break;
default:
return SP_ERR_ARG;
}
#else
data->term.c_cflag &= ~CSTOPB;
switch (stopbits) {
case 1:
data->term.c_cflag &= ~CSTOPB;
break;
case 2:
data->term.c_cflag |= CSTOPB;
break;
default:
return SP_ERR_ARG;
}
#endif
return SP_OK;
}
static int set_rts(struct sp_port_data *data, int rts)
{
#ifdef _WIN32
switch (rts) {
case SP_RTS_OFF:
data->dcb.fRtsControl = RTS_CONTROL_DISABLE;
break;
case SP_RTS_ON:
data->dcb.fRtsControl = RTS_CONTROL_ENABLE;
break;
case SP_RTS_FLOW_CONTROL:
data->dcb.fRtsControl = RTS_CONTROL_HANDSHAKE;
break;
default:
return SP_ERR_ARG;
}
#else
data->rts = rts;
#endif
return SP_OK;
}
static int set_cts(struct sp_port_data *data, int cts)
{
#ifdef _WIN32
switch (cts) {
case SP_CTS_IGNORE:
data->dcb.fOutxCtsFlow = FALSE;
break;
case SP_CTS_FLOW_CONTROL:
data->dcb.fOutxCtsFlow = TRUE;
break;
default:
return SP_ERR_ARG;
}
#else
data->cts = cts;
#endif
return SP_OK;
}
static int set_dtr(struct sp_port_data *data, int dtr)
{
#ifdef _WIN32
switch (dtr) {
case SP_DTR_OFF:
data->dcb.fDtrControl = DTR_CONTROL_DISABLE;
break;
case SP_DTR_ON:
data->dcb.fDtrControl = DTR_CONTROL_ENABLE;
break;
case SP_DTR_FLOW_CONTROL:
data->dcb.fDtrControl = DTR_CONTROL_HANDSHAKE;
break;
default:
return SP_ERR_ARG;
}
#else
data->dtr = dtr;
#endif
return SP_OK;
}
static int set_dsr(struct sp_port_data *data, int dsr)
{
#ifdef _WIN32
switch (dsr) {
case SP_DSR_IGNORE:
data->dcb.fOutxDsrFlow = FALSE;
break;
case SP_DSR_FLOW_CONTROL:
data->dcb.fOutxDsrFlow = TRUE;
break;
default:
return SP_ERR_ARG;
}
#else
data->dsr = dsr;
#endif
return SP_OK;
}
static int set_xon_xoff(struct sp_port_data *data, int xon_xoff)
{
#ifdef _WIN32
switch (xon_xoff) {
case SP_XONXOFF_DISABLED:
data->dcb.fInX = FALSE;
data->dcb.fOutX = FALSE;
break;
case SP_XONXOFF_IN:
data->dcb.fInX = TRUE;
data->dcb.fOutX = FALSE;
break;
case SP_XONXOFF_OUT:
data->dcb.fInX = FALSE;
data->dcb.fOutX = TRUE;
break;
case SP_XONXOFF_INOUT:
data->dcb.fInX = TRUE;
data->dcb.fOutX = TRUE;
break;
default:
return SP_ERR_ARG;
}
#else
data->term.c_iflag &= ~(IXON | IXOFF | IXANY);
switch (xon_xoff) {
case SP_XONXOFF_DISABLED:
break;
case SP_XONXOFF_IN:
data->term.c_iflag |= IXOFF;
break;
case SP_XONXOFF_OUT:
data->term.c_iflag |= IXON | IXANY;
break;
case SP_XONXOFF_INOUT:
data->term.c_iflag |= IXON | IXOFF | IXANY;
break;
default:
return SP_ERR_ARG;
}
#endif
return SP_OK;
}
static int apply_config(struct sp_port *port, struct sp_port_data *data)
{
#ifdef _WIN32
if (!SetCommState(port->hdl, &data->dcb))
return SP_ERR_FAIL;
#else
int controlbits;
/* Asymmetric use of RTS/CTS not supported yet. */
if ((data->rts == SP_RTS_FLOW_CONTROL) != (data->cts == SP_CTS_FLOW_CONTROL))
return SP_ERR_ARG;
/* DTR/DSR flow control not supported yet. */
if (data->dtr == SP_DTR_FLOW_CONTROL || data->dsr == SP_DSR_FLOW_CONTROL)
return SP_ERR_ARG;
if (data->rts == SP_RTS_FLOW_CONTROL)
data->term.c_iflag |= CRTSCTS;
else
{
controlbits = TIOCM_RTS;
if (ioctl(port->fd, data->rts == SP_RTS_ON ? TIOCMBIS : TIOCMBIC,
&controlbits) < 0)
return SP_ERR_FAIL;
}
controlbits = TIOCM_DTR;
if (ioctl(port->fd, data->dtr == SP_DTR_ON ? TIOCMBIS : TIOCMBIC,
&controlbits) < 0)
return SP_ERR_FAIL;
/* Write the configured settings. */
if (tcsetattr(port->fd, TCSADRAIN, &data->term) < 0)
return SP_ERR_FAIL;
#endif
return SP_OK;
}
#define TRY(x) do { int ret = x; if (ret != SP_OK) return ret; } while (0)
#define TRY_SET(x, y) do { if (y >= 0) TRY(set_##x(&data, y)); } while (0)
#define TRY_SET_CONFIG(x) TRY_SET(x, config->x)
int sp_set_config(struct sp_port *port, struct sp_port_config *config)
{
struct sp_port_data data;
TRY(start_config(port, &data));
TRY_SET_CONFIG(baudrate);
TRY_SET_CONFIG(bits);
TRY_SET_CONFIG(parity);
TRY_SET_CONFIG(stopbits);
TRY_SET_CONFIG(rts);
TRY_SET_CONFIG(cts);
TRY_SET_CONFIG(dtr);
TRY_SET_CONFIG(dsr);
TRY_SET_CONFIG(xon_xoff);
TRY(apply_config(port, &data));
return SP_OK;
}
int sp_get_config(struct sp_port *port, struct sp_port_config *config)
{
struct sp_port_data data;
unsigned int i;
TRY(start_config(port, &data));
#ifdef _WIN32
for (i = 0; i < NUM_STD_BAUDRATES; i++) {
if (data.dcb.BaudRate == std_baudrates[i].index) {
config->baudrate = std_baudrates[i].value; config->baudrate = std_baudrates[i].value;
break; break;
} }
@ -909,12 +582,12 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
if (i == NUM_STD_BAUDRATES) if (i == NUM_STD_BAUDRATES)
/* BaudRate field can be either an index or a custom baud rate. */ /* BaudRate field can be either an index or a custom baud rate. */
config->baudrate = data.dcb.BaudRate; config->baudrate = data->dcb.BaudRate;
config->bits = data.dcb.ByteSize; config->bits = data->dcb.ByteSize;
if (data.dcb.fParity) if (data->dcb.fParity)
switch (data.dcb.Parity) { switch (data->dcb.Parity) {
case NOPARITY: case NOPARITY:
config->parity = SP_PARITY_NONE; config->parity = SP_PARITY_NONE;
break; break;
@ -930,7 +603,7 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
else else
config->parity = SP_PARITY_NONE; config->parity = SP_PARITY_NONE;
switch (data.dcb.StopBits) { switch (data->dcb.StopBits) {
case ONESTOPBIT: case ONESTOPBIT:
config->stopbits = 1; config->stopbits = 1;
break; break;
@ -941,7 +614,7 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
config->stopbits = -1; config->stopbits = -1;
} }
switch (data.dcb.fRtsControl) { switch (data->dcb.fRtsControl) {
case RTS_CONTROL_DISABLE: case RTS_CONTROL_DISABLE:
config->rts = SP_RTS_OFF; config->rts = SP_RTS_OFF;
break; break;
@ -955,9 +628,9 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
config->rts = -1; config->rts = -1;
} }
config->cts = data.dcb.fOutxCtsFlow ? SP_CTS_FLOW_CONTROL : SP_CTS_IGNORE; config->cts = data->dcb.fOutxCtsFlow ? SP_CTS_FLOW_CONTROL : SP_CTS_IGNORE;
switch (data.dcb.fDtrControl) { switch (data->dcb.fDtrControl) {
case DTR_CONTROL_DISABLE: case DTR_CONTROL_DISABLE:
config->dtr = SP_DTR_OFF; config->dtr = SP_DTR_OFF;
break; break;
@ -971,10 +644,17 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
config->dtr = -1; config->dtr = -1;
} }
config->dsr = data.dcb.fOutxDsrFlow ? SP_DSR_FLOW_CONTROL : SP_DSR_IGNORE; config->dsr = data->dcb.fOutxDsrFlow ? SP_DSR_FLOW_CONTROL : SP_DSR_IGNORE;
#else
#else // !_WIN32
if (tcgetattr(port->fd, &data->term) < 0)
return SP_ERR_FAIL;
if (ioctl(port->fd, TIOCMGET, &data->controlbits) < 0)
return SP_ERR_FAIL;
for (i = 0; i < NUM_STD_BAUDRATES; i++) { for (i = 0; i < NUM_STD_BAUDRATES; i++) {
if (cfgetispeed(&data.term) == std_baudrates[i].index) { if (cfgetispeed(&data->term) == std_baudrates[i].index) {
config->baudrate = std_baudrates[i].value; config->baudrate = std_baudrates[i].value;
break; break;
} }
@ -983,7 +663,7 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
if (i == NUM_STD_BAUDRATES) if (i == NUM_STD_BAUDRATES)
config->baudrate = -1; config->baudrate = -1;
switch (data.term.c_cflag & CSIZE) { switch (data->term.c_cflag & CSIZE) {
case CS8: case CS8:
config->bits = 8; config->bits = 8;
break; break;
@ -1000,67 +680,326 @@ int sp_get_config(struct sp_port *port, struct sp_port_config *config)
config->bits = -1; config->bits = -1;
} }
if (!(data.term.c_cflag & PARENB) && (data.term.c_iflag & IGNPAR)) if (!(data->term.c_cflag & PARENB) && (data->term.c_iflag & IGNPAR))
config->parity = SP_PARITY_NONE; config->parity = SP_PARITY_NONE;
else if (!(data.term.c_cflag & PARENB) || (data.term.c_iflag & IGNPAR)) else if (!(data->term.c_cflag & PARENB) || (data->term.c_iflag & IGNPAR))
config->parity = -1; config->parity = -1;
else else
config->parity = (data.term.c_cflag & PARODD) ? SP_PARITY_ODD : SP_PARITY_EVEN; config->parity = (data->term.c_cflag & PARODD) ? SP_PARITY_ODD : SP_PARITY_EVEN;
config->stopbits = (data.term.c_cflag & CSTOPB) ? 2 : 1; config->stopbits = (data->term.c_cflag & CSTOPB) ? 2 : 1;
if (data.term.c_cflag & CRTSCTS) { if (data->term.c_cflag & CRTSCTS) {
config->rts = SP_RTS_FLOW_CONTROL; config->rts = SP_RTS_FLOW_CONTROL;
config->cts = SP_CTS_FLOW_CONTROL; config->cts = SP_CTS_FLOW_CONTROL;
} else { } else {
config->rts = (data.controlbits & TIOCM_RTS) ? SP_RTS_ON : SP_RTS_OFF; config->rts = (data->controlbits & TIOCM_RTS) ? SP_RTS_ON : SP_RTS_OFF;
config->cts = SP_CTS_IGNORE; config->cts = SP_CTS_IGNORE;
} }
config->dtr = (data.controlbits & TIOCM_DTR) ? SP_DTR_ON : SP_DTR_OFF; config->dtr = (data->controlbits & TIOCM_DTR) ? SP_DTR_ON : SP_DTR_OFF;
config->dsr = SP_DSR_IGNORE; config->dsr = SP_DSR_IGNORE;
#endif #endif
return SP_OK; return SP_OK;
} }
int sp_set_flowcontrol(struct sp_port *port, int flowcontrol) static int set_config(struct sp_port *port, struct sp_port_data *data, struct sp_port_config *config)
{
unsigned int i;
#ifdef _WIN32
if (config->baudrate >= 0)
{
for (i = 0; i < NUM_STD_BAUDRATES; i++) {
if (config->baudrate == std_baudrates[i].value) {
data->dcb.BaudRate = std_baudrates[i].index;
break;
}
}
if (i == NUM_STD_BAUDRATES)
data->dcb.BaudRate = config->baudrate;
}
if (config->bits >= 0)
data->dcb.ByteSize = config->bits;
if (config->parity >= 0) {
switch (config->parity) {
/* Note: There's also SPACEPARITY, MARKPARITY (unneeded so far). */
case SP_PARITY_NONE:
data->dcb.Parity = NOPARITY;
break;
case SP_PARITY_EVEN:
data->dcb.Parity = EVENPARITY;
break;
case SP_PARITY_ODD:
data->dcb.Parity = ODDPARITY;
break;
default:
return SP_ERR_ARG;
}
}
if (config->stopbits >= 0) {
switch (config->stopbits) {
/* Note: There's also ONE5STOPBITS == 1.5 (unneeded so far). */
case 1:
data->dcb.StopBits = ONESTOPBIT;
break;
case 2:
data->dcb.StopBits = TWOSTOPBITS;
break;
default:
return SP_ERR_ARG;
}
}
if (config->rts >= 0) {
switch (config->rts) {
case SP_RTS_OFF:
data->dcb.fRtsControl = RTS_CONTROL_DISABLE;
break;
case SP_RTS_ON:
data->dcb.fRtsControl = RTS_CONTROL_ENABLE;
break;
case SP_RTS_FLOW_CONTROL:
data->dcb.fRtsControl = RTS_CONTROL_HANDSHAKE;
break;
default:
return SP_ERR_ARG;
}
}
if (config->cts >= 0) {
switch (config->cts) {
case SP_CTS_IGNORE:
data->dcb.fOutxCtsFlow = FALSE;
break;
case SP_CTS_FLOW_CONTROL:
data->dcb.fOutxCtsFlow = TRUE;
break;
default:
return SP_ERR_ARG;
}
}
if (config->dtr >= 0) {
switch (config->dtr) {
case SP_DTR_OFF:
data->dcb.fDtrControl = DTR_CONTROL_DISABLE;
break;
case SP_DTR_ON:
data->dcb.fDtrControl = DTR_CONTROL_ENABLE;
break;
case SP_DTR_FLOW_CONTROL:
data->dcb.fDtrControl = DTR_CONTROL_HANDSHAKE;
break;
default:
return SP_ERR_ARG;
}
}
if (config->dsr >= 0) {
switch (config->dsr) {
case SP_DSR_IGNORE:
data->dcb.fOutxDsrFlow = FALSE;
break;
case SP_DSR_FLOW_CONTROL:
data->dcb.fOutxDsrFlow = TRUE;
break;
default:
return SP_ERR_ARG;
}
}
if (config->xon_xoff >= 0) {
switch (config->xon_xoff) {
case SP_XONXOFF_DISABLED:
data->dcb.fInX = FALSE;
data->dcb.fOutX = FALSE;
break;
case SP_XONXOFF_IN:
data->dcb.fInX = TRUE;
data->dcb.fOutX = FALSE;
break;
case SP_XONXOFF_OUT:
data->dcb.fInX = FALSE;
data->dcb.fOutX = TRUE;
break;
case SP_XONXOFF_INOUT:
data->dcb.fInX = TRUE;
data->dcb.fOutX = TRUE;
break;
default:
return SP_ERR_ARG;
}
}
if (!SetCommState(port->hdl, &data->dcb))
return SP_ERR_FAIL;
#else // !_WIN32
if (config->baudrate >= 0)
{
for (i = 0; i < NUM_STD_BAUDRATES; i++) {
if (config->baudrate == std_baudrates[i].value) {
if (cfsetospeed(&data->term, std_baudrates[i].index) < 0)
return SP_ERR_FAIL;
if (cfsetispeed(&data->term, std_baudrates[i].index) < 0)
return SP_ERR_FAIL;
break;
}
}
if (i == NUM_STD_BAUDRATES)
return SP_ERR_ARG;
}
if (config->bits >= 0) {
data->term.c_cflag &= ~CSIZE;
switch (config->bits) {
case 8:
data->term.c_cflag |= CS8;
break;
case 7:
data->term.c_cflag |= CS7;
break;
case 6:
data->term.c_cflag |= CS6;
break;
default:
return SP_ERR_ARG;
}
}
if (config->parity >= 0) {
data->term.c_iflag &= ~IGNPAR;
data->term.c_cflag &= ~(PARENB | PARODD);
switch (config->parity) {
case SP_PARITY_NONE:
data->term.c_iflag |= IGNPAR;
break;
case SP_PARITY_EVEN:
data->term.c_cflag |= PARENB;
break;
case SP_PARITY_ODD:
data->term.c_cflag |= PARENB | PARODD;
break;
default:
return SP_ERR_ARG;
}
}
if (config->stopbits >= 0) {
data->term.c_cflag &= ~CSTOPB;
switch (config->stopbits) {
case 1:
data->term.c_cflag &= ~CSTOPB;
break;
case 2:
data->term.c_cflag |= CSTOPB;
break;
default:
return SP_ERR_ARG;
}
}
if (config->rts >= 0 || config->cts >= 0)
{
/* Asymmetric use of RTS/CTS not supported yet. */
if (data->term.c_iflag & CRTSCTS) {
/* Flow control can only be disabled for both RTS & CTS together. */
if (config->rts >= 0 && config->rts != SP_RTS_FLOW_CONTROL) {
if (config->cts != SP_CTS_IGNORE)
return SP_ERR_ARG;
}
if (config->cts >= 0 && config->cts != SP_CTS_FLOW_CONTROL) {
if (config->rts <= 0 || config->rts == SP_RTS_FLOW_CONTROL)
return SP_ERR_ARG;
}
} else {
/* Flow control can only be enabled for both RTS & CTS together. */
if (((config->rts == SP_RTS_FLOW_CONTROL) && (config->cts != SP_CTS_FLOW_CONTROL)) ||
((config->cts == SP_CTS_FLOW_CONTROL) && (config->rts != SP_RTS_FLOW_CONTROL)))
return SP_ERR_ARG;
}
if (config->rts >= 0) {
if (config->rts == SP_RTS_FLOW_CONTROL) {
data->term.c_iflag |= CRTSCTS;
} else {
int controlbits = TIOCM_RTS;
if (ioctl(port->fd, config->rts == SP_RTS_ON ? TIOCMBIS : TIOCMBIC,
&controlbits) < 0)
return SP_ERR_FAIL;
}
}
}
if (config->dtr >= 0 || config->dsr >= 0)
{
/* DTR/DSR flow control not supported yet. */
if (config->dtr == SP_DTR_FLOW_CONTROL || config->dsr == SP_DSR_FLOW_CONTROL)
return SP_ERR_ARG;
if (config->dtr >= 0) {
int controlbits = TIOCM_DTR;
if (ioctl(port->fd, config->dtr == SP_DTR_ON ? TIOCMBIS : TIOCMBIC,
&controlbits) < 0)
return SP_ERR_FAIL;
}
}
if (config->xon_xoff >= 0) {
data->term.c_iflag &= ~(IXON | IXOFF | IXANY);
switch (config->xon_xoff) {
case SP_XONXOFF_DISABLED:
break;
case SP_XONXOFF_IN:
data->term.c_iflag |= IXOFF;
break;
case SP_XONXOFF_OUT:
data->term.c_iflag |= IXON | IXANY;
break;
case SP_XONXOFF_INOUT:
data->term.c_iflag |= IXON | IXOFF | IXANY;
break;
default:
return SP_ERR_ARG;
}
}
if (tcsetattr(port->fd, TCSADRAIN, &data->term) < 0)
return SP_ERR_FAIL;
#endif
return SP_OK;
}
#define TRY(x) do { int ret = x; if (ret != SP_OK) return ret; } while (0)
int sp_set_config(struct sp_port *port, struct sp_port_config *config)
{ {
struct sp_port_data data; struct sp_port_data data;
struct sp_port_config prev_config;
TRY(start_config(port, &data)); TRY(get_config(port, &data, &prev_config));
TRY(set_config(port, &data, config));
if (flowcontrol == SP_FLOWCONTROL_XONXOFF)
TRY_SET(xon_xoff, SP_XONXOFF_INOUT);
else
TRY_SET(xon_xoff, SP_XONXOFF_DISABLED);
if (flowcontrol == SP_FLOWCONTROL_RTSCTS) {
TRY_SET(rts, SP_RTS_FLOW_CONTROL);
TRY_SET(cts, SP_CTS_FLOW_CONTROL);
} else {
TRY_SET(rts, SP_RTS_ON);
TRY_SET(cts, SP_CTS_IGNORE);
}
if (flowcontrol == SP_FLOWCONTROL_DTRDSR) {
TRY_SET(dtr, SP_DTR_FLOW_CONTROL);
TRY_SET(dsr, SP_DSR_FLOW_CONTROL);
} else {
TRY_SET(dtr, SP_DTR_ON);
TRY_SET(dsr, SP_DSR_IGNORE);
}
TRY(apply_config(port, &data));
return SP_OK; return SP_OK;
} }
#define CREATE_SETTER(x) int sp_set_##x(struct sp_port *port, int x) { \ #define CREATE_SETTER(x) int sp_set_##x(struct sp_port *port, int x) { \
struct sp_port_data data; \ struct sp_port_data data; \
TRY(start_config(port, &data)); \ struct sp_port_config config; \
TRY(set_##x(&data, x)); \ TRY(get_config(port, &data, &config)); \
TRY(apply_config(port, &data)); \ config.x = x; \
TRY(set_config(port, &data, &config)); \
return SP_OK; \ return SP_OK; \
} }
@ -1074,6 +1013,41 @@ CREATE_SETTER(dtr)
CREATE_SETTER(dsr) CREATE_SETTER(dsr)
CREATE_SETTER(xon_xoff) CREATE_SETTER(xon_xoff)
int sp_set_flowcontrol(struct sp_port *port, int flowcontrol)
{
struct sp_port_data data;
struct sp_port_config config;
TRY(get_config(port, &data, &config));
if (flowcontrol == SP_FLOWCONTROL_XONXOFF)
config.xon_xoff = SP_XONXOFF_INOUT;
else
config.xon_xoff = SP_XONXOFF_DISABLED;
if (flowcontrol == SP_FLOWCONTROL_RTSCTS) {
config.rts = SP_RTS_FLOW_CONTROL;
config.cts = SP_CTS_FLOW_CONTROL;
} else {
if (config.rts == SP_RTS_FLOW_CONTROL)
config.rts = SP_RTS_ON;
config.cts = SP_CTS_IGNORE;
}
if (flowcontrol == SP_FLOWCONTROL_DTRDSR) {
config.dtr = SP_DTR_FLOW_CONTROL;
config.dsr = SP_DSR_FLOW_CONTROL;
} else {
if (config.dtr == SP_DTR_FLOW_CONTROL)
config.dtr = SP_DTR_ON;
config.dsr = SP_DSR_IGNORE;
}
TRY(set_config(port, &data, &config));
return SP_OK;
}
int sp_last_error_code(void) int sp_last_error_code(void)
{ {
#ifdef _WIN32 #ifdef _WIN32