OSCAR-code/oscar/SleepLib/deviceconnection.cpp
sawinglogz 553cf59a95 Move connection creation to DeviceConnectionManager.
Calling openConnection will return an open connection or nullptr.
Deleting the connection will close it.

SerialPort now uses this under the hood, while still presenting
the QSerialPort-compatible interface.
2020-07-10 11:51:52 -04:00

990 lines
25 KiB
C++

/* 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 <QtSerialPort/QSerialPortInfo>
#include <QFile>
#include <QBuffer>
#include <QDateTime>
#include <QDebug>
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 XmlReplay
{
public:
XmlReplay(class QFile * file);
XmlReplay(QXmlStreamReader & xml);
~XmlReplay();
template<class T> inline T* getNextEvent();
protected:
void deserialize(QXmlStreamReader & xml);
void deserializeEvents(QXmlStreamReader & xml);
// TODO: maybe the QList should be a QHash on the timestamp?
// Then indices would be iterators over a sorted list of keys.
QHash<QString,QList<class XmlReplayEvent*>> m_events;
QHash<QString,int> m_indices;
class XmlReplayEvent* getNextEvent(const QString & type);
};
class XmlReplayEvent
{
public:
XmlReplayEvent();
virtual ~XmlReplayEvent() = default;
virtual const QString & tag() const = 0;
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<QString,FactoryMethod> s_factories;
QDateTime m_time;
virtual void write(QXmlStreamWriter & /*xml*/) const {}
virtual void read(QXmlStreamReader & /*xml*/) {}
};
QHash<QString,XmlReplayEvent::FactoryMethod> 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 list : m_events.values()) {
for (auto event : list) {
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 name = xml.name().toString();
XmlReplayEvent* event = XmlReplayEvent::createInstance(name);
if (event) {
xml >> *event;
auto & events = m_events[name];
events.append(event);
} else {
xml.skipCurrentElement();
}
}
}
XmlReplayEvent* XmlReplay::getNextEvent(const QString & type)
{
XmlReplayEvent* event = nullptr;
if (m_events.contains(type)) {
auto & events = m_events[type];
int i = m_indices[type];
if (i < events.size()) {
event = events[i];
// TODO: if we're simulating the original timing, return nullptr if we haven't reached this event's time yet;
// otherwise:
m_indices[type] = i + 1;
}
}
return event;
}
template<class T>
T* XmlReplay::getNextEvent()
{
T* event = dynamic_cast<T*>(getNextEvent(T::TAG));
return event;
}
// MARK: -
// MARK: XML record/playback event base class
XmlReplayEvent::XmlReplayEvent()
: m_time(QDateTime::currentDateTime())
{
}
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<typename T> QXmlStreamWriter & operator<<(QXmlStreamWriter & xml, const QList<T> & list)
{
for (auto & item : list) {
xml << item;
}
return xml;
}
template<typename T> QXmlStreamReader & operator>>(QXmlStreamReader & xml, QList<T> & 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 <typename Derived>
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<XmlReplayEvent*>(instance);
}
};
#define REGISTER_XMLREPLAYEVENT(tag, type) \
template<> const QString XmlReplayBase<type>::TAG = tag; \
template<> const bool XmlReplayBase<type>::registered = XmlReplayEvent::registerClass(XmlReplayBase<type>::TAG, XmlReplayBase<type>::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[type].contains(name)) {
qWarning() << type << "connection to" << name << "already open";
return nullptr;
}
DeviceConnection* conn = s_factories[type](name, m_record, m_replay);
if (conn) {
if (conn->open()) {
m_connections[type][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();
Q_ASSERT(s_factories.contains(type));
if (m_connections[type].contains(name)) {
if (m_connections[type][name] == conn) {
m_connections[type].remove(name);
} else {
qWarning() << type << "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<SerialPortConnection*>(getInstance().openConnection(SerialPortConnection::TYPE, portName));
}
QHash<QString,DeviceConnection::FactoryMethod> 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<DeviceConnection*>(new T(name, record, replay)); }
// MARK: -
// MARK: Device manager events
class GetAvailablePortsEvent : public XmlReplayBase<GetAvailablePortsEvent>
{
public:
QList<SerialPortInfo> m_ports;
protected:
virtual void write(QXmlStreamWriter & xml) const
{
xml << m_ports;
}
virtual void read(QXmlStreamReader & xml)
{
xml >> m_ports;
}
};
REGISTER_XMLREPLAYEVENT("getAvailablePorts", GetAvailablePortsEvent);
QList<SerialPortInfo> DeviceConnectionManager::getAvailablePorts()
{
GetAvailablePortsEvent event;
if (!m_replay) {
for (auto & info : QSerialPortInfo::availablePorts()) {
event.m_ports.append(SerialPortInfo(info));
}
} else {
auto replayEvent = m_replay->getNextEvent<GetAvailablePortsEvent>();
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> SerialPortInfo::availablePorts()
{
return DeviceConnectionManager::getInstance().getAvailablePorts();
}
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 <serial> 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<QString,QString> m_values;
QList<QString> 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();
}