处理高速数据时避免 QSerialPort 超时
Avoiding Timeout on QSerialPort when handling high speed data
我正在开发一个 windows 应用程序,它以 600Hz 的频率从传感器接收数据。在五分之二的情况下,我的 IO 线程成功地从传感器读取了 4 个字节的数据并将其传递给 GUI 线程。
问题是五分之三,QSerialPort 有莫名其妙的超时,其中 QSerialPort 的 waitForReadyRead() returns false 和 serial.errorString() 有超时错误。在这种情况下,它永远不会读取数据。如果我在超时错误的情况下从串行端口读取,我将在下一个 waitForReadyRead 中读取 2000 多个字节的数据,这些数据将以块的形式传送,这使得我的应用程序的实时数据接收方面已经过时。
我试过使用串行端口的 readyRead() 信号,但它表现出相同的行为,即。如果出现超时错误,则不会触发任何 readyRead() 信号。
更新: 我可以使用非阻塞读取的 Qt 终端示例 ([QT_INSTALL_EXAMPLES]/serialport/terminal) 重现该问题。该错误的频率要低得多,但它肯定仍然存在。
更新: 使用串行端口监视器,我可以看到当它卡住时,Qt 终端示例卡在 IOCTL_SERIAL_WAIT_ON_MASK,我的示例卡在 IRP_MJ_WRITE DOWN 就在 IOCT_SERIAL_WAIT_ON_MASK 之后。其他终端软件从未发生过这种情况,这使我认为问题肯定出在 Qt 上。
Pastebin of Serial Port Monitor Output
void IOThread::run(){
QSerialPort serial;
serial.setPortName(portname)
serial.setBaudRage(QSerialPort::Baud115200);
serial.setStopBits(QSerialPort::OneStop)
serial.setParity(QSerialPort::NoParity);
serial.setDataBits(QSerialPort::Data8);
serial.setFlowControl(QSerialPort::NoFlowControl);
if(!serial.open(QIODevice::ReadWrite)
{
qDebug() << "Error Opening Port";
return;
}
else
{
qDebug() << "Error Message: " << serial.errorString() // prints "Unknown Error"
}
while(true)
{
if(serial.waitForReadyRead(1000))
{
qDebug() << "Normal read";
reception_buffer = serial.readAll();
}
else
{
qDebug() << "Timeout";
/* serial.readAll() here will read nothing but force next read to read huge chunk of data */
continue;
}
}
// Process data...
}
尝试一下是否有任何不同:
while (true) {
QByteArray reception_buffer;
if (serial.waitForReadyRead(1000)) {
reception_buffer = serial.readAll();
while (serial.waitForReadyRead(10)) {
reception_buffer += serial.readAll();
}
qDebug() << "reception_buffer ready";
}
else {
qDebug() << "Timeout";
}
}
如果你想防止waitForReadyRead
超时,你可以设置:
if(serial.waitForReadyRead(-1))
bool QSerialPort::waitForReadyRead(int msecs = 30000)会在msecs毫秒后超时;默认超时为 30000 毫秒。如果msecs为-1,函数不会超时。
gets stuck on IOCTL_SERIAL_WAIT_ON_MASK
很可能是您的硬件或驱动程序出了问题。 QSP 使用异步通知,基于 WaitCommEvent。如果 WaitCommEvent 卡住 - 那么您的设备或驱动程序有问题(很可能)。
感谢 QSerialPort 的工作人员,Qt 5.10.1 中的这个错误通过应用此补丁得到解决:https://codereview.qt-project.org/#/c/225277/
"QSP may ignore all the read events when the data comes to the device
within opening. In this case, even re-opening of a device does not
help. Reason is that the QWinOverlappedIoNotifier is enabled after
than the startAsyncCommunication() called, that probably, leads to
ignoring for all EV_RXCHAR events. A workaround is to enable the
notifier before than any of I/O operation called." - Denis Shienkov, QSerialPort Maintainer
我正在开发一个 windows 应用程序,它以 600Hz 的频率从传感器接收数据。在五分之二的情况下,我的 IO 线程成功地从传感器读取了 4 个字节的数据并将其传递给 GUI 线程。
问题是五分之三,QSerialPort 有莫名其妙的超时,其中 QSerialPort 的 waitForReadyRead() returns false 和 serial.errorString() 有超时错误。在这种情况下,它永远不会读取数据。如果我在超时错误的情况下从串行端口读取,我将在下一个 waitForReadyRead 中读取 2000 多个字节的数据,这些数据将以块的形式传送,这使得我的应用程序的实时数据接收方面已经过时。
我试过使用串行端口的 readyRead() 信号,但它表现出相同的行为,即。如果出现超时错误,则不会触发任何 readyRead() 信号。
更新: 我可以使用非阻塞读取的 Qt 终端示例 ([QT_INSTALL_EXAMPLES]/serialport/terminal) 重现该问题。该错误的频率要低得多,但它肯定仍然存在。
更新: 使用串行端口监视器,我可以看到当它卡住时,Qt 终端示例卡在 IOCTL_SERIAL_WAIT_ON_MASK,我的示例卡在 IRP_MJ_WRITE DOWN 就在 IOCT_SERIAL_WAIT_ON_MASK 之后。其他终端软件从未发生过这种情况,这使我认为问题肯定出在 Qt 上。
Pastebin of Serial Port Monitor Output
void IOThread::run(){
QSerialPort serial;
serial.setPortName(portname)
serial.setBaudRage(QSerialPort::Baud115200);
serial.setStopBits(QSerialPort::OneStop)
serial.setParity(QSerialPort::NoParity);
serial.setDataBits(QSerialPort::Data8);
serial.setFlowControl(QSerialPort::NoFlowControl);
if(!serial.open(QIODevice::ReadWrite)
{
qDebug() << "Error Opening Port";
return;
}
else
{
qDebug() << "Error Message: " << serial.errorString() // prints "Unknown Error"
}
while(true)
{
if(serial.waitForReadyRead(1000))
{
qDebug() << "Normal read";
reception_buffer = serial.readAll();
}
else
{
qDebug() << "Timeout";
/* serial.readAll() here will read nothing but force next read to read huge chunk of data */
continue;
}
}
// Process data...
}
尝试一下是否有任何不同:
while (true) {
QByteArray reception_buffer;
if (serial.waitForReadyRead(1000)) {
reception_buffer = serial.readAll();
while (serial.waitForReadyRead(10)) {
reception_buffer += serial.readAll();
}
qDebug() << "reception_buffer ready";
}
else {
qDebug() << "Timeout";
}
}
如果你想防止waitForReadyRead
超时,你可以设置:
if(serial.waitForReadyRead(-1))
bool QSerialPort::waitForReadyRead(int msecs = 30000)会在msecs毫秒后超时;默认超时为 30000 毫秒。如果msecs为-1,函数不会超时。
gets stuck on IOCTL_SERIAL_WAIT_ON_MASK
很可能是您的硬件或驱动程序出了问题。 QSP 使用异步通知,基于 WaitCommEvent。如果 WaitCommEvent 卡住 - 那么您的设备或驱动程序有问题(很可能)。
感谢 QSerialPort 的工作人员,Qt 5.10.1 中的这个错误通过应用此补丁得到解决:https://codereview.qt-project.org/#/c/225277/
"QSP may ignore all the read events when the data comes to the device within opening. In this case, even re-opening of a device does not help. Reason is that the QWinOverlappedIoNotifier is enabled after than the startAsyncCommunication() called, that probably, leads to ignoring for all EV_RXCHAR events. A workaround is to enable the notifier before than any of I/O operation called." - Denis Shienkov, QSerialPort Maintainer