Add plumbing for intercepting and logging serial port activity.

This commit is contained in:
sawinglogz 2020-06-11 22:29:46 -04:00
parent 3afa8c87c1
commit 3210ecd1ba
2 changed files with 207 additions and 2 deletions

View File

@ -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<QString,int> m_values;
QList<QString> 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() << "<setPortName>";
return m_port.setPortName(name);
}
bool SerialPort::open(QIODevice::OpenMode mode)
{
qDebug() << "<open>";
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: <set time="FOO" name="baudrate" value="19200"/>
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() << "<clear>";
return m_port.clear(directions);
}
qint64 SerialPort::bytesAvailable() const
{
qDebug() << "<bytesAvailable>";
return m_port.bytesAvailable();
}
qint64 SerialPort::read(char *data, qint64 maxSize)
{
qDebug() << "<rx>";
return m_port.read(data, maxSize);
}
qint64 SerialPort::write(const char *data, qint64 maxSize)
{
qDebug() << "<tx>";
return m_port.write(data, maxSize);
}
bool SerialPort::flush()
{
qDebug() << "<flush>";
return m_port.flush();
}
void SerialPort::close()
{
qDebug() << "<close>";
return m_port.close();
}
void SerialPort::onReadyRead()
{
qDebug() << "<readyRead>";
emit readyRead();
}

View File

@ -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