mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-06 03:00:43 +00:00
Add support for signals and serial port reading to XML replay.
Replay now passes its initial regression test when the oximeter is unplugged.
This commit is contained in:
parent
5947751291
commit
918f4af2c1
@ -59,6 +59,13 @@ protected:
|
|||||||
QList<XmlReplayEvent*> m_events;
|
QList<XmlReplayEvent*> m_events;
|
||||||
|
|
||||||
XmlReplayEvent* getNextEvent(const QString & type, const QString & id = "");
|
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
|
class XmlReplayEvent
|
||||||
@ -79,8 +86,10 @@ public:
|
|||||||
|
|
||||||
void set(const QString & name, const QString & value)
|
void set(const QString & name, const QString & value)
|
||||||
{
|
{
|
||||||
|
if (!m_values.contains(name)) {
|
||||||
|
m_keys.append(name);
|
||||||
|
}
|
||||||
m_values[name] = value;
|
m_values[name] = value;
|
||||||
m_keys.append(name);
|
|
||||||
}
|
}
|
||||||
void set(const QString & name, qint64 value)
|
void set(const QString & name, qint64 value)
|
||||||
{
|
{
|
||||||
@ -89,11 +98,8 @@ public:
|
|||||||
void setData(const char* data, qint64 length)
|
void setData(const char* data, qint64 length)
|
||||||
{
|
{
|
||||||
Q_ASSERT(usesData() == true);
|
Q_ASSERT(usesData() == true);
|
||||||
QStringList bytes;
|
QByteArray bytes = QByteArray::fromRawData(data, length);
|
||||||
for (qint64 i = 0; i < length; i++) {
|
m_data = bytes.toHex(' ').toUpper();
|
||||||
bytes.append(QString("%1").arg((unsigned char) data[i], 2, 16, QChar('0')).toUpper());
|
|
||||||
}
|
|
||||||
m_data = bytes.join(QChar(' '));
|
|
||||||
}
|
}
|
||||||
inline QString get(const QString & name) const
|
inline QString get(const QString & name) const
|
||||||
{
|
{
|
||||||
@ -105,17 +111,7 @@ public:
|
|||||||
QByteArray getData() const
|
QByteArray getData() const
|
||||||
{
|
{
|
||||||
Q_ASSERT(usesData() == true);
|
Q_ASSERT(usesData() == true);
|
||||||
QByteArray data;
|
return QByteArray::fromHex(m_data.toUtf8());
|
||||||
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;
|
|
||||||
}
|
}
|
||||||
inline bool ok() const { return m_values.contains("error") == false; }
|
inline bool ok() const { return m_values.contains("error") == false; }
|
||||||
operator QString() const
|
operator QString() const
|
||||||
@ -144,6 +140,13 @@ protected:
|
|||||||
QDateTime m_time;
|
QDateTime m_time;
|
||||||
XmlReplayEvent* m_next;
|
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<QString,QString> m_values;
|
QHash<QString,QString> m_values;
|
||||||
QList<QString> m_keys;
|
QList<QString> m_keys;
|
||||||
QString m_data;
|
QString m_data;
|
||||||
@ -178,6 +181,28 @@ protected:
|
|||||||
};
|
};
|
||||||
QHash<QString,XmlReplayEvent::FactoryMethod> XmlReplayEvent::s_factories;
|
QHash<QString,XmlReplayEvent::FactoryMethod> 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)
|
XmlRecorder::XmlRecorder(QFile* stream)
|
||||||
: m_file(stream), m_xml(new QXmlStreamWriter(stream))
|
: m_file(stream), m_xml(new QXmlStreamWriter(stream))
|
||||||
@ -216,12 +241,14 @@ void XmlRecorder::epilogue()
|
|||||||
}
|
}
|
||||||
|
|
||||||
XmlReplay::XmlReplay(QFile* file)
|
XmlReplay::XmlReplay(QFile* file)
|
||||||
|
: m_pendingSignal(nullptr)
|
||||||
{
|
{
|
||||||
QXmlStreamReader xml(file);
|
QXmlStreamReader xml(file);
|
||||||
deserialize(xml);
|
deserialize(xml);
|
||||||
}
|
}
|
||||||
|
|
||||||
XmlReplay::XmlReplay(QXmlStreamReader & xml)
|
XmlReplay::XmlReplay(QXmlStreamReader & xml)
|
||||||
|
: m_pendingSignal(nullptr)
|
||||||
{
|
{
|
||||||
deserialize(xml);
|
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<QObject*>(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* XmlReplay::getNextEvent(const QString & type, const QString & id)
|
||||||
{
|
{
|
||||||
XmlReplayEvent* event = nullptr;
|
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)) {
|
if (m_eventIndex.contains(type)) {
|
||||||
auto & ids = m_eventIndex[type];
|
auto & ids = m_eventIndex[type];
|
||||||
if (ids.contains(id)) {
|
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;
|
return event;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -305,7 +362,7 @@ T* XmlReplay::getNextEvent(const QString & id)
|
|||||||
// MARK: XML record/playback event base class
|
// MARK: XML record/playback event base class
|
||||||
|
|
||||||
XmlReplayEvent::XmlReplayEvent()
|
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<SerialPortInfo> DeviceConnectionManager::getAvailableSerialPorts()
|
QList<SerialPortInfo> DeviceConnectionManager::getAvailableSerialPorts()
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
GetAvailableSerialPortsEvent event;
|
GetAvailableSerialPortsEvent event;
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -741,20 +799,6 @@ public:
|
|||||||
};
|
};
|
||||||
REGISTER_XMLREPLAYEVENT("set", SetValueEvent);
|
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<GetValueEvent>
|
class GetValueEvent : public XmlReplayBase<GetValueEvent>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
@ -833,6 +877,13 @@ public:
|
|||||||
};
|
};
|
||||||
REGISTER_XMLREPLAYEVENT("tx", TransmitDataEvent);
|
REGISTER_XMLREPLAYEVENT("tx", TransmitDataEvent);
|
||||||
|
|
||||||
|
class ReadyReadEvent : public XmlReplayBase<ReadyReadEvent>
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
ReadyReadEvent() { m_signal = "onReadyRead"; }
|
||||||
|
};
|
||||||
|
REGISTER_XMLREPLAYEVENT("readyRead", ReadyReadEvent);
|
||||||
|
|
||||||
|
|
||||||
// MARK: -
|
// MARK: -
|
||||||
// MARK: Serial port connection
|
// MARK: Serial port connection
|
||||||
@ -860,6 +911,7 @@ bool SerialPortConnection::open()
|
|||||||
qWarning() << "serial connection to" << m_name << "already opened";
|
qWarning() << "serial connection to" << m_name << "already opened";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
OpenConnectionEvent event("serial", m_name);
|
OpenConnectionEvent event("serial", m_name);
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -884,6 +936,7 @@ bool SerialPortConnection::open()
|
|||||||
|
|
||||||
bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions directions)
|
bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions directions)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
SetValueEvent event("baudRate", baudRate);
|
SetValueEvent event("baudRate", baudRate);
|
||||||
event.set("directions", directions);
|
event.set("directions", directions);
|
||||||
|
|
||||||
@ -900,6 +953,7 @@ bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions
|
|||||||
|
|
||||||
bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits)
|
bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
SetValueEvent event("setDataBits", dataBits);
|
SetValueEvent event("setDataBits", dataBits);
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -915,6 +969,7 @@ bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits)
|
|||||||
|
|
||||||
bool SerialPortConnection::setParity(QSerialPort::Parity parity)
|
bool SerialPortConnection::setParity(QSerialPort::Parity parity)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
SetValueEvent event("setParity", parity);
|
SetValueEvent event("setParity", parity);
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -930,6 +985,7 @@ bool SerialPortConnection::setParity(QSerialPort::Parity parity)
|
|||||||
|
|
||||||
bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits)
|
bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
SetValueEvent event("setStopBits", stopBits);
|
SetValueEvent event("setStopBits", stopBits);
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -945,6 +1001,7 @@ bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits)
|
|||||||
|
|
||||||
bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl)
|
bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
SetValueEvent event("setFlowControl", flowControl);
|
SetValueEvent event("setFlowControl", flowControl);
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -960,6 +1017,7 @@ bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl)
|
|||||||
|
|
||||||
bool SerialPortConnection::clear(QSerialPort::Directions directions)
|
bool SerialPortConnection::clear(QSerialPort::Directions directions)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
ClearConnectionEvent event;
|
ClearConnectionEvent event;
|
||||||
event.set("directions", directions);
|
event.set("directions", directions);
|
||||||
|
|
||||||
@ -976,6 +1034,7 @@ bool SerialPortConnection::clear(QSerialPort::Directions directions)
|
|||||||
|
|
||||||
qint64 SerialPortConnection::bytesAvailable() const
|
qint64 SerialPortConnection::bytesAvailable() const
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
GetValueEvent event("bytesAvailable");
|
GetValueEvent event("bytesAvailable");
|
||||||
qint64 result;
|
qint64 result;
|
||||||
|
|
||||||
@ -1000,6 +1059,7 @@ qint64 SerialPortConnection::bytesAvailable() const
|
|||||||
|
|
||||||
qint64 SerialPortConnection::read(char *data, qint64 maxSize)
|
qint64 SerialPortConnection::read(char *data, qint64 maxSize)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
qint64 len;
|
qint64 len;
|
||||||
ReceiveDataEvent event;
|
ReceiveDataEvent event;
|
||||||
|
|
||||||
@ -1015,7 +1075,7 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize)
|
|||||||
checkResult(len, event);
|
checkResult(len, event);
|
||||||
} else {
|
} else {
|
||||||
// TODO: this should chain off the most recent write's and readyRead's m_next
|
// TODO: this should chain off the most recent write's and readyRead's m_next
|
||||||
auto replayEvent = m_replay->getNextEvent<ReceiveDataEvent>(event.id());
|
auto replayEvent = m_replay->getNextEvent<ReceiveDataEvent>();
|
||||||
event.copyIf(replayEvent);
|
event.copyIf(replayEvent);
|
||||||
if (!replayEvent) {
|
if (!replayEvent) {
|
||||||
qWarning() << "reading data past replay";
|
qWarning() << "reading data past replay";
|
||||||
@ -1025,11 +1085,23 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize)
|
|||||||
|
|
||||||
bool ok;
|
bool ok;
|
||||||
len = event.get("len").toLong(&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";
|
qWarning() << event << "has bad len";
|
||||||
len = -1;
|
len = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
event.record(m_record);
|
||||||
qDebug().noquote() << event;
|
qDebug().noquote() << event;
|
||||||
|
|
||||||
return len;
|
return len;
|
||||||
@ -1037,6 +1109,7 @@ qint64 SerialPortConnection::read(char *data, qint64 maxSize)
|
|||||||
|
|
||||||
qint64 SerialPortConnection::write(const char *data, qint64 maxSize)
|
qint64 SerialPortConnection::write(const char *data, qint64 maxSize)
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
qint64 len;
|
qint64 len;
|
||||||
TransmitDataEvent event;
|
TransmitDataEvent event;
|
||||||
event.setData(data, maxSize);
|
event.setData(data, maxSize);
|
||||||
@ -1071,6 +1144,7 @@ qint64 SerialPortConnection::write(const char *data, qint64 maxSize)
|
|||||||
|
|
||||||
bool SerialPortConnection::flush()
|
bool SerialPortConnection::flush()
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
FlushConnectionEvent event;
|
FlushConnectionEvent event;
|
||||||
|
|
||||||
if (!m_replay) {
|
if (!m_replay) {
|
||||||
@ -1086,6 +1160,7 @@ bool SerialPortConnection::flush()
|
|||||||
|
|
||||||
void SerialPortConnection::close()
|
void SerialPortConnection::close()
|
||||||
{
|
{
|
||||||
|
XmlReplayLock lock(this, m_replay);
|
||||||
CloseConnectionEvent event("serial", m_name);
|
CloseConnectionEvent event("serial", m_name);
|
||||||
|
|
||||||
// TODO: the separate connection stream will have an enclosing "connection" tag with these
|
// TODO: the separate connection stream will have an enclosing "connection" tag with these
|
||||||
@ -1113,14 +1188,22 @@ void SerialPortConnection::close()
|
|||||||
|
|
||||||
void SerialPortConnection::onReadyRead()
|
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?
|
// Because clients typically leave this as Qt::AutoConnection, the below emit may
|
||||||
// Probably add an ordered linked list of events, a peekNextEvent, getNextEvent(void),
|
// execute immediately in this thread, so we have to release the lock before sending
|
||||||
// and event->replay() method that calls the appropriate method. (May as well have the
|
// the signal.
|
||||||
// destructor walk the links list rather than the per-type lists.)
|
|
||||||
qDebug().noquote() << event;
|
|
||||||
|
|
||||||
|
// 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();
|
emit readyRead();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -106,6 +106,7 @@ void DeviceConnectionTests::testOximeterConnection()
|
|||||||
DeviceConnectionManager & devices = DeviceConnectionManager::getInstance();
|
DeviceConnectionManager & devices = DeviceConnectionManager::getInstance();
|
||||||
devices.record(string);
|
devices.record(string);
|
||||||
|
|
||||||
|
/*
|
||||||
// new API
|
// new API
|
||||||
QString portName = "cu.SLAB_USBtoUART";
|
QString portName = "cu.SLAB_USBtoUART";
|
||||||
{
|
{
|
||||||
@ -129,25 +130,30 @@ void DeviceConnectionTests::testOximeterConnection()
|
|||||||
devices.record(nullptr);
|
devices.record(nullptr);
|
||||||
|
|
||||||
qDebug().noquote() << string;
|
qDebug().noquote() << string;
|
||||||
string = "";
|
*/
|
||||||
|
|
||||||
devices.record(string);
|
|
||||||
SerialOximeter * oxi = qobject_cast<SerialOximeter*>(lookupLoader(cms50f37_class_name));
|
SerialOximeter * oxi = qobject_cast<SerialOximeter*>(lookupLoader(cms50f37_class_name));
|
||||||
Q_ASSERT(oxi);
|
Q_ASSERT(oxi);
|
||||||
if (oxi->openDevice()) {
|
|
||||||
oxi->resetDevice();
|
QFile file("cms50f37.xml");
|
||||||
int session_count = oxi->getSessionCount();
|
if (!file.exists()) {
|
||||||
qDebug() << session_count;
|
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();
|
qDebug() << "Replaying oximeter connection";
|
||||||
devices.record(nullptr);
|
Q_ASSERT(file.open(QFile::ReadOnly));
|
||||||
|
devices.replay(&file);
|
||||||
qDebug().noquote() << string;
|
|
||||||
|
|
||||||
devices.replay(string);
|
|
||||||
oxi = qobject_cast<SerialOximeter*>(lookupLoader(cms50f37_class_name));
|
|
||||||
Q_ASSERT(oxi);
|
|
||||||
if (oxi->openDevice()) {
|
if (oxi->openDevice()) {
|
||||||
oxi->resetDevice();
|
oxi->resetDevice();
|
||||||
int session_count = oxi->getSessionCount();
|
int session_count = oxi->getSessionCount();
|
||||||
@ -156,4 +162,5 @@ void DeviceConnectionTests::testOximeterConnection()
|
|||||||
oxi->closeDevice();
|
oxi->closeDevice();
|
||||||
oxi->trashRecords();
|
oxi->trashRecords();
|
||||||
devices.replay(nullptr);
|
devices.replay(nullptr);
|
||||||
|
file.close();
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user