From 3210ecd1bad2a99c4efca5d20c44ae2d30791976 Mon Sep 17 00:00:00 2001 From: sawinglogz <3787776-sawinglogz@users.noreply.gitlab.com> Date: Thu, 11 Jun 2020 22:29:46 -0400 Subject: [PATCH] Add plumbing for intercepting and logging serial port activity. --- oscar/SleepLib/deviceconnection.cpp | 178 +++++++++++++++++++++++++++- oscar/SleepLib/deviceconnection.h | 31 ++++- 2 files changed, 207 insertions(+), 2 deletions(-) diff --git a/oscar/SleepLib/deviceconnection.cpp b/oscar/SleepLib/deviceconnection.cpp index 2cc6615a..bbe0d7b9 100644 --- a/oscar/SleepLib/deviceconnection.cpp +++ b/oscar/SleepLib/deviceconnection.cpp @@ -526,5 +526,181 @@ bool SerialPortInfo::operator==(const SerialPortInfo & other) const // MARK: - // MARK: Serial port connection -// TODO +// TODO: log these to XML + +class SetValueEvent +{ +public: + SetValueEvent(const QString & name, int value) + { + set(name, value); + } + void set(const QString & name, int value) + { + m_values[name] = value; + m_keys.append(name); + } + inline bool ok() const { return m_values.contains("error") == false; } + operator QString() const; + +protected: + QHash m_values; + QList m_keys; + friend QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const SetValueEvent & event); +}; + +SetValueEvent::operator QString() const +{ + QString out; + QXmlStreamWriter xml(&out); + xml << *this; + return out; +} + +QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const SetValueEvent & event) +{ + xml.writeStartElement("set"); + for (auto key : event.m_keys) { + xml.writeAttribute(key, QString::number(event.m_values[key])); + } + xml.writeEndElement(); + return xml; +} + + +SerialPort::SerialPort() +{ + connect(&m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); +} + +SerialPort::~SerialPort() +{ + disconnect(&m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); +} + +void SerialPort::setPortName(const QString &name) +{ + qDebug() << ""; + return m_port.setPortName(name); +} + +bool SerialPort::open(QIODevice::OpenMode mode) +{ + qDebug() << ""; + return m_port.open(mode); +} + +bool SerialPort::setBaudRate(qint32 baudRate, QSerialPort::Directions directions) +{ + SetValueEvent event("baudRate", baudRate); + event.set("directions", directions); + + bool ok = m_port.setBaudRate(baudRate, directions); + if (!ok) { + QSerialPort::SerialPortError error = m_port.error(); + event.set("error", error); + } + qDebug().noquote() << event; + + return event.ok(); +} +// TODO: + +bool SerialPort::setDataBits(QSerialPort::DataBits dataBits) +{ + SetValueEvent event("setDataBits", dataBits); + + bool ok = m_port.setDataBits(dataBits); + if (!ok) { + QSerialPort::SerialPortError error = m_port.error(); + event.set("error", error); + } + qDebug().noquote() << event; + + return event.ok(); +} + +bool SerialPort::setParity(QSerialPort::Parity parity) +{ + SetValueEvent event("setParity", parity); + + bool ok = m_port.setParity(parity); + if (!ok) { + QSerialPort::SerialPortError error = m_port.error(); + event.set("error", error); + } + qDebug().noquote() << event; + + return event.ok(); +} + +bool SerialPort::setStopBits(QSerialPort::StopBits stopBits) +{ + SetValueEvent event("setStopBits", stopBits); + + bool ok = m_port.setStopBits(stopBits); + if (!ok) { + QSerialPort::SerialPortError error = m_port.error(); + event.set("error", error); + } + qDebug().noquote() << event; + + return event.ok(); +} + +bool SerialPort::setFlowControl(QSerialPort::FlowControl flowControl) +{ + SetValueEvent event("setFlowControl", flowControl); + + bool ok = m_port.setFlowControl(flowControl); + if (!ok) { + QSerialPort::SerialPortError error = m_port.error(); + event.set("error", error); + } + qDebug().noquote() << event; + + return event.ok(); +} + +bool SerialPort::clear(QSerialPort::Directions directions) +{ + qDebug() << ""; + return m_port.clear(directions); +} + +qint64 SerialPort::bytesAvailable() const +{ + qDebug() << ""; + return m_port.bytesAvailable(); +} + +qint64 SerialPort::read(char *data, qint64 maxSize) +{ + qDebug() << ""; + return m_port.read(data, maxSize); +} + +qint64 SerialPort::write(const char *data, qint64 maxSize) +{ + qDebug() << ""; + return m_port.write(data, maxSize); +} + +bool SerialPort::flush() +{ + qDebug() << ""; + return m_port.flush(); +} + +void SerialPort::close() +{ + qDebug() << ""; + return m_port.close(); +} + +void SerialPort::onReadyRead() +{ + qDebug() << ""; + emit readyRead(); +} diff --git a/oscar/SleepLib/deviceconnection.h b/oscar/SleepLib/deviceconnection.h index 24a441e7..c70c5ca1 100644 --- a/oscar/SleepLib/deviceconnection.h +++ b/oscar/SleepLib/deviceconnection.h @@ -52,8 +52,37 @@ public: // TODO: This class may eventually be internal to a DeviceConnection class, // but for now it is used to provide support for recording and playback of // serial port connections before refactoring. -class SerialPort : public QSerialPort +class SerialPort : public QObject { + Q_OBJECT + +private: + QSerialPort m_port; + +private slots: + void onReadyRead(); + +signals: + void readyRead(); + +public: + SerialPort(); + virtual ~SerialPort(); + + void setPortName(const QString &name); + bool open(QIODevice::OpenMode mode); + bool setBaudRate(qint32 baudRate, QSerialPort::Directions directions = QSerialPort::AllDirections); + bool setDataBits(QSerialPort::DataBits dataBits); + bool setParity(QSerialPort::Parity parity); + bool setStopBits(QSerialPort::StopBits stopBits); + bool setFlowControl(QSerialPort::FlowControl flowControl); + bool clear(QSerialPort::Directions directions = QSerialPort::AllDirections); + qint64 bytesAvailable() const; + qint64 read(char *data, qint64 maxSize); + qint64 write(const char *data, qint64 maxSize); + bool flush(); + void close(); + }; // TODO: This class's functionality will eventually be internal to a