/* Device Connection Class Implementation * * Copyright (c) 2020 The OSCAR Team * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of the source code * for more details. */ #include "deviceconnection.h" #include #include #include #include #include static QString hex(int i) { return QString("0x") + QString::number(i, 16).toUpper(); } // MARK: - // MARK: XML record/playback base classes class XmlRecorder { public: XmlRecorder(class QFile * file); XmlRecorder(QString & string); ~XmlRecorder(); inline QXmlStreamWriter & xml() { return *m_xml; } protected: QFile* m_file; // nullptr for non-file recordings QXmlStreamWriter* m_xml; void prologue(); void epilogue(); }; class XmlReplayEvent; class XmlReplay { public: XmlReplay(class QFile * file); XmlReplay(QXmlStreamReader & xml); ~XmlReplay(); template inline T* getNextEvent(const QString & id = ""); protected: void deserialize(QXmlStreamReader & xml); void deserializeEvents(QXmlStreamReader & xml); QHash>> m_eventIndex; QList m_events; XmlReplayEvent* getNextEvent(const QString & type, const QString & id = ""); }; class XmlReplayEvent { public: XmlReplayEvent(); virtual ~XmlReplayEvent() = default; virtual const QString & tag() const = 0; virtual const QString & id() const { static const QString none(""); return none; }; void record(XmlRecorder* xml); friend QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const XmlReplayEvent & event); friend QXmlStreamReader & operator>>(QXmlStreamReader & xml, XmlReplayEvent & event); typedef XmlReplayEvent* (*FactoryMethod)(); static bool registerClass(const QString & tag, FactoryMethod factory); static XmlReplayEvent* createInstance(const QString & tag); protected: static QHash s_factories; QDateTime m_time; XmlReplayEvent* m_next; virtual void write(QXmlStreamWriter & /*xml*/) const {} virtual void read(QXmlStreamReader & /*xml*/) {} friend class XmlReplay; }; QHash XmlReplayEvent::s_factories; XmlRecorder::XmlRecorder(QFile* stream) : m_file(stream), m_xml(new QXmlStreamWriter(stream)) { prologue(); } XmlRecorder::XmlRecorder(QString & string) : m_file(nullptr), m_xml(new QXmlStreamWriter(&string)) { prologue(); } XmlRecorder::~XmlRecorder() { epilogue(); delete m_xml; } void XmlRecorder::prologue() { Q_ASSERT(m_xml); m_xml->setAutoFormatting(true); m_xml->setAutoFormattingIndent(2); m_xml->writeStartElement("xmlreplay"); m_xml->writeStartElement("events"); } void XmlRecorder::epilogue() { Q_ASSERT(m_xml); m_xml->writeEndElement(); // close events // TODO: write out any inline connections m_xml->writeEndElement(); // close xmlreplay } XmlReplay::XmlReplay(QFile* file) { QXmlStreamReader xml(file); deserialize(xml); } XmlReplay::XmlReplay(QXmlStreamReader & xml) { deserialize(xml); } XmlReplay::~XmlReplay() { for (auto event : m_events) { delete event; } } void XmlReplay::deserialize(QXmlStreamReader & xml) { if (xml.readNextStartElement()) { if (xml.name() == "xmlreplay") { while (xml.readNextStartElement()) { if (xml.name() == "events") { deserializeEvents(xml); // else TODO: inline connections } else { qWarning() << "unexpected payload in replay XML:" << xml.name(); xml.skipCurrentElement(); } } } } } void XmlReplay::deserializeEvents(QXmlStreamReader & xml) { while (xml.readNextStartElement()) { QString type = xml.name().toString(); XmlReplayEvent* event = XmlReplayEvent::createInstance(type); if (event) { xml >> *event; // Add to list if (m_events.isEmpty() == false) { m_events.last()->m_next = event; } m_events.append(event); // Add to index const QString & id = event->id(); auto & events = m_eventIndex[type][id]; events.append(event); } else { xml.skipCurrentElement(); } } } XmlReplayEvent* XmlReplay::getNextEvent(const QString & type, const QString & id) { XmlReplayEvent* event = nullptr; if (m_eventIndex.contains(type)) { auto & ids = m_eventIndex[type]; if (ids.contains(id)) { auto & events = ids[id]; if (events.isEmpty() == false) { event = events.first(); // TODO: if we're simulating the original timing, return nullptr if we haven't reached this event's time yet; // otherwise: events.removeFirst(); } } } return event; } template T* XmlReplay::getNextEvent(const QString & id) { T* event = dynamic_cast(getNextEvent(T::TAG, id)); return event; } // MARK: - // MARK: XML record/playback event base class XmlReplayEvent::XmlReplayEvent() : m_time(QDateTime::currentDateTime()), m_next(nullptr) { } void XmlReplayEvent::record(XmlRecorder* writer) { // Do nothing if we're not recording. if (writer != nullptr) { writer->xml() << *this; } } bool XmlReplayEvent::registerClass(const QString & tag, XmlReplayEvent::FactoryMethod factory) { if (s_factories.contains(tag)) { qWarning() << "Event class already registered for tag" << tag; return false; } s_factories[tag] = factory; return true; } XmlReplayEvent* XmlReplayEvent::createInstance(const QString & tag) { XmlReplayEvent* event = nullptr; XmlReplayEvent::FactoryMethod factory = s_factories.value(tag); if (factory == nullptr) { qWarning() << "No event class registered for XML tag" << tag; } else { event = factory(); } return event; } QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const XmlReplayEvent & event) { QDateTime time = event.m_time.toOffsetFromUtc(event.m_time.offsetFromUtc()); // force display of UTC offset #if QT_VERSION < QT_VERSION_CHECK(5,9,0) // TODO: Can we please deprecate support for Qt older than 5.9? QString timestamp = time.toString(Qt::ISODate); #else QString timestamp = time.toString(Qt::ISODateWithMs); #endif xml.writeStartElement(event.tag()); xml.writeAttribute("time", timestamp); event.write(xml); xml.writeEndElement(); return xml; } QXmlStreamReader & operator>>(QXmlStreamReader & xml, XmlReplayEvent & event) { Q_ASSERT(xml.isStartElement() && xml.name() == event.tag()); QDateTime time; if (xml.attributes().hasAttribute("time")) { #if QT_VERSION < QT_VERSION_CHECK(5,9,0) // TODO: Can we please deprecate support for Qt older than 5.9? time = QDateTime::fromString(xml.attributes().value("time").toString(), Qt::ISODate); #else time = QDateTime::fromString(xml.attributes().value("time").toString(), Qt::ISODateWithMs); #endif } else { qWarning() << "Missing timestamp in" << xml.name() << "tag, using current time"; time = QDateTime::currentDateTime(); } event.read(xml); return xml; } template QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const QList & list) { for (auto & item : list) { xml << item; } return xml; } template QXmlStreamReader & operator>>(QXmlStreamReader & xml, QList & list) { list.clear(); while (xml.readNextStartElement()) { T item; xml >> item; list.append(item); } return xml; } // We use this extra CRTP templating so that concrete event subclasses require as little code as possible. template class XmlReplayBase : public XmlReplayEvent { public: static const QString TAG; static const bool registered; virtual const QString & tag() const { return TAG; }; static XmlReplayEvent* createInstance() { Derived* instance = new Derived(); return static_cast(instance); } }; #define REGISTER_XMLREPLAYEVENT(tag, type) \ template<> const QString XmlReplayBase::TAG = tag; \ template<> const bool XmlReplayBase::registered = XmlReplayEvent::registerClass(XmlReplayBase::TAG, XmlReplayBase::createInstance); // MARK: - // MARK: Device connection manager inline DeviceConnectionManager & DeviceConnectionManager::getInstance() { static DeviceConnectionManager instance; return instance; } DeviceConnectionManager::DeviceConnectionManager() : m_record(nullptr), m_replay(nullptr) { } void DeviceConnectionManager::record(QFile* stream) { if (m_record) { delete m_record; } if (stream) { m_record = new XmlRecorder(stream); } else { // nullptr turns off recording m_record = nullptr; } } void DeviceConnectionManager::record(QString & string) { if (m_record) { delete m_record; } m_record = new XmlRecorder(string); } void DeviceConnectionManager::replay(const QString & string) { QXmlStreamReader xml(string); reset(); if (m_replay) { delete m_replay; } m_replay = new XmlReplay(xml); } void DeviceConnectionManager::replay(QFile* file) { reset(); if (m_replay) { delete m_replay; } if (file) { m_replay = new XmlReplay(file); } else { // nullptr turns off replay m_replay = nullptr; } } DeviceConnection* DeviceConnectionManager::openConnection(const QString & type, const QString & name) { if (!s_factories.contains(type)) { qWarning() << "Unknown device connection type:" << type; return nullptr; } if (m_connections.contains(name)) { qWarning() << "connection to" << name << "already open"; return nullptr; } DeviceConnection* conn = s_factories[type](name, m_record, m_replay); if (conn) { if (conn->open()) { m_connections[name] = conn; } else { qWarning().noquote() << "unable to open" << type << "connection to" << name; delete conn; conn = nullptr; } } else { qWarning() << "unable to create" << type << "connection to" << name; } // TODO: record event return conn; } void DeviceConnectionManager::connectionClosed(DeviceConnection* conn) { Q_ASSERT(conn); const QString & type = conn->type(); const QString & name = conn->name(); if (m_connections.contains(name)) { if (m_connections[name] == conn) { m_connections.remove(name); } else { qWarning() << "connection to" << name << "not created by openConnection!"; } } else { qWarning() << type << "connection to" << name << "missing"; } // TODO: record event } // Temporary convenience function for code that still supports only serial ports. SerialPortConnection* DeviceConnectionManager::openSerialPortConnection(const QString & portName) { return dynamic_cast(getInstance().openConnection(SerialPortConnection::TYPE, portName)); } QHash DeviceConnectionManager::s_factories; bool DeviceConnectionManager::registerClass(const QString & type, DeviceConnection::FactoryMethod factory) { if (s_factories.contains(type)) { qWarning() << "Connection class already registered for type" << type; return false; } s_factories[type] = factory; return true; } #define REGISTER_DEVICECONNECTION(type, T) \ const QString T::TYPE = type; \ const bool T::registered = DeviceConnectionManager::registerClass(T::TYPE, T::createInstance); \ DeviceConnection* T::createInstance(const QString & name, XmlRecorder* record, XmlReplay* replay) { return static_cast(new T(name, record, replay)); } // MARK: - // MARK: Device manager events class GetAvailableSerialPortsEvent : public XmlReplayBase { public: QList m_ports; protected: virtual void write(QXmlStreamWriter & xml) const { xml << m_ports; } virtual void read(QXmlStreamReader & xml) { xml >> m_ports; } }; REGISTER_XMLREPLAYEVENT("getAvailableSerialPorts", GetAvailableSerialPortsEvent); QList DeviceConnectionManager::getAvailableSerialPorts() { GetAvailableSerialPortsEvent event; if (!m_replay) { for (auto & info : QSerialPortInfo::availablePorts()) { event.m_ports.append(SerialPortInfo(info)); } } else { auto replayEvent = m_replay->getNextEvent(); if (replayEvent) { event.m_ports = replayEvent->m_ports; } else { // If there are no replay events available, reuse the most recent state. event.m_ports = m_serialPorts; } } m_serialPorts = event.m_ports; event.record(m_record); return event.m_ports; } // MARK: - // MARK: Serial port info SerialPortInfo::SerialPortInfo(const QSerialPortInfo & other) { if (other.isNull() == false) { m_info["portName"] = other.portName(); m_info["systemLocation"] = other.systemLocation(); m_info["description"] = other.description(); m_info["manufacturer"] = other.manufacturer(); m_info["serialNumber"] = other.serialNumber(); if (other.hasVendorIdentifier()) { m_info["vendorIdentifier"] = other.vendorIdentifier(); } if (other.hasProductIdentifier()) { m_info["productIdentifier"] = other.productIdentifier(); } } } SerialPortInfo::SerialPortInfo(const SerialPortInfo & other) : m_info(other.m_info) { } SerialPortInfo::SerialPortInfo(const QString & data) { QXmlStreamReader xml(data); xml.readNextStartElement(); xml >> *this; } SerialPortInfo::SerialPortInfo() { } // TODO: This is a temporary wrapper until we begin refactoring. QList SerialPortInfo::availablePorts() { return DeviceConnectionManager::getInstance().getAvailableSerialPorts(); } QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const SerialPortInfo & info) { xml.writeStartElement("serial"); if (info.isNull() == false) { xml.writeAttribute("portName", info.portName()); xml.writeAttribute("systemLocation", info.systemLocation()); xml.writeAttribute("description", info.description()); xml.writeAttribute("manufacturer", info.manufacturer()); xml.writeAttribute("serialNumber", info.serialNumber()); if (info.hasVendorIdentifier()) { xml.writeAttribute("vendorIdentifier", hex(info.vendorIdentifier())); } if (info.hasProductIdentifier()) { xml.writeAttribute("productIdentifier", hex(info.productIdentifier())); } } xml.writeEndElement(); return xml; } QXmlStreamReader & operator>>(QXmlStreamReader & xml, SerialPortInfo & info) { if (xml.atEnd() == false && xml.isStartElement() && xml.name() == "serial") { for (auto & attribute : xml.attributes()) { QString name = attribute.name().toString(); QString value = attribute.value().toString(); if (name == "vendorIdentifier" || name == "productIdentifier") { bool ok; quint16 id = value.toUInt(&ok, 0); if (ok) { info.m_info[name] = id; } else { qWarning() << "invalid" << name << "value" << value; } } else { info.m_info[name] = value; } } } else { qWarning() << "no tag"; } xml.skipCurrentElement(); return xml; } SerialPortInfo::operator QString() const { QString out; QXmlStreamWriter xml(&out); xml << *this; return out; } bool SerialPortInfo::operator==(const SerialPortInfo & other) const { return m_info == other.m_info; } // MARK: - // MARK: Device connection base class DeviceConnection::DeviceConnection(const QString & name, XmlRecorder* record, XmlReplay* replay) : m_name(name), m_record(record), m_replay(replay), m_opened(false) { } DeviceConnection::~DeviceConnection() { } // TODO: log these to XML stream class ConnectionEvent { public: ConnectionEvent(const QString & tag) : m_tag(tag) { } void set(const QString & name, const QString & value) { m_values[name] = value; m_keys.append(name); } void set(const QString & name, qint64 value) { set(name, QString::number(value)); } void setData(const char* data, qint64 length) { 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(' ')); } inline bool ok() const { return m_values.contains("error") == false; } operator QString() const; protected: const QString m_tag; QHash m_values; QList m_keys; QString m_data; friend QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const ConnectionEvent & event); }; class SetValueEvent : public ConnectionEvent { public: SetValueEvent(const QString & name, int value) : ConnectionEvent("set") { set(name, value); } }; ConnectionEvent::operator QString() const { QString out; QXmlStreamWriter xml(&out); xml << *this; return out; } QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const ConnectionEvent & event) { xml.writeStartElement(event.m_tag); for (auto key : event.m_keys) { xml.writeAttribute(key, event.m_values[key]); } if (!event.m_data.isEmpty()) { xml.writeCharacters(event.m_data); } xml.writeEndElement(); return xml; } // MARK: - // MARK: Serial port connection REGISTER_DEVICECONNECTION("serial", SerialPortConnection); SerialPortConnection::SerialPortConnection(const QString & name, XmlRecorder* record, XmlReplay* replay) : DeviceConnection(name, record, replay) { connect(&m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); } SerialPortConnection::~SerialPortConnection() { if (m_opened) { close(); DeviceConnectionManager::getInstance().connectionClosed(this); } disconnect(&m_port, SIGNAL(readyRead()), this, SLOT(onReadyRead())); } bool SerialPortConnection::open() { if (m_opened) { qWarning() << "serial connection to" << m_name << "already opened"; return false; } ConnectionEvent event("openConnection"); event.set("type", "serial"); event.set("name", m_name); m_port.setPortName(m_name); checkResult(m_port.open(QSerialPort::ReadWrite), event); // TODO: send this event back to manager to log qDebug().noquote() << event; m_opened = event.ok(); return event.ok(); } bool SerialPortConnection::setBaudRate(qint32 baudRate, QSerialPort::Directions directions) { SetValueEvent event("baudRate", baudRate); event.set("directions", directions); checkResult(m_port.setBaudRate(baudRate, directions), event); qDebug().noquote() << event; return event.ok(); } bool SerialPortConnection::setDataBits(QSerialPort::DataBits dataBits) { SetValueEvent event("setDataBits", dataBits); checkResult(m_port.setDataBits(dataBits), event); qDebug().noquote() << event; return event.ok(); } bool SerialPortConnection::setParity(QSerialPort::Parity parity) { SetValueEvent event("setParity", parity); checkResult(m_port.setParity(parity), event); qDebug().noquote() << event; return event.ok(); } bool SerialPortConnection::setStopBits(QSerialPort::StopBits stopBits) { SetValueEvent event("setStopBits", stopBits); checkResult(m_port.setStopBits(stopBits), event); qDebug().noquote() << event; return event.ok(); } bool SerialPortConnection::setFlowControl(QSerialPort::FlowControl flowControl) { SetValueEvent event("setFlowControl", flowControl); checkResult(m_port.setFlowControl(flowControl), event); qDebug().noquote() << event; return event.ok(); } bool SerialPortConnection::clear(QSerialPort::Directions directions) { ConnectionEvent event("clear"); event.set("directions", directions); checkResult(m_port.clear(directions), event); qDebug().noquote() << event; return event.ok(); } qint64 SerialPortConnection::bytesAvailable() const { ConnectionEvent event("get"); qint64 result = m_port.bytesAvailable(); event.set("bytesAvailable", result); checkResult(result, event); qDebug().noquote() << event; return result; } qint64 SerialPortConnection::read(char *data, qint64 maxSize) { ConnectionEvent event("rx"); qint64 len = m_port.read(data, maxSize); if (len > 0) { event.setData(data, len); } event.set("len", len); if (len != maxSize) { event.set("req", maxSize); } checkResult(len, event); qDebug().noquote() << event; return len; } qint64 SerialPortConnection::write(const char *data, qint64 maxSize) { ConnectionEvent event("tx"); event.setData(data, maxSize); qint64 len = m_port.write(data, maxSize); event.set("len", len); if (len != maxSize) { event.set("req", maxSize); } checkResult(len, event); qDebug().noquote() << event; return len; } bool SerialPortConnection::flush() { ConnectionEvent event("flush"); checkResult(m_port.flush(), event); qDebug().noquote() << event; return event.ok(); } void SerialPortConnection::close() { ConnectionEvent event("closeConnection"); event.set("type", "serial"); event.set("name", m_name); // TODO: the separate connection stream will have an enclosing "connection" tag with these // attributes. The main device connection manager stream will log this openConnection/ // closeConnection pair. We'll also need to include a loader ID and stream version number // in the "connection" tag, so that if we ever have to change a loader's download code, // the older replays will still work as expected. m_port.close(); checkError(event); qDebug().noquote() << event; } void SerialPortConnection::onReadyRead() { ConnectionEvent event("readyRead"); // 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; emit readyRead(); } void SerialPortConnection::checkResult(bool ok, ConnectionEvent & event) const { QSerialPort::SerialPortError error = m_port.error(); if (ok && error == QSerialPort::NoError) return; event.set("error", error); if (ok) event.set("ok", ok); // we don't expect to see this, but we should know if it happens } void SerialPortConnection::checkResult(qint64 len, ConnectionEvent & event) const { QSerialPort::SerialPortError error = m_port.error(); if (len < 0 || error != QSerialPort::NoError) { event.set("error", error); } } void SerialPortConnection::checkError(ConnectionEvent & event) const { QSerialPort::SerialPortError error = m_port.error(); if (error != QSerialPort::NoError) { event.set("error", error); } } // MARK: - // MARK: SerialPort legacy class SerialPort::SerialPort() { } SerialPort::~SerialPort() { if (m_conn) { close(); } } void SerialPort::setPortName(const QString &name) { m_portName = name; } bool SerialPort::open(QIODevice::OpenMode mode) { Q_ASSERT(!m_conn); Q_ASSERT(mode == QSerialPort::ReadWrite); m_conn = DeviceConnectionManager::openSerialPortConnection(m_portName); if (m_conn) { connect(m_conn, SIGNAL(readyRead()), this, SLOT(onReadyRead())); } return m_conn != nullptr; } bool SerialPort::setBaudRate(qint32 baudRate, QSerialPort::Directions directions) { Q_ASSERT(m_conn); return m_conn->setBaudRate(baudRate, directions); } bool SerialPort::setDataBits(QSerialPort::DataBits dataBits) { Q_ASSERT(m_conn); return m_conn->setDataBits(dataBits); } bool SerialPort::setParity(QSerialPort::Parity parity) { Q_ASSERT(m_conn); return m_conn->setParity(parity); } bool SerialPort::setStopBits(QSerialPort::StopBits stopBits) { Q_ASSERT(m_conn); return m_conn->setStopBits(stopBits); } bool SerialPort::setFlowControl(QSerialPort::FlowControl flowControl) { Q_ASSERT(m_conn); return m_conn->setFlowControl(flowControl); } bool SerialPort::clear(QSerialPort::Directions directions) { Q_ASSERT(m_conn); return m_conn->clear(directions); } qint64 SerialPort::bytesAvailable() const { Q_ASSERT(m_conn); return m_conn->bytesAvailable(); } qint64 SerialPort::read(char *data, qint64 maxSize) { Q_ASSERT(m_conn); return m_conn->read(data, maxSize); } qint64 SerialPort::write(const char *data, qint64 maxSize) { Q_ASSERT(m_conn); return m_conn->write(data, maxSize); } bool SerialPort::flush() { Q_ASSERT(m_conn); return m_conn->flush(); } void SerialPort::close() { Q_ASSERT(m_conn); disconnect(m_conn, SIGNAL(readyRead()), this, SLOT(onReadyRead())); delete m_conn; // this will close the connection m_conn = nullptr; } void SerialPort::onReadyRead() { emit readyRead(); }