I have a QthreadPool to connect to upto 5 serial devices at the same time. each serial device is defined as:
struct serialDevice
{
std::shared_ptr<QSerialPort> port;
QByteArray portData;
};
Inside a GUI and upon clicking a button, I call a serial device manager serialManager->startAcquisition(portNames)
as follows:
#include "serialmanager.h"
serialManager::serialManager(QObject *parent) : QObject(parent)
{
pool = new QThreadPool();
pool->setMaxThreadCount(5);
}
void serialManager::startAcquisition(QStringList pNames)
{
foreach (QString port, pNames)
{
this->stablishConnection(port);
}
}
void serialManager::stablishConnection(QString pName)
{
ComPort *workerPort = new ComPort();
workerPort->setAutoDelete(true);
workerPort->setName(pName);
connect(this, SIGNAL(finished()), workerPort, SLOT(onTaskeFinished()));
pool->start(workerPort);
}
Once a device is connected I poll the device. The thread terminates when a stop function is called.
I expect that readyRead()
signal is being emitted, in order to fill inside the sensor->portData
:
#include "serialmanager.h"
#include <QEventLoop>
ComPort::ComPort(QObject *parent) : QObject(parent)
{
}
void ComPort::run()
{
QEventLoop loop;
connect(this, SIGNAL(finished()), &loop, SLOT(quit()));
sensor = std::make_shared<serialDevice> ();
sensor->port = std::make_shared<QSerialPort> ();
sensor->port->setPortName(this->portName);
sensor->port->setBaudRate(QSerialPort::Baud19200);
sensor->port->setDataBits(QSerialPort::Data8);
sensor->port->setFlowControl(QSerialPort::NoFlowControl);
sensor->port->setParity(QSerialPort::NoParity);
sensor->port->setStopBits(QSerialPort::OneStop);
QString strPort;
if (sensor->port->open(QIODevice::ReadWrite) == true)
{
connect(sensor->port.get(), SIGNAL(error(QSerialPort::SerialPortError)), this, SLOT(onErrorReceived()));
connect(sensor->port.get(), SIGNAL(readyRead()), this, SLOT(onReadyRead()));
}
else
{
strPort = "> " + sensor->port->portName() + " device is not streaming: " + sensor->port->errorString();
emit this->newMsgAvailable(strPort);
emit this->finished();
}
loop.exec();
}
void ComPort::onReadyRead()
{
QByteArray bytes = this->sensor->port->readAll();
this->sensor->portData.append(bytes);
}
void ComPort::onTaskeFinished()
{
QString strPort = "> Finished streaming at " + sensor->port->portName();
emit newMsgAvailable(strPort);
emit this->finished();
}
void ComPort::onErrorReceived()
{
QString strPort = "> Data stream on " + sensor->port->portName() + " failed: " + sensor->port->errorString();
emit newMsgAvailable(strPort);
}
Problem definition:
The readyRead()
signal was not emitted. I have read here, that to process events in a QRunnable
, a thread needs to have its own event loop. So I included QEventLoop loop
to the Run()
function of each ComPort
. Now readyRead()
is being triggered and I receive data from each port for a while, and then all of a sudden the application crashes with a segmentation fault as:
0 VerifierDisableFaultInjectionExclusionRange C:\WINDOWS\syswow64\verifier.dll 0x53457378
1 VerifierDisableFaultInjectionExclusionRange C:\WINDOWS\syswow64\verifier.dll 0x53457495
2 VerifierDisableFaultInjectionExclusionRange C:\WINDOWS\syswow64\verifier.dll 0x5345b651
3 VerifierCheckPageHeapAllocation C:\WINDOWS\syswow64\verifier.dll 0x53460b12
4 ?? C:\WINDOWS\SysWOW64\vfbasics.dll 0x533d179f
5 msvcrt!free C:\WINDOWS\SysWOW64\msvcrt.dll 0x7509b0f9
6 ?? 0x19120000
7 ?? C:\WINDOWS\SysWOW64\vfbasics.dll 0x533d26a5
8 QArrayData::deallocate 130
9 QTypedArrayData<char>::deallocate 234 0x6b9cbcfd
10 QByteArray::resize 1448 0x6b79bc23
11 QRingBuffer::append 383 0x645893ae
12 QSerialPortPrivate::_q_completeAsyncRead 529 0x6458523f
13 QSerialPort::qt_static_metacall 353 0x64582dae
14 QMetaObject::activate 3681 0x6b955027
15 QMetaObject::activate 3547 0x6b95498c
16 QWinEventNotifier::activated 134 0x6b9ad529
17 QWinEventNotifier::event 241 0x6b976af1
18 QApplicationPrivate::notify_helper 3499 0xf33df61
19 QApplication::notify 2952 0xf33ba13
20 QCoreApplication::notifyInternal 935 0x6b929eee
... <More>