无法写入串行设备但可以从中读取
Unable to write to serial device but able to read from it
我正在使用非阻塞 IO,我只是尝试写入和读取串行端口。串口读取正常,写入不正常。
这就是我设置串口线的方式。如您所见,它设置为非阻塞和规范的。也许有些标志是多余的,但它们大多取自此处的示例:Serial setup example
tSerial::tSerial(const char *serialPort, bool echo)
{
// Open the serial port. Change device path as needed (currently set to an standard FTDI USB-UART cable type device)
m_serialPort = open(serialPort, O_RDWR);
// Create new termios struct, we call it 'tty' for convention
struct termios tty;
// Read in existing settings, and handle any error
if (tcgetattr(m_serialPort, &tty) != 0)
{
spdlog::error("[tSerial] error {} from tcgetattr: {}", errno, strerror(errno));
throw std::runtime_error("Failed to get existing serial settings");
}
tty.c_cflag &= ~PARENB; // Clear parity bit, disabling parity (most common)
tty.c_cflag &= ~CSTOPB; // Clear stop field, only one stop bit used in communication (most common)
tty.c_cflag &= ~CSIZE; // Clear all bits that set the data size
tty.c_cflag |= CS8; // 8 bits per byte (most common)
tty.c_cflag &= ~CRTSCTS; // Disable RTS/CTS hardware flow control (most common)
tty.c_cflag |= CREAD | CLOCAL; // Turn on READ & ignore ctrl lines (CLOCAL = 1)
// tty.c_lflag &= ~ICANON;
tty.c_cflag |= ICANON;
if (!echo)
{
tty.c_lflag &= ~ECHO; // Disable echo
}
else
{
tty.c_lflag |= ECHO; // Enable echo
}
// tty.c_lflag &= ~ECHOE; // Disable erasure
// tty.c_lflag &= ~ECHONL; // Disable new-line echo
// tty.c_lflag &= ~ISIG; // Disable interpretation of INTR, QUIT and SUSP
tty.c_iflag &= ~(IXON | IXOFF | IXANY); // Turn off s/w flow ctrl
tty.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL); // Disable any special handling of received bytes
tty.c_oflag &= ~OPOST; // Prevent special interpretation of output bytes (e.g. newline chars)
tty.c_oflag &= ~ONLCR; // Prevent conversion of newline to carriage return/line feed
// Set in/out baud rate to be 115200
cfsetispeed(&tty, B115200);
cfsetospeed(&tty, B115200);
// Save tty settings, also checking for error
if (tcsetattr(m_serialPort, TCSANOW, &tty) != 0)
{
spdlog::error("[tSerial] error {} from tcsetattr: {}", errno, strerror(errno));
throw std::runtime_error("Failed to set new serial settings");
}
// Set non-blocking
int flags;
if ((flags = fcntl(m_serialPort, F_GETFL, 0)) == -1)
{
flags = 0;
}
fcntl(m_serialPort, F_SETFL, flags | O_NONBLOCK);
}
我就是这样写的。我正在使用 select
等待 IO 资源可用,因为尝试直接写入它会出现错误 Resource temporarily unavailable
。但是等了20秒,设备还是不可用
void tSerial::writeSerial(std::string message)
{
int n;
fd_set rfds;
FD_ZERO(&rfds);
FD_SET(m_serialPort, &rfds);
spdlog::debug("[tSerial] writing command to serial {}", message);
struct timeval tv;
tv.tv_sec = 20;
tv.tv_usec = 0;
int retval = select(1, NULL, &rfds, NULL, &tv);
if (retval == -1)
{
spdlog::error("[tSerial] select error");
}
else if (retval)
{
n = write(m_serialPort, message.c_str(), message.length());
tcflush(m_serialPort, TCIOFLUSH);
if (n < 0)
{
spdlog::error("[tSerial] error writing: {}", strerror(errno));
}
}
else
{
spdlog::error("[tSerial] Resource unavailable after 20 seconds wait");
}
}
I am using select
to wait for the IO resource to become available ... But even waiting 20 seconds, the device is still unavailable.
您的程序未按预期运行,因为您没有正确编写 select() 调用:
int retval = select(1, NULL, &rfds, NULL, &tv);
根据 man 页面,第一个参数“ 应设置为三个集合中任何一个中编号最大的文件描述符,再加上 1。"
由于您正在传递值 1
,因此 select() 可以检查的唯一文件描述符是 stdin
(它是只读的并且永远不会准备好输出)并且永远不会(文件描述符)打开的串行终端。
相反,系统调用需要
select(m_serialPort + 1, ...);
您的代码还有其他问题。
(1) 非阻塞模式的使用值得怀疑。
似乎您更愿意向程序添加额外的代码以等待,而不是让 OS 为您完成。如果您的程序没有有效且高效地利用其时间片,那么您最好让 OS 管理系统资源。
(2) 在 write() 之后使用 tcflush() 是荒谬和错误的!
您可能打算使用 tcdrain().
研究 man 页面。
(3) 当您复制和修改代码时,您至少引入了一个错误。
ICANON
在 c_lflag
成员中,而不在 c_cflag
.
中
(4) 良好的编码习惯是使用有意义的变量名。
重用用于读取参数的变量名,因为写入参数草率且令人困惑:
fd_set rfds;
...
int retval = select(..., NULL, &rfds, NULL, &tv);
而不是 rfds
,您的程序应该使用 wfds
作为第三个参数。
... trying to write to it directly gives the error Resource temporarily unavailable
使用select()时的问题很容易解释。
对于上述问题,您需要提供一个最小的、可重现的示例,即一个完整的程序。
我正在使用非阻塞 IO,我只是尝试写入和读取串行端口。串口读取正常,写入不正常。
这就是我设置串口线的方式。如您所见,它设置为非阻塞和规范的。也许有些标志是多余的,但它们大多取自此处的示例:Serial setup example
tSerial::tSerial(const char *serialPort, bool echo)
{
// Open the serial port. Change device path as needed (currently set to an standard FTDI USB-UART cable type device)
m_serialPort = open(serialPort, O_RDWR);
// Create new termios struct, we call it 'tty' for convention
struct termios tty;
// Read in existing settings, and handle any error
if (tcgetattr(m_serialPort, &tty) != 0)
{
spdlog::error("[tSerial] error {} from tcgetattr: {}", errno, strerror(errno));
throw std::runtime_error("Failed to get existing serial settings");
}
tty.c_cflag &= ~PARENB; // Clear parity bit, disabling parity (most common)
tty.c_cflag &= ~CSTOPB; // Clear stop field, only one stop bit used in communication (most common)
tty.c_cflag &= ~CSIZE; // Clear all bits that set the data size
tty.c_cflag |= CS8; // 8 bits per byte (most common)
tty.c_cflag &= ~CRTSCTS; // Disable RTS/CTS hardware flow control (most common)
tty.c_cflag |= CREAD | CLOCAL; // Turn on READ & ignore ctrl lines (CLOCAL = 1)
// tty.c_lflag &= ~ICANON;
tty.c_cflag |= ICANON;
if (!echo)
{
tty.c_lflag &= ~ECHO; // Disable echo
}
else
{
tty.c_lflag |= ECHO; // Enable echo
}
// tty.c_lflag &= ~ECHOE; // Disable erasure
// tty.c_lflag &= ~ECHONL; // Disable new-line echo
// tty.c_lflag &= ~ISIG; // Disable interpretation of INTR, QUIT and SUSP
tty.c_iflag &= ~(IXON | IXOFF | IXANY); // Turn off s/w flow ctrl
tty.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL); // Disable any special handling of received bytes
tty.c_oflag &= ~OPOST; // Prevent special interpretation of output bytes (e.g. newline chars)
tty.c_oflag &= ~ONLCR; // Prevent conversion of newline to carriage return/line feed
// Set in/out baud rate to be 115200
cfsetispeed(&tty, B115200);
cfsetospeed(&tty, B115200);
// Save tty settings, also checking for error
if (tcsetattr(m_serialPort, TCSANOW, &tty) != 0)
{
spdlog::error("[tSerial] error {} from tcsetattr: {}", errno, strerror(errno));
throw std::runtime_error("Failed to set new serial settings");
}
// Set non-blocking
int flags;
if ((flags = fcntl(m_serialPort, F_GETFL, 0)) == -1)
{
flags = 0;
}
fcntl(m_serialPort, F_SETFL, flags | O_NONBLOCK);
}
我就是这样写的。我正在使用 select
等待 IO 资源可用,因为尝试直接写入它会出现错误 Resource temporarily unavailable
。但是等了20秒,设备还是不可用
void tSerial::writeSerial(std::string message)
{
int n;
fd_set rfds;
FD_ZERO(&rfds);
FD_SET(m_serialPort, &rfds);
spdlog::debug("[tSerial] writing command to serial {}", message);
struct timeval tv;
tv.tv_sec = 20;
tv.tv_usec = 0;
int retval = select(1, NULL, &rfds, NULL, &tv);
if (retval == -1)
{
spdlog::error("[tSerial] select error");
}
else if (retval)
{
n = write(m_serialPort, message.c_str(), message.length());
tcflush(m_serialPort, TCIOFLUSH);
if (n < 0)
{
spdlog::error("[tSerial] error writing: {}", strerror(errno));
}
}
else
{
spdlog::error("[tSerial] Resource unavailable after 20 seconds wait");
}
}
I am using
select
to wait for the IO resource to become available ... But even waiting 20 seconds, the device is still unavailable.
您的程序未按预期运行,因为您没有正确编写 select() 调用:
int retval = select(1, NULL, &rfds, NULL, &tv);
根据 man 页面,第一个参数“ 应设置为三个集合中任何一个中编号最大的文件描述符,再加上 1。"
由于您正在传递值 1
,因此 select() 可以检查的唯一文件描述符是 stdin
(它是只读的并且永远不会准备好输出)并且永远不会(文件描述符)打开的串行终端。
相反,系统调用需要
select(m_serialPort + 1, ...);
您的代码还有其他问题。
(1) 非阻塞模式的使用值得怀疑。
似乎您更愿意向程序添加额外的代码以等待,而不是让 OS 为您完成。如果您的程序没有有效且高效地利用其时间片,那么您最好让 OS 管理系统资源。
(2) 在 write() 之后使用 tcflush() 是荒谬和错误的!
您可能打算使用 tcdrain().
研究 man 页面。
(3) 当您复制和修改代码时,您至少引入了一个错误。
ICANON
在 c_lflag
成员中,而不在 c_cflag
.
(4) 良好的编码习惯是使用有意义的变量名。
重用用于读取参数的变量名,因为写入参数草率且令人困惑:
fd_set rfds;
...
int retval = select(..., NULL, &rfds, NULL, &tv);
而不是 rfds
,您的程序应该使用 wfds
作为第三个参数。
... trying to write to it directly gives the error
Resource temporarily unavailable
使用select()时的问题很容易解释。
对于上述问题,您需要提供一个最小的、可重现的示例,即一个完整的程序。