From 918f4af2c1f89f3e41162b12cce968bf0d3de739 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Thu, 18 Jun 2020 22:05:43 -0400 Subject: [PATCH] Add support for signals and serial port reading to XML replay. Replay now passes its initial regression test when the oximeter is unplugged. --- oscar/SleepLib/deviceconnection.cpp | 163 +++++++++++++++++++------- oscar/tests/deviceconnectiontests.cpp | 39 +++--- 2 files changed, 146 insertions(+), 56 deletions(-) diff --git a/oscar/SleepLib/deviceconnection.cpp b/oscar/SleepLib/deviceconnection.cpp index 09d21fd3..aabf2bd2 100644 --- a/oscar/SleepLib/deviceconnection.cpp +++ b/oscar/SleepLib/deviceconnection.cpp @@ -59,6 +59,13 @@ protected: QList m_events; XmlReplayEvent* getNextEvent(const QString & type, const QString & id = ""); + + XmlReplayEvent* m_pendingSignal; + QMutex m_lock; + inline void lock() { m_lock.lock(); } + inline void unlock() { m_lock.unlock(); } + void processPendingSignals(const QObject* target); + friend class XmlReplayLock; }; class XmlReplayEvent @@ -79,8 +86,10 @@ public: void set(const QString & name, const QString & value) { + if (!m_values.contains(name)) { + m_keys.append(name); + } m_values[name] = value; - m_keys.append(name); } void set(const QString & name, qint64 value) { @@ -89,11 +98,8 @@ public: void setData(const char* data, qint64 length) { Q_ASSERT(usesData() == true); - QStringList bytes; - for (qint64 i = 0; i < length; i++) { - bytes.append(QString("%1").arg((unsigned char) data[i], 2, 16, QChar('0')).toUpper()); - } - m_data = bytes.join(QChar(' ')); + QByteArray bytes = QByteArray::fromRawData(data, length); + m_data = bytes.toHex(' ').toUpper(); } inline QString get(const QString & name) const { @@ -105,17 +111,7 @@ public: QByteArray getData() const { Q_ASSERT(usesData() == true); - QByteArray data; - QStringList bytes = m_data.split(" "); - data.reserve(bytes.size()); - for (auto & b : bytes) { - bool ok; - data.append((char) b.toShort(&ok, 16)); - if (!ok) { - qWarning() << "xml tag" << tag() << "has invalid data:" << b; - } - } - return data; + return QByteArray::fromHex(m_data.toUtf8()); } inline bool ok() const { return m_values.contains("error") == false; } operator QString() const @@ -144,6 +140,13 @@ protected: QDateTime m_time; XmlReplayEvent* m_next; + const char* m_signal; + inline bool isSignal() const { return m_signal != nullptr; } + virtual void signal(QObject* target) + { + QMetaObject::invokeMethod(target, m_signal, Qt::QueuedConnection); + } + QHash m_values; QList m_keys; QString m_data; @@ -178,6 +181,28 @@ protected: }; QHash XmlReplayEvent::s_factories; +class XmlReplayLock +{ +public: + XmlReplayLock(const QObject* obj, XmlReplay* replay) + : m_target(obj), m_replay(replay) + { + if (m_replay) { + m_replay->lock(); + } + } + ~XmlReplayLock() + { + if (m_replay) { + m_replay->processPendingSignals(m_target); + m_replay->unlock(); + } + } + +protected: + const QObject* m_target; + XmlReplay* m_replay; +}; XmlRecorder::XmlRecorder(QFile* stream) : m_file(stream), m_xml(new QXmlStreamWriter(stream)) @@ -216,12 +241,14 @@ void XmlRecorder::epilogue() } XmlReplay::XmlReplay(QFile* file) + : m_pendingSignal(nullptr) { QXmlStreamReader xml(file); deserialize(xml); } XmlReplay::XmlReplay(QXmlStreamReader & xml) + : m_pendingSignal(nullptr) { deserialize(xml); } @@ -274,10 +301,34 @@ void XmlReplay::deserializeEvents(QXmlStreamReader & xml) } } +void XmlReplay::processPendingSignals(const QObject* target) +{ + if (m_pendingSignal) { + // It is safe to re-cast this as non-const because signals are deferred + // and cannot alter the underlying target until the const method holding + // the lock releases it at function exit. + m_pendingSignal->signal(const_cast(target)); + + XmlReplayEvent* next = m_pendingSignal->m_next; + if (next && next->isSignal() == false) { + next = nullptr; + } + if (next) { + qDebug() << "UNTESTED: multiple signal events in a row:" << m_pendingSignal->tag() << next->tag(); + } + m_pendingSignal = next; + } +} + XmlReplayEvent* XmlReplay::getNextEvent(const QString & type, const QString & id) { XmlReplayEvent* event = nullptr; + if (m_lock.tryLock()) { + qWarning() << "XML replay" << type << "object not locked by event handler!"; + m_lock.unlock(); + } + if (m_eventIndex.contains(type)) { auto & ids = m_eventIndex[type]; if (ids.contains(id)) { @@ -290,6 +341,12 @@ XmlReplayEvent* XmlReplay::getNextEvent(const QString & type, const QString & id } } } + + if (event && event->m_next && event->m_next->isSignal()) { + Q_ASSERT(m_pendingSignal == nullptr); // if this ever fails, we may need m_pendingSignal to be a list + m_pendingSignal = event->m_next; + } + return event; } @@ -305,7 +362,7 @@ T* XmlReplay::getNextEvent(const QString & id) // MARK: XML record/playback event base class XmlReplayEvent::XmlReplayEvent() - : m_time(QDateTime::currentDateTime()), m_next(nullptr) + : m_time(QDateTime::currentDateTime()), m_next(nullptr), m_signal(nullptr) { } @@ -571,6 +628,7 @@ REGISTER_XMLREPLAYEVENT("getAvailableSerialPorts", GetAvailableSerialPortsEvent) QList DeviceConnectionManager::getAvailableSerialPorts() { + XmlReplayLock lock(this, m_replay); GetAvailableSerialPortsEvent event; if (!m_replay) { @@ -741,20 +799,6 @@ public: }; REGISTER_XMLREPLAYEVENT("set", SetValueEvent); -/* -ConnectionEvent::operator QString() const -{ -} - -QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const ConnectionEvent & event) -{ - for (auto key : event.m_keys) { - } - if (!event.m_data.isEmpty()) { - } -} -*/ - class GetValueEvent : public XmlReplayBase { public: @@ -833,6 +877,13 @@ public: }; REGISTER_XMLREPLAYEVENT("tx", TransmitDataEvent); +class ReadyReadEvent : public XmlReplayBase +{ +public: + ReadyReadEvent() { m_signal = "onReadyRead"; } +}; +REGISTER_XMLREPLAYEVENT("readyRead", ReadyReadEvent); + // MARK: - // MARK: Serial port connection @@ -860,6 +911,7 @@ bool SerialPortConnection::open() qWarning() << "serial connection to" << m_name << "already opened"; return false; } + XmlReplayLock lock(this, m_replay); OpenConnectionEvent event("serial", m_name); if (!m_replay) { @@ -884,6 +936,7 @@ bool SerialPortConnection::open() bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions directions) { + XmlReplayLock lock(this, m_replay); SetValueEvent event("baudRate", baudRate); event.set("directions", directions); @@ -900,6 +953,7 @@ bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits) { + XmlReplayLock lock(this, m_replay); SetValueEvent event("setDataBits", dataBits); if (!m_replay) { @@ -915,6 +969,7 @@ bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits) bool SerialPortConnection::setParity(QSerialPort::Parity parity) { + XmlReplayLock lock(this, m_replay); SetValueEvent event("setParity", parity); if (!m_replay) { @@ -930,6 +985,7 @@ bool SerialPortConnection::setParity(QSerialPort::Parity parity) bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits) { + XmlReplayLock lock(this, m_replay); SetValueEvent event("setStopBits", stopBits); if (!m_replay) { @@ -945,6 +1001,7 @@ bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits) bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl) { + XmlReplayLock lock(this, m_replay); SetValueEvent event("setFlowControl", flowControl); if (!m_replay) { @@ -960,6 +1017,7 @@ bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl) bool SerialPortConnection::clear(QSerialPort::Directions directions) { + XmlReplayLock lock(this, m_replay); ClearConnectionEvent event; event.set("directions", directions); @@ -976,6 +1034,7 @@ bool SerialPortConnection::clear(QSerialPort::Directions directions) qint64 SerialPortConnection::bytesAvailable() const { + XmlReplayLock lock(this, m_replay); GetValueEvent event("bytesAvailable"); qint64 result; @@ -1000,6 +1059,7 @@ qint64 SerialPortConnection::bytesAvailable() const qint64 SerialPortConnection::read(char *data, qint64 maxSize) { + XmlReplayLock lock(this, m_replay); qint64 len; ReceiveDataEvent event; @@ -1015,7 +1075,7 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize) checkResult(len, event); } else { // TODO: this should chain off the most recent write's and readyRead's m_next - auto replayEvent = m_replay->getNextEvent(event.id()); + auto replayEvent = m_replay->getNextEvent(); event.copyIf(replayEvent); if (!replayEvent) { qWarning() << "reading data past replay"; @@ -1025,11 +1085,23 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize) bool ok; len = event.get("len").toLong(&ok); - if (!ok) { + if (ok) { + if (event.ok()) { + if (len != maxSize) { + qWarning() << "replay of" << len << "bytes but" << maxSize << "requested"; + } + if (len > maxSize) { + len = maxSize; + } + QByteArray replayData = event.getData(); + memcpy(data, replayData, len); + } + } else { qWarning() << event << "has bad len"; len = -1; } } + event.record(m_record); qDebug().noquote() << event; return len; @@ -1037,6 +1109,7 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize) qint64 SerialPortConnection::write(const char *data, qint64 maxSize) { + XmlReplayLock lock(this, m_replay); qint64 len; TransmitDataEvent event; event.setData(data, maxSize); @@ -1071,6 +1144,7 @@ qint64 SerialPortConnection::write(const char *data, qint64 maxSize) bool SerialPortConnection::flush() { + XmlReplayLock lock(this, m_replay); FlushConnectionEvent event; if (!m_replay) { @@ -1086,6 +1160,7 @@ bool SerialPortConnection::flush() void SerialPortConnection::close() { + XmlReplayLock lock(this, m_replay); CloseConnectionEvent event("serial", m_name); // TODO: the separate connection stream will have an enclosing "connection" tag with these @@ -1113,14 +1188,22 @@ void SerialPortConnection::close() void SerialPortConnection::onReadyRead() { - ConnectionEvent event("readyRead"); + { + // Wait until the replay signaler (if any) has released its lock. + XmlReplayLock lock(this, m_replay); + + // This needs to be recorded before the signal below, since the slot may trigger more events. + ReadyReadEvent event; + event.record(m_record); + qDebug().noquote() << event; + } - // TODO: Most of the playback API reponds to the caller. How do we replay port-driven events? - // Probably add an ordered linked list of events, a peekNextEvent, getNextEvent(void), - // and event->replay() method that calls the appropriate method. (May as well have the - // destructor walk the links list rather than the per-type lists.) - qDebug().noquote() << event; + // Because clients typically leave this as Qt::AutoConnection, the below emit may + // execute immediately in this thread, so we have to release the lock before sending + // the signal. + // Unlike client-called events, We don't need to handle replay differently here, + // because the replay will signal this slot just like the serial port. emit readyRead(); } diff --git a/oscar/tests/deviceconnectiontests.cpp b/oscar/tests/deviceconnectiontests.cpp index 7b0cd890..5ebe4962 100644 --- a/oscar/tests/deviceconnectiontests.cpp +++ b/oscar/tests/deviceconnectiontests.cpp @@ -106,6 +106,7 @@ void DeviceConnectionTests::testOximeterConnection() DeviceConnectionManager & devices = DeviceConnectionManager::getInstance(); devices.record(string); + /* // new API QString portName = "cu.SLAB_USBtoUART"; { @@ -129,25 +130,30 @@ void DeviceConnectionTests::testOximeterConnection() devices.record(nullptr); qDebug().noquote() << string; - string = ""; - - devices.record(string); + */ + SerialOximeter * oxi = qobject_cast(lookupLoader(cms50f37_class_name)); Q_ASSERT(oxi); - if (oxi->openDevice()) { - oxi->resetDevice(); - int session_count = oxi->getSessionCount(); - qDebug() << session_count; + + QFile file("cms50f37.xml"); + if (!file.exists()) { + qDebug() << "Recording oximeter connection"; + Q_ASSERT(file.open(QFile::ReadWrite)); + devices.record(&file); + if (oxi->openDevice()) { + oxi->resetDevice(); + int session_count = oxi->getSessionCount(); + qDebug() << session_count; + } + oxi->closeDevice(); + oxi->trashRecords(); + devices.record(nullptr); + file.close(); } - oxi->closeDevice(); - oxi->trashRecords(); - devices.record(nullptr); - - qDebug().noquote() << string; - - devices.replay(string); - oxi = qobject_cast(lookupLoader(cms50f37_class_name)); - Q_ASSERT(oxi); + + qDebug() << "Replaying oximeter connection"; + Q_ASSERT(file.open(QFile::ReadOnly)); + devices.replay(&file); if (oxi->openDevice()) { oxi->resetDevice(); int session_count = oxi->getSessionCount(); @@ -156,4 +162,5 @@ void DeviceConnectionTests::testOximeterConnection() oxi->closeDevice(); oxi->trashRecords(); devices.replay(nullptr); + file.close(); }