2020-07-09 22:07:03 +00:00
/* Device Connection Manager
2020-06-04 18:32:03 +00:00
*
* 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"
2020-06-21 18:19:46 +00:00
# include "version.h"
2020-06-05 21:01:58 +00:00
# include <QtSerialPort/QSerialPortInfo>
2020-06-11 19:58:34 +00:00
# include <QFile>
2020-06-29 01:28:25 +00:00
# include <QFileInfo>
# include <QDir>
2020-06-11 19:58:34 +00:00
# include <QBuffer>
2020-06-06 20:53:47 +00:00
# include <QDateTime>
2020-06-30 15:35:23 +00:00
# include <QXmlStreamReader>
# include <QXmlStreamWriter>
2020-06-05 21:01:58 +00:00
# include <QDebug>
static QString hex ( int i )
{
return QString ( " 0x " ) + QString : : number ( i , 16 ) . toUpper ( ) ;
}
2020-06-06 20:53:47 +00:00
// MARK: -
2020-06-11 20:27:52 +00:00
// MARK: XML record/playback base classes
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
/*
* XML recording base class
*
* While this can be used on its own via the public constructors , it is
* typically used as a base class for a subclasses that handle specific
* events .
*
* A single instance of this class can write a linear sequence of events to
* XML , either to a string ( for testing ) or to a file ( for production use ) .
*
* Sometimes , however , there is need for certain sequences to be treated as
* separate , either due to multithreading ( such recording as multiple
* simultaneous connections ) , or in order to treat a certain excerpt ( such
* as data download that we might wish to archive ) separately .
*
* These sequences are handled as " substreams " of the parent stream . The
* parent stream will typically record a substream ' s open / close or start /
* stop along with its ID . The substream will be written to a separate XML
* stream identified by that ID . Substreams are implemented as subclasses of
* this base class .
*
* TODO : At the moment , only file - based substreams are supported . In theory
* it should be possible to cache string - based substreams and then insert
* them inline into the parent after the substream - close event is recorded .
*/
2020-06-13 20:58:46 +00:00
class XmlRecorder
2020-06-11 19:58:34 +00:00
{
public :
2020-07-09 22:07:03 +00:00
static const QString TAG ; // default tag if no subclass
2020-06-21 18:19:46 +00:00
2020-07-09 22:07:03 +00:00
XmlRecorder ( class QFile * file , const QString & tag = XmlRecorder : : TAG ) ; // record XML to the given file
XmlRecorder ( QString & string , const QString & tag = XmlRecorder : : TAG ) ; // record XML to the given string
virtual ~ XmlRecorder ( ) ; // write the epilogue and close the recorder
2020-07-10 01:19:54 +00:00
XmlRecorder * closeSubstream ( ) ; // convenience function to close out a substream and return its parent
2020-06-11 19:58:34 +00:00
inline QXmlStreamWriter & xml ( ) { return * m_xml ; }
2020-06-16 20:48:52 +00:00
inline void lock ( ) { m_mutex . lock ( ) ; }
inline void unlock ( ) { m_mutex . unlock ( ) ; }
2020-06-11 19:58:34 +00:00
protected :
2020-07-09 22:07:03 +00:00
XmlRecorder ( XmlRecorder * parent , const QString & id , const QString & tag ) ; // constructor used by substreams
QXmlStreamWriter * addSubstream ( XmlRecorder * child , const QString & id ) ; // initialize a child substream, used by above constructor
const QString m_tag ; // opening/closing tag for this instance
QFile * m_file ; // nullptr for non-file recordings
QXmlStreamWriter * m_xml ; // XML output stream
QMutex m_mutex ; // force one thread at a time to write to m_xml
XmlRecorder * m_parent ; // parent instance of a substream
2020-06-11 19:58:34 +00:00
2020-06-21 18:19:46 +00:00
virtual void prologue ( ) ;
virtual void epilogue ( ) ;
2020-06-11 19:58:34 +00:00
} ;
2020-06-21 18:19:46 +00:00
const QString XmlRecorder : : TAG = " xmlreplay " ;
2020-06-11 19:58:34 +00:00
2020-06-16 20:43:16 +00:00
class XmlReplayEvent ;
2020-07-09 22:07:03 +00:00
/*
* XML replay base class
*
* A single instance of this class caches events from a previously recorded
* XML stream , either from a string ( for testing ) or from a file ( for
* production use ) .
*
* Unlike recording , the replay need not be strictly linear . In fact , the
* implementation is designed to allow for limited reordering during replay ,
* so that minor changes to code should result in sensible replay until a
* new recording can be made .
*
* There are two aspects to this reordering :
*
* First , events can be retrieved ( and consumed ) in any order , being
* retrieved by type and ID ( and then in order within that type and ID ) .
*
* Second , events that are flagged as random - access ( see randomAccess below )
* will cause the above retrieval to subsequently begin searching on or
* after the random - access event ' s timestamp ( except for other random - access
* events , which are always searched from the beginning . )
*
* This allow non - stateful events to be replayed arbitrarily , and for
* stateful events ( such as commands sent to a device ) to be approximated
* ( where subsequent data received matches the command sent ) .
*
* Furthermore , when events are triggered in the same order as they were
* during recordering , the above reordering will have no effect , and the
* original ordering will be replayed identically .
*
* See XmlRecorder above for a discussion of substreams .
*/
2020-06-11 19:58:34 +00:00
class XmlReplay
{
public :
2020-07-09 22:07:03 +00:00
XmlReplay ( class QFile * file , const QString & tag = XmlRecorder : : TAG ) ; // replay XML from the given file
XmlReplay ( QXmlStreamReader & xml , const QString & tag = XmlRecorder : : TAG ) ; // replay XML from the given stream
2020-06-21 18:19:46 +00:00
virtual ~ XmlReplay ( ) ;
2020-07-10 01:19:54 +00:00
XmlReplay * closeSubstream ( ) ; // convenience function to close out a substream and return its parent
2020-07-09 22:07:03 +00:00
template < class T > inline T * getNextEvent ( const QString & id = " " ) ; // typesafe accessor to retrieve and consume the next matching event
2020-06-11 19:58:34 +00:00
2020-06-21 18:19:46 +00:00
2020-06-11 19:58:34 +00:00
protected :
2020-07-09 22:07:03 +00:00
XmlReplay ( XmlReplay * parent , const QString & id , const QString & tag = XmlRecorder : : TAG ) ; // constructor used by substreams
QXmlStreamReader * findSubstream ( XmlReplay * child , const QString & id ) ; // initialize a child substream, used by above constructor
2020-06-29 01:28:25 +00:00
2020-06-11 19:58:34 +00:00
void deserialize ( QXmlStreamReader & xml ) ;
void deserializeEvents ( QXmlStreamReader & xml ) ;
2020-07-09 22:07:03 +00:00
const QString m_tag ; // opening/closing tag for this instance
QFile * m_file ; // nullptr for non-file replay
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
QHash < QString , QHash < QString , QList < XmlReplayEvent * > > > m_eventIndex ; // type and ID-based index into the events, see discussion of reordering above
QHash < QString , QHash < QString , int > > m_indexPosition ; // positions at which to begin searching the index, updated by random-access events
QList < XmlReplayEvent * > m_events ; // linear list of all events in their original order
2020-06-11 19:58:34 +00:00
2020-06-16 20:43:16 +00:00
XmlReplayEvent * getNextEvent ( const QString & type , const QString & id = " " ) ;
2020-06-19 17:10:43 +00:00
void seekToTime ( const QDateTime & time ) ;
2020-06-19 02:05:43 +00:00
2020-07-09 22:07:03 +00:00
XmlReplayEvent * m_pendingSignal ; // the signal (if any) that should be replayed as soon as the current event has been processed
QMutex m_lock ; // prevent signals from being dispatched while an event is being processed, see XmlReplayLock below
2020-06-19 02:05:43 +00:00
inline void lock ( ) { m_lock . lock ( ) ; }
inline void unlock ( ) { m_lock . unlock ( ) ; }
void processPendingSignals ( const QObject * target ) ;
friend class XmlReplayLock ;
2020-06-29 01:28:25 +00:00
2020-07-09 22:07:03 +00:00
XmlReplay * m_parent ; // parent instance of a substream
2020-06-11 19:58:34 +00:00
} ;
2020-07-09 22:07:03 +00:00
/*
* XML replay event base class
*
* This class is used to represent a replayable event . An event is created
* when performing any replayable action , and then recorded ( via record ( ) )
* when appropriate . During replay , an event is retrieved from the XmlReplay
* instance and its previously recorded result should be returned instead of
* performing the original action .
*
* Subclasses are created as subclasses of the XmlReplayBase template ( see
* below ) , which handles their factory method and tag registration .
*
* Subclasses that should be retrieved by ID as well as type will need to
* override id ( ) to return the ID to use for indexing .
*
* Subclasses that represent signal events ( rather than API calls ) will need
* to set their m_signal string to the name of the signal to be emitted , and
* additionally override signal ( ) if they need to pass parameters with the
* signal .
*
* Subclasses that represent random - access events ( see discussion above )
* will need to override randomAccess ( ) to return true .
*
* Subclasses whose XML contains raw hexadecimal data will need to override
* usesData ( ) to return true . Subclasses whose XML contains other data
* ( such as complex data types ) will instead need to override read ( ) and
* write ( ) .
*/
2020-06-11 19:58:34 +00:00
class XmlReplayEvent
2020-06-06 20:53:47 +00:00
{
2020-06-11 19:58:34 +00:00
public :
XmlReplayEvent ( ) ;
virtual ~ XmlReplayEvent ( ) = default ;
2020-07-09 22:07:03 +00:00
//! \brief Return the XML tag used for this event. Automatically overridden for subclasses by template.
2020-06-11 19:58:34 +00:00
virtual const QString & tag ( ) const = 0 ;
2020-07-09 22:07:03 +00:00
//! \brief Return the ID for this event, if applicable. Subclasses should override this if their events should be retrieved by ID.
2020-06-18 20:14:38 +00:00
virtual const QString id ( ) const { static const QString none ( " " ) ; return none ; }
2020-07-09 22:07:03 +00:00
//! \brief True if this event represents a "random-access" event that should cause subsequent event searches to start after this event's timestamp. Subclasses that represent such a state change should override this method.
2020-06-19 17:10:43 +00:00
virtual bool randomAccess ( ) const { return false ; }
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
//! \brief Record this event to the given XML recorder, doing nothing if the recorder is null.
2020-06-29 01:28:25 +00:00
void record ( XmlRecorder * xml ) const ;
2020-07-09 22:07:03 +00:00
// Serialize this event to an XML stream.
2020-06-11 19:58:34 +00:00
friend QXmlStreamWriter & operator < < ( QXmlStreamWriter & xml , const XmlReplayEvent & event ) ;
2020-07-09 22:07:03 +00:00
// Deserialize this event's contents from an XML stream. The instance is first created via createInstance() based on the tag.
2020-06-11 19:58:34 +00:00
friend QXmlStreamReader & operator > > ( QXmlStreamReader & xml , XmlReplayEvent & event ) ;
2020-07-09 22:07:03 +00:00
2020-07-10 01:19:54 +00:00
// Write the tag's attributes and contents.
2020-06-29 01:28:25 +00:00
void writeTag ( QXmlStreamWriter & xml ) const ;
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
// Event subclass registration and instance creation
2020-06-11 19:58:34 +00:00
typedef XmlReplayEvent * ( * FactoryMethod ) ( ) ;
static bool registerClass ( const QString & tag , FactoryMethod factory ) ;
static XmlReplayEvent * createInstance ( const QString & tag ) ;
2020-07-09 22:07:03 +00:00
//! \brief Add the given key/value to the event. This will be written as an XML attribute in the order it added.
2020-06-18 20:14:38 +00:00
void set ( const QString & name , const QString & value )
{
2020-06-19 02:05:43 +00:00
if ( ! m_values . contains ( name ) ) {
m_keys . append ( name ) ;
}
2020-06-18 20:14:38 +00:00
m_values [ name ] = value ;
}
2020-07-09 22:07:03 +00:00
//! \brief Add the given key/integer to the event. This will be written as an XML attribute in the order it added.
2020-06-18 20:14:38 +00:00
void set ( const QString & name , qint64 value )
{
set ( name , QString : : number ( value ) ) ;
}
2020-07-09 22:07:03 +00:00
//! \brief Add the raw data to the event. This will be written in hexadecimal as content of the event's XML tag.
2020-06-18 20:14:38 +00:00
void setData ( const char * data , qint64 length )
{
Q_ASSERT ( usesData ( ) = = true ) ;
2020-06-19 02:05:43 +00:00
QByteArray bytes = QByteArray : : fromRawData ( data , length ) ;
m_data = bytes . toHex ( ' ' ) . toUpper ( ) ;
2020-06-18 20:14:38 +00:00
}
2020-07-09 22:07:03 +00:00
//! \brief Get the value for the given key.
2020-06-18 20:14:38 +00:00
inline QString get ( const QString & name ) const
{
if ( ! m_values . contains ( name ) ) {
qWarning ( ) . noquote ( ) < < * this < < " missing attribute: " < < name ;
}
return m_values [ name ] ;
}
2020-07-09 22:07:03 +00:00
//! \brief Get the raw data for this event.
2020-06-18 20:14:38 +00:00
QByteArray getData ( ) const
{
Q_ASSERT ( usesData ( ) = = true ) ;
2020-06-19 19:56:47 +00:00
if ( m_data . isEmpty ( ) ) {
qWarning ( ) . noquote ( ) < < " replaying event with missing data " < < * this ;
QByteArray empty ;
return empty ; // toUtf8() below crashes with an empty string.
}
2020-06-19 02:05:43 +00:00
return QByteArray : : fromHex ( m_data . toUtf8 ( ) ) ;
2020-06-18 20:14:38 +00:00
}
2020-07-09 22:07:03 +00:00
//! \brief True if there are no errors in this event, or false if the "error" attribute is set.
2020-06-18 20:14:38 +00:00
inline bool ok ( ) const { return m_values . contains ( " error " ) = = false ; }
2020-07-09 22:07:03 +00:00
//! \brief Return a string of this event as an XML tag.
2020-06-18 20:14:38 +00:00
operator QString ( ) const
{
QString out ;
QXmlStreamWriter xml ( & out ) ;
xml < < * this ;
return out ;
}
2020-07-09 22:07:03 +00:00
//! \brief Copy the result from the retrieved replay event (if any) into the current event.
2020-06-18 20:14:38 +00:00
void copyIf ( const XmlReplayEvent * other )
{
// Leave the proposed event alone if there was no replay event.
if ( other = = nullptr ) {
return ;
}
// Do not copy timestamp.
m_values = other - > m_values ;
m_keys = other - > m_keys ;
m_data = other - > m_data ;
}
2020-06-29 01:28:25 +00:00
protected :
2020-07-09 22:07:03 +00:00
//! \brief Copy the timestamp as well as the results. This is necessary for replaying substreams that use the timestamp as part of their ID.
2020-06-29 01:28:25 +00:00
void copy ( const XmlReplayEvent & other )
{
copyIf ( & other ) ;
m_time = other . m_time ;
}
2020-06-18 20:14:38 +00:00
2020-06-11 19:58:34 +00:00
protected :
2020-07-09 22:07:03 +00:00
static QHash < QString , FactoryMethod > s_factories ; // registered subclass factory methods, arranged by XML tag
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
QDateTime m_time ; // timestamp of event
XmlReplayEvent * m_next ; // next recorded event, used during replay to trigger signals that automatically fire after an event is processed
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
const char * m_signal ; // name of the signal to be emitted for this event, if any
2020-06-19 02:05:43 +00:00
inline bool isSignal ( ) const { return m_signal ! = nullptr ; }
2020-07-09 22:07:03 +00:00
//! \brief Send a signal to the target object. Subclasses may override this to send signal arguments.
2020-06-19 02:05:43 +00:00
virtual void signal ( QObject * target )
{
2020-07-09 22:07:03 +00:00
// Queue the signal so that it won't be processed before the current event returns to its caller.
// (See XmlReplayLock below.)
2020-06-19 02:05:43 +00:00
QMetaObject : : invokeMethod ( target , m_signal , Qt : : QueuedConnection ) ;
}
2020-07-09 22:07:03 +00:00
QHash < QString , QString > m_values ; // hash of key/value pairs for this event, written as attributes of the XML tag
QList < QString > m_keys ; // list of keys so that attributes will be written in the order they were set
QString m_data ; // hexademical string representing this event's raw data, written as contents of the XML tag
2020-06-18 20:14:38 +00:00
2020-07-09 22:07:03 +00:00
//! \brief Returns whether this event contains raw data. Defaults to false, so subclasses that use raw data must override this.
2020-06-18 20:14:38 +00:00
virtual bool usesData ( ) const { return false ; }
2020-07-09 22:07:03 +00:00
//! \brief Write any attributes or content needed specific to event. Subclasses may override this to support complex data types.
2020-06-18 20:14:38 +00:00
virtual void write ( QXmlStreamWriter & xml ) const
{
2020-07-09 22:07:03 +00:00
// Write key/value pairs as attributes in the order they were set.
2020-06-18 20:14:38 +00:00
for ( auto key : m_keys ) {
xml . writeAttribute ( key , m_values [ key ] ) ;
}
if ( ! m_data . isEmpty ( ) ) {
Q_ASSERT ( usesData ( ) = = true ) ;
xml . writeCharacters ( m_data ) ;
}
}
2020-07-09 22:07:03 +00:00
//! \brief Read any attributes or content specific to this event. Subclasses may override this to support complex data types.
2020-06-18 20:14:38 +00:00
virtual void read ( QXmlStreamReader & xml )
{
QXmlStreamAttributes attribs = xml . attributes ( ) ;
for ( auto & attrib : attribs ) {
2020-07-09 22:07:03 +00:00
if ( attrib . name ( ) ! = " time " ) { // skip outer timestamp, which is decoded by operator>>
2020-06-18 20:14:38 +00:00
set ( attrib . name ( ) . toString ( ) , attrib . value ( ) . toString ( ) ) ;
}
}
if ( usesData ( ) ) {
m_data = xml . readElementText ( ) ;
} else {
xml . skipCurrentElement ( ) ;
}
}
2020-06-16 20:43:16 +00:00
friend class XmlReplay ;
2020-06-11 19:58:34 +00:00
} ;
QHash < QString , XmlReplayEvent : : FactoryMethod > XmlReplayEvent : : s_factories ;
2020-07-09 22:07:03 +00:00
/*
* XML replay lock class
*
* An instance of this class should be created on the stack during any replayable
* event . Exiting scope will release the lock , at which point any signals that
* need to be replayed will be queued .
*
* Has no effect if events are not being replayed .
*/
2020-06-19 02:05:43 +00:00
class XmlReplayLock
{
public :
2020-07-09 22:07:03 +00:00
//! \brief Temporarily lock the XML replay (if any) until exiting scope, at which point any pending signals will be sent to the specified object.
2020-06-19 02:05:43 +00:00
XmlReplayLock ( const QObject * obj , XmlReplay * replay )
: m_target ( obj ) , m_replay ( replay )
{
if ( m_replay ) {
2020-07-09 22:07:03 +00:00
// Prevent any triggered signal events from processing until the triggering lock is released.
2020-06-19 02:05:43 +00:00
m_replay - > lock ( ) ;
}
}
~ XmlReplayLock ( )
{
if ( m_replay ) {
m_replay - > processPendingSignals ( m_target ) ;
m_replay - > unlock ( ) ;
}
}
protected :
2020-07-09 22:07:03 +00:00
const QObject * m_target ; // target object to receive any pending signals
XmlReplay * m_replay ; // replay instance, or nullptr if not replaying
2020-06-19 02:05:43 +00:00
} ;
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
// Derive the filepath for the given substream ID relative to the parent stream.
2020-06-29 01:28:25 +00:00
static QString substreamFilepath ( QFile * parent , const QString & id )
{
Q_ASSERT ( parent ) ;
QFileInfo info ( * parent ) ;
QString path = info . canonicalPath ( ) + QDir : : separator ( ) + info . completeBaseName ( ) + " - " + id + " .xml " ;
return path ;
}
2020-06-21 18:19:46 +00:00
XmlRecorder : : XmlRecorder ( QFile * stream , const QString & tag )
2020-06-29 01:28:25 +00:00
: m_tag ( tag ) , m_file ( stream ) , m_xml ( new QXmlStreamWriter ( stream ) ) , m_parent ( nullptr )
2020-06-11 19:58:34 +00:00
{
prologue ( ) ;
2020-06-06 20:53:47 +00:00
}
2020-06-21 18:19:46 +00:00
XmlRecorder : : XmlRecorder ( QString & string , const QString & tag )
2020-06-29 01:28:25 +00:00
: m_tag ( tag ) , m_file ( nullptr ) , m_xml ( new QXmlStreamWriter ( & string ) ) , m_parent ( nullptr )
2020-06-06 20:53:47 +00:00
{
2020-06-11 19:58:34 +00:00
prologue ( ) ;
2020-06-06 20:53:47 +00:00
}
2020-07-09 22:07:03 +00:00
// Protected constructor for substreams
2020-06-29 01:28:25 +00:00
XmlRecorder : : XmlRecorder ( XmlRecorder * parent , const QString & id , const QString & tag )
: m_tag ( tag ) , m_file ( nullptr ) , m_xml ( nullptr ) , m_parent ( parent )
{
Q_ASSERT ( m_parent ) ;
m_xml = m_parent - > addSubstream ( this , id ) ;
if ( m_xml = = nullptr ) {
qWarning ( ) < < " Not recording " < < id ;
static QString null ;
m_xml = new QXmlStreamWriter ( & null ) ;
}
2020-07-10 01:19:54 +00:00
prologue ( ) ;
2020-06-29 01:28:25 +00:00
}
2020-07-09 22:07:03 +00:00
// Initialize a child recording substream.
2020-06-29 01:28:25 +00:00
QXmlStreamWriter * XmlRecorder : : addSubstream ( XmlRecorder * child , const QString & id )
{
Q_ASSERT ( child ) ;
QXmlStreamWriter * xml = nullptr ;
if ( m_file ) {
QString childPath = substreamFilepath ( m_file , id ) ;
child - > m_file = new QFile ( childPath ) ;
if ( child - > m_file - > open ( QIODevice : : WriteOnly | QIODevice : : Append ) ) {
xml = new QXmlStreamWriter ( child - > m_file ) ;
qDebug ( ) < < " Recording to " < < childPath ;
} else {
qWarning ( ) < < " Unable to open " < < childPath < < " for writing " ;
}
} else {
qWarning ( ) < < " String-based substreams are not supported " ;
// Maybe some day support string-based substreams:
// - parent passes string to child
// - on connectionClosed, parent asks recorder to flush string to stream
}
return xml ;
}
2020-06-13 20:58:46 +00:00
XmlRecorder : : ~ XmlRecorder ( )
2020-06-06 20:53:47 +00:00
{
2020-06-11 19:58:34 +00:00
epilogue ( ) ;
delete m_xml ;
2020-07-09 22:07:03 +00:00
// File substreams manage their own file.
2020-06-29 01:28:25 +00:00
if ( m_parent & & m_file ) {
delete m_file ;
}
}
2020-07-09 22:07:03 +00:00
// Close out a substream and return its parent.
2020-07-10 01:19:54 +00:00
XmlRecorder * XmlRecorder : : closeSubstream ( )
2020-06-29 01:28:25 +00:00
{
auto parent = m_parent ;
delete this ;
return parent ;
2020-06-11 19:58:34 +00:00
}
2020-06-13 20:58:46 +00:00
void XmlRecorder : : prologue ( )
2020-06-11 19:58:34 +00:00
{
Q_ASSERT ( m_xml ) ;
m_xml - > setAutoFormatting ( true ) ;
m_xml - > setAutoFormattingIndent ( 2 ) ;
2020-07-10 01:19:54 +00:00
m_xml - > writeStartElement ( m_tag ) ; // open enclosing tag
2020-06-11 19:58:34 +00:00
}
2020-06-13 20:58:46 +00:00
void XmlRecorder : : epilogue ( )
2020-06-11 19:58:34 +00:00
{
Q_ASSERT ( m_xml ) ;
2020-06-21 18:19:46 +00:00
m_xml - > writeEndElement ( ) ; // close enclosing tag
2020-06-11 19:58:34 +00:00
}
2020-07-09 22:07:03 +00:00
2020-06-21 18:19:46 +00:00
XmlReplay : : XmlReplay ( QFile * file , const QString & tag )
2020-06-29 01:28:25 +00:00
: m_tag ( tag ) , m_file ( file ) , m_pendingSignal ( nullptr ) , m_parent ( nullptr )
2020-06-11 19:58:34 +00:00
{
2020-06-29 01:28:25 +00:00
Q_ASSERT ( file ) ;
QFileInfo info ( * file ) ;
qDebug ( ) < < " Replaying from " < < info . canonicalFilePath ( ) ;
2020-06-11 19:58:34 +00:00
QXmlStreamReader xml ( file ) ;
deserialize ( xml ) ;
}
2020-06-21 18:19:46 +00:00
XmlReplay : : XmlReplay ( QXmlStreamReader & xml , const QString & tag )
2020-06-29 01:28:25 +00:00
: m_tag ( tag ) , m_file ( nullptr ) , m_pendingSignal ( nullptr ) , m_parent ( nullptr )
2020-06-11 19:58:34 +00:00
{
deserialize ( xml ) ;
}
2020-07-09 22:07:03 +00:00
// Protected constructor for substreams
2020-06-29 01:28:25 +00:00
XmlReplay : : XmlReplay ( XmlReplay * parent , const QString & id , const QString & tag )
: m_tag ( tag ) , m_file ( nullptr ) , m_pendingSignal ( nullptr ) , m_parent ( parent )
{
Q_ASSERT ( m_parent ) ;
auto xml = m_parent - > findSubstream ( this , id ) ;
if ( xml ) {
deserialize ( * xml ) ;
delete xml ;
} else {
qWarning ( ) < < " Not replaying " < < id ;
}
}
2020-07-09 22:07:03 +00:00
// Initialize a child replay substream.
2020-06-29 01:28:25 +00:00
QXmlStreamReader * XmlReplay : : findSubstream ( XmlReplay * child , const QString & id )
{
Q_ASSERT ( child ) ;
QXmlStreamReader * xml = nullptr ;
if ( m_file ) {
QString childPath = substreamFilepath ( m_file , id ) ;
child - > m_file = new QFile ( childPath ) ;
if ( child - > m_file - > open ( QIODevice : : ReadOnly ) ) {
xml = new QXmlStreamReader ( child - > m_file ) ;
qDebug ( ) < < " Replaying from " < < childPath ;
} else {
qWarning ( ) < < " Unable to open " < < childPath < < " for reading " ;
}
} else {
qWarning ( ) < < " String-based substreams are not supported " ;
// Maybe some day support string-based substreams:
// - when deserializing, use e.g. ConnectionEvents to cache the substream strings
// - then return a QXmlStreamReader here using that string
}
return xml ;
}
2020-06-11 19:58:34 +00:00
XmlReplay : : ~ XmlReplay ( )
{
2020-06-16 20:43:16 +00:00
for ( auto event : m_events ) {
delete event ;
2020-06-11 19:58:34 +00:00
}
2020-07-09 22:07:03 +00:00
// File substreams manage their own file.
2020-06-29 01:28:25 +00:00
if ( m_parent & & m_file ) {
delete m_file ;
}
}
2020-07-09 22:07:03 +00:00
// Close out a substream and return its parent.
2020-07-10 01:19:54 +00:00
XmlReplay * XmlReplay : : closeSubstream ( )
2020-06-29 01:28:25 +00:00
{
auto parent = m_parent ;
delete this ;
return parent ;
2020-06-11 19:58:34 +00:00
}
void XmlReplay : : deserialize ( QXmlStreamReader & xml )
{
if ( xml . readNextStartElement ( ) ) {
2020-06-21 18:19:46 +00:00
if ( xml . name ( ) = = m_tag ) {
deserializeEvents ( xml ) ;
} else {
qWarning ( ) < < " unexpected payload in replay XML: " < < xml . name ( ) ;
xml . skipCurrentElement ( ) ;
2020-06-11 19:58:34 +00:00
}
}
}
void XmlReplay : : deserializeEvents ( QXmlStreamReader & xml )
{
while ( xml . readNextStartElement ( ) ) {
2020-06-16 20:43:16 +00:00
QString type = xml . name ( ) . toString ( ) ;
XmlReplayEvent * event = XmlReplayEvent : : createInstance ( type ) ;
2020-06-11 19:58:34 +00:00
if ( event ) {
xml > > * event ;
2020-06-16 20:43:16 +00:00
// 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 ] ;
2020-06-11 19:58:34 +00:00
events . append ( event ) ;
} else {
xml . skipCurrentElement ( ) ;
}
}
}
2020-07-09 22:07:03 +00:00
// Queue any pending signals when a replay lock is released.
2020-06-19 02:05:43 +00:00
void XmlReplay : : processPendingSignals ( const QObject * target )
{
if ( m_pendingSignal ) {
2020-06-19 17:10:43 +00:00
XmlReplayEvent * pending = m_pendingSignal ;
m_pendingSignal = nullptr ;
// Dequeue the signal from the index; this may update m_pendingSignal.
XmlReplayEvent * dequeued = getNextEvent ( pending - > tag ( ) , pending - > id ( ) ) ;
if ( dequeued ! = pending ) {
qWarning ( ) < < " triggered signal doesn't match dequeued signal! " < < pending < < dequeued ;
}
2020-06-19 02:05:43 +00:00
// 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.
2020-06-19 17:10:43 +00:00
pending - > signal ( const_cast < QObject * > ( target ) ) ;
}
}
2020-07-09 22:07:03 +00:00
// Update the positions at which to begin searching the index, so that only events on or after the given time are returned by getNextEvent.
2020-06-19 17:10:43 +00:00
void XmlReplay : : seekToTime ( const QDateTime & time )
{
for ( auto & type : m_eventIndex . keys ( ) ) {
for ( auto & key : m_eventIndex [ type ] . keys ( ) ) {
2020-07-09 22:07:03 +00:00
// Find the index of the first event on or after the given time.
2020-06-19 17:10:43 +00:00
auto & events = m_eventIndex [ type ] [ key ] ;
int pos ;
for ( pos = 0 ; pos < events . size ( ) ; pos + + ) {
auto & event = events . at ( pos ) ;
// Random-access events should always start searching from position 0.
if ( event - > randomAccess ( ) | | event - > m_time > = time ) {
break ;
}
}
// If pos == events.size(), that means there are no more events of this type
// after the given time.
m_indexPosition [ type ] [ key ] = pos ;
}
2020-06-19 02:05:43 +00:00
}
}
2020-07-09 22:07:03 +00:00
// Find and return the next event of the given type with the given ID, or nullptr if no more events match.
2020-06-16 20:43:16 +00:00
XmlReplayEvent * XmlReplay : : getNextEvent ( const QString & type , const QString & id )
2020-06-11 19:58:34 +00:00
{
XmlReplayEvent * event = nullptr ;
2020-07-09 22:07:03 +00:00
// Event handlers should always be wrapped in an XmlReplayLock, so warn if that's not the case.
2020-06-19 02:05:43 +00:00
if ( m_lock . tryLock ( ) ) {
qWarning ( ) < < " XML replay " < < type < < " object not locked by event handler! " ;
m_lock . unlock ( ) ;
}
2020-07-09 22:07:03 +00:00
// Search the index for the next matching event (if any).
2020-06-16 20:43:16 +00:00
if ( m_eventIndex . contains ( type ) ) {
auto & ids = m_eventIndex [ type ] ;
if ( ids . contains ( id ) ) {
auto & events = ids [ id ] ;
2020-06-19 17:10:43 +00:00
// Start at the position identified by the previous random-access event.
int pos = m_indexPosition [ type ] [ id ] ;
if ( pos < events . size ( ) ) {
event = events . at ( pos ) ;
2020-06-16 20:43:16 +00:00
// TODO: if we're simulating the original timing, return nullptr if we haven't reached this event's time yet;
// otherwise:
2020-06-19 17:10:43 +00:00
events . removeAt ( pos ) ;
2020-06-16 20:43:16 +00:00
}
2020-06-11 19:58:34 +00:00
}
}
2020-06-19 02:05:43 +00:00
2020-07-09 22:07:03 +00:00
// If this is a random-access event, we need to update the index positions for all non-random-access events.
2020-06-19 17:10:43 +00:00
if ( event & & event - > randomAccess ( ) ) {
seekToTime ( event - > m_time ) ;
}
2020-07-09 22:07:03 +00:00
// If the event following this one is a signal (that replay needs to trigger), save it as pending
// so that it can be emitted when the replay lock for this event is released.
2020-06-19 02:05:43 +00:00
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 ;
}
2020-06-11 19:58:34 +00:00
return event ;
}
2020-07-09 22:07:03 +00:00
// Public, typesafe wrapper for getNextEvent above.
2020-06-11 19:58:34 +00:00
template < class T >
2020-06-16 20:43:16 +00:00
T * XmlReplay : : getNextEvent ( const QString & id )
2020-06-11 19:58:34 +00:00
{
2020-06-16 20:43:16 +00:00
T * event = dynamic_cast < T * > ( getNextEvent ( T : : TAG , id ) ) ;
2020-06-11 19:58:34 +00:00
return event ;
}
// MARK: -
2020-06-11 20:27:52 +00:00
// MARK: XML record/playback event base class
2020-06-11 19:58:34 +00:00
XmlReplayEvent : : XmlReplayEvent ( )
2020-06-19 02:05:43 +00:00
: m_time ( QDateTime : : currentDateTime ( ) ) , m_next ( nullptr ) , m_signal ( nullptr )
2020-06-11 19:58:34 +00:00
{
}
2020-06-29 01:28:25 +00:00
void XmlReplayEvent : : record ( XmlRecorder * writer ) const
2020-06-11 19:58:34 +00:00
{
// Do nothing if we're not recording.
if ( writer ! = nullptr ) {
2020-06-16 20:48:52 +00:00
writer - > lock ( ) ;
2020-06-11 19:58:34 +00:00
writer - > xml ( ) < < * this ;
2020-06-16 20:48:52 +00:00
writer - > unlock ( ) ;
2020-06-11 19:58:34 +00:00
}
}
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 ;
}
2020-06-29 01:28:25 +00:00
void XmlReplayEvent : : writeTag ( QXmlStreamWriter & xml ) const
2020-06-11 19:58:34 +00:00
{
2020-06-29 01:28:25 +00:00
QDateTime time = m_time . toOffsetFromUtc ( m_time . offsetFromUtc ( ) ) ; // force display of UTC offset
2020-06-11 19:58:34 +00:00
# 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 . writeAttribute ( " time " , timestamp ) ;
2020-07-09 22:07:03 +00:00
// Call this event's overridable write method.
2020-06-29 01:28:25 +00:00
write ( xml ) ;
}
2020-06-11 19:58:34 +00:00
2020-06-29 01:28:25 +00:00
QXmlStreamWriter & operator < < ( QXmlStreamWriter & xml , const XmlReplayEvent & event )
{
2020-07-10 01:19:54 +00:00
xml . writeStartElement ( event . tag ( ) ) ;
2020-06-29 01:28:25 +00:00
event . writeTag ( xml ) ;
2020-06-11 19:58:34 +00:00
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 " ) ) {
2020-06-06 20:53:47 +00:00
# if QT_VERSION < QT_VERSION_CHECK(5,9,0)
// TODO: Can we please deprecate support for Qt older than 5.9?
2020-06-11 19:58:34 +00:00
time = QDateTime : : fromString ( xml . attributes ( ) . value ( " time " ) . toString ( ) , Qt : : ISODate ) ;
2020-06-06 20:53:47 +00:00
# else
2020-06-11 19:58:34 +00:00
time = QDateTime : : fromString ( xml . attributes ( ) . value ( " time " ) . toString ( ) , Qt : : ISODateWithMs ) ;
2020-06-06 20:53:47 +00:00
# endif
2020-06-11 19:58:34 +00:00
} else {
qWarning ( ) < < " Missing timestamp in " < < xml . name ( ) < < " tag, using current time " ;
time = QDateTime : : currentDateTime ( ) ;
2020-06-06 20:53:47 +00:00
}
2020-06-19 17:10:43 +00:00
event . m_time = time ;
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
// Call this event's overridable read method.
2020-06-11 19:58:34 +00:00
event . read ( xml ) ;
return xml ;
2020-06-06 20:53:47 +00:00
}
2020-07-09 22:07:03 +00:00
// Convenience template for serializing QLists to XML
2020-06-11 19:58:34 +00:00
template < typename T > QXmlStreamWriter & operator < < ( QXmlStreamWriter & xml , const QList < T > & list )
{
for ( auto & item : list ) {
xml < < item ;
}
return xml ;
}
2020-07-09 22:07:03 +00:00
// Convenience template for deserializing QLists from XML
2020-06-11 19:58:34 +00:00
template < typename T > QXmlStreamReader & operator > > ( QXmlStreamReader & xml , QList < T > & list )
{
list . clear ( ) ;
while ( xml . readNextStartElement ( ) ) {
T item ;
xml > > item ;
list . append ( item ) ;
}
return xml ;
}
2020-07-09 22:07:03 +00:00
/*
* Intermediate parent class of concrete event subclasses .
*
* We use this extra CRTP templating so that concrete event subclasses
* require as little code as possible :
*
* The subclass ' s tag and factory method are automatically generated by this
* template .
*/
2020-06-11 19:58:34 +00:00
template < typename Derived >
class XmlReplayBase : public XmlReplayEvent
{
public :
static const QString TAG ;
static const bool registered ;
virtual const QString & tag ( ) const { return TAG ; } ;
2020-06-06 20:53:47 +00:00
2020-06-11 19:58:34 +00:00
static XmlReplayEvent * createInstance ( )
{
Derived * instance = new Derived ( ) ;
return static_cast < XmlReplayEvent * > ( instance ) ;
}
} ;
2020-07-09 22:07:03 +00:00
/*
* Macro to define an XmlReplayEvent subclass ' s tag and automatically
* register the subclass at global - initialization time , before main ( )
*/
2020-06-11 19:58:34 +00:00
# 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: -
2020-06-11 20:27:52 +00:00
// MARK: Device connection manager
2020-07-09 22:07:03 +00:00
/*
* DeviceRecorder / DeviceReplay subclasses of XmlRecorder / XmlReplay
*
* Used by DeviceConnectionManager to record its activity , such as
* port scanning and connection opening / closing .
*/
2020-06-21 18:19:46 +00:00
class DeviceRecorder : public XmlRecorder
{
public :
static const QString TAG ;
2020-06-29 01:28:25 +00:00
DeviceRecorder ( class QFile * file ) : XmlRecorder ( file , DeviceRecorder : : TAG ) { m_xml - > writeAttribute ( " oscar " , getVersion ( ) . toString ( ) ) ; }
DeviceRecorder ( QString & string ) : XmlRecorder ( string , DeviceRecorder : : TAG ) { m_xml - > writeAttribute ( " oscar " , getVersion ( ) . toString ( ) ) ; }
2020-06-21 18:19:46 +00:00
} ;
const QString DeviceRecorder : : TAG = " devicereplay " ;
class DeviceReplay : public XmlReplay
{
public :
DeviceReplay ( class QFile * file ) : XmlReplay ( file , DeviceRecorder : : TAG ) { }
DeviceReplay ( QXmlStreamReader & xml ) : XmlReplay ( xml , DeviceRecorder : : TAG ) { }
} ;
2020-07-09 22:07:03 +00:00
// Return singleton instance of DeviceConnectionManager, creating it if necessary.
2020-06-11 20:27:52 +00:00
inline DeviceConnectionManager & DeviceConnectionManager : : getInstance ( )
{
static DeviceConnectionManager instance ;
return instance ;
}
2020-07-09 22:07:03 +00:00
// Protected constructor
2020-06-11 20:27:52 +00:00
DeviceConnectionManager : : DeviceConnectionManager ( )
: m_record ( nullptr ) , m_replay ( nullptr )
{
}
2020-06-11 19:58:34 +00:00
void DeviceConnectionManager : : record ( QFile * stream )
2020-06-06 20:53:47 +00:00
{
if ( m_record ) {
2020-06-11 19:58:34 +00:00
delete m_record ;
}
if ( stream ) {
2020-06-21 18:19:46 +00:00
m_record = new DeviceRecorder ( stream ) ;
2020-06-11 19:58:34 +00:00
} else {
// nullptr turns off recording
m_record = nullptr ;
2020-06-06 20:53:47 +00:00
}
}
2020-06-11 19:58:34 +00:00
void DeviceConnectionManager : : record ( QString & string )
2020-06-06 20:53:47 +00:00
{
2020-06-11 19:58:34 +00:00
if ( m_record ) {
delete m_record ;
}
2020-06-21 18:19:46 +00:00
m_record = new DeviceRecorder ( string ) ;
2020-06-11 19:58:34 +00:00
}
2020-06-06 20:53:47 +00:00
2020-06-11 19:58:34 +00:00
void DeviceConnectionManager : : replay ( const QString & string )
{
QXmlStreamReader xml ( string ) ;
reset ( ) ;
if ( m_replay ) {
delete m_replay ;
}
2020-06-21 18:19:46 +00:00
m_replay = new DeviceReplay ( xml ) ;
2020-06-11 19:58:34 +00:00
}
2020-06-06 20:53:47 +00:00
2020-06-11 19:58:34 +00:00
void DeviceConnectionManager : : replay ( QFile * file )
{
reset ( ) ;
2020-06-06 20:53:47 +00:00
if ( m_replay ) {
2020-06-11 19:58:34 +00:00
delete m_replay ;
}
if ( file ) {
2020-06-21 18:19:46 +00:00
m_replay = new DeviceReplay ( file ) ;
2020-06-06 20:53:47 +00:00
} else {
2020-06-11 19:58:34 +00:00
// nullptr turns off replay
m_replay = nullptr ;
}
}
2020-06-15 21:15:37 +00:00
DeviceConnection * DeviceConnectionManager : : openConnection ( const QString & type , const QString & name )
{
if ( ! s_factories . contains ( type ) ) {
qWarning ( ) < < " Unknown device connection type: " < < type ;
return nullptr ;
}
2020-06-16 16:19:32 +00:00
if ( m_connections . contains ( name ) ) {
qWarning ( ) < < " connection to " < < name < < " already open " ;
2020-06-15 21:15:37 +00:00
return nullptr ;
}
2020-07-09 22:07:03 +00:00
// Recording/replay (if any) is handled by the connection.
2020-06-15 21:15:37 +00:00
DeviceConnection * conn = s_factories [ type ] ( name , m_record , m_replay ) ;
if ( conn ) {
if ( conn - > open ( ) ) {
2020-06-16 16:19:32 +00:00
m_connections [ name ] = conn ;
2020-06-15 21:15:37 +00:00
} else {
qWarning ( ) . noquote ( ) < < " unable to open " < < type < < " connection to " < < name ;
delete conn ;
conn = nullptr ;
}
} else {
qWarning ( ) < < " unable to create " < < type < < " connection to " < < name ;
}
return conn ;
}
2020-07-09 22:07:03 +00:00
// Called by connections to deregister themselves.
2020-06-15 21:15:37 +00:00
void DeviceConnectionManager : : connectionClosed ( DeviceConnection * conn )
{
Q_ASSERT ( conn ) ;
const QString & type = conn - > type ( ) ;
const QString & name = conn - > name ( ) ;
2020-06-16 16:19:32 +00:00
if ( m_connections . contains ( name ) ) {
if ( m_connections [ name ] = = conn ) {
m_connections . remove ( name ) ;
2020-06-15 21:15:37 +00:00
} else {
2020-06-16 16:19:32 +00:00
qWarning ( ) < < " connection to " < < name < < " not created by openConnection! " ;
2020-06-15 21:15:37 +00:00
}
} else {
qWarning ( ) < < type < < " connection to " < < name < < " missing " ;
}
}
// 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 ;
}
2020-07-09 22:07:03 +00:00
// Since there are relatively few connection types, don't bother with a CRTP
// parent class. Instead, this macro defines the factory method, and the
// subclass will need to declare createInstance() and TYPE manually.
2020-06-15 21:15:37 +00:00
# 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 ) ) ; }
2020-06-11 19:58:34 +00:00
// MARK: -
2020-06-11 20:27:52 +00:00
// MARK: Device manager events
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
// See XmlReplayEvent discussion of complex data types above.
2020-06-16 16:19:32 +00:00
class GetAvailableSerialPortsEvent : public XmlReplayBase < GetAvailableSerialPortsEvent >
2020-06-11 19:58:34 +00:00
{
public :
QList < SerialPortInfo > m_ports ;
protected :
virtual void write ( QXmlStreamWriter & xml ) const
{
xml < < m_ports ;
}
virtual void read ( QXmlStreamReader & xml )
{
xml > > m_ports ;
}
} ;
2020-06-16 16:19:32 +00:00
REGISTER_XMLREPLAYEVENT ( " getAvailableSerialPorts " , GetAvailableSerialPortsEvent ) ;
2020-06-11 19:58:34 +00:00
2020-06-16 16:19:32 +00:00
QList < SerialPortInfo > DeviceConnectionManager : : getAvailableSerialPorts ( )
2020-06-11 19:58:34 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-16 16:19:32 +00:00
GetAvailableSerialPortsEvent event ;
2020-06-11 19:58:34 +00:00
if ( ! m_replay ) {
2020-07-09 22:07:03 +00:00
// Query the actual hardware present.
2020-06-06 20:53:47 +00:00
for ( auto & info : QSerialPortInfo : : availablePorts ( ) ) {
2020-06-11 19:58:34 +00:00
event . m_ports . append ( SerialPortInfo ( info ) ) ;
}
} else {
2020-06-16 16:19:32 +00:00
auto replayEvent = m_replay - > getNextEvent < GetAvailableSerialPortsEvent > ( ) ;
2020-06-11 19:58:34 +00:00
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 ;
2020-06-06 20:53:47 +00:00
}
}
2020-06-11 19:58:34 +00:00
m_serialPorts = event . m_ports ;
2020-06-06 20:53:47 +00:00
2020-06-11 19:58:34 +00:00
event . record ( m_record ) ;
return event . m_ports ;
2020-06-06 20:53:47 +00:00
}
2020-06-11 19:58:34 +00:00
2020-06-05 21:01:58 +00:00
// MARK: -
2020-06-11 20:27:52 +00:00
// MARK: Serial port info
2020-06-04 18:32:03 +00:00
2020-07-09 22:07:03 +00:00
/*
* This class is both a drop - in replacement for QSerialPortInfo and
* supports XML serialization for the GetAvailableSerialPortsEvent above .
*/
2020-06-04 18:32:03 +00:00
SerialPortInfo : : SerialPortInfo ( const QSerialPortInfo & other )
{
2020-06-05 21:01:58 +00:00
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 ( ) ;
}
}
2020-06-04 18:32:03 +00:00
}
SerialPortInfo : : SerialPortInfo ( const SerialPortInfo & other )
2020-06-05 21:01:58 +00:00
: m_info ( other . m_info )
{
}
SerialPortInfo : : SerialPortInfo ( const QString & data )
2020-06-04 18:32:03 +00:00
{
2020-06-05 21:01:58 +00:00
QXmlStreamReader xml ( data ) ;
xml . readNextStartElement ( ) ;
xml > > * this ;
2020-06-04 18:32:03 +00:00
}
2020-06-11 19:58:34 +00:00
SerialPortInfo : : SerialPortInfo ( )
{
}
2020-07-09 22:07:03 +00:00
// TODO: This method is a temporary wrapper that mimics the QSerialPortInfo interface until we begin refactoring.
2020-06-04 18:32:03 +00:00
QList < SerialPortInfo > SerialPortInfo : : availablePorts ( )
{
2020-06-16 16:19:32 +00:00
return DeviceConnectionManager : : getInstance ( ) . getAvailableSerialPorts ( ) ;
2020-06-04 18:32:03 +00:00
}
2020-06-05 21:01:58 +00:00
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 " ;
}
2020-06-11 19:58:34 +00:00
xml . skipCurrentElement ( ) ;
2020-06-05 21:01:58 +00:00
return xml ;
}
SerialPortInfo : : operator QString ( ) const
{
QString out ;
QXmlStreamWriter xml ( & out ) ;
xml < < * this ;
return out ;
}
2020-06-11 19:58:34 +00:00
bool SerialPortInfo : : operator = = ( const SerialPortInfo & other ) const
{
return m_info = = other . m_info ;
}
2020-06-11 20:27:52 +00:00
2020-06-30 15:35:23 +00:00
// MARK: -
// MARK: Device connection base class
2020-07-09 22:07:03 +00:00
/*
* Event recorded in the Device Connection Manager XML stream that indicates
* a connection was opened ( or attempted ) . On success , a ConnectionEvent
* ( see below ) will begin the connection ' s substream .
*/
2020-06-30 15:35:23 +00:00
class OpenConnectionEvent : public XmlReplayBase < OpenConnectionEvent >
{
public :
OpenConnectionEvent ( ) { }
OpenConnectionEvent ( const QString & type , const QString & name )
{
set ( " type " , type ) ;
set ( " name " , name ) ;
}
virtual const QString id ( ) const { return m_values [ " name " ] ; }
} ;
REGISTER_XMLREPLAYEVENT ( " openConnection " , OpenConnectionEvent ) ;
2020-07-09 22:07:03 +00:00
/*
* Event created when a connection is successfully opened , used as the
* enclosing tag for the connection substream .
*/
2020-06-29 01:28:25 +00:00
class ConnectionEvent : public XmlReplayBase < ConnectionEvent >
{
public :
ConnectionEvent ( ) { Q_ASSERT ( false ) ; } // Implement if we ever support string-based substreams
2020-06-30 15:35:23 +00:00
ConnectionEvent ( const OpenConnectionEvent & trigger )
2020-06-29 01:28:25 +00:00
{
copy ( trigger ) ;
}
virtual const QString id ( ) const
{
QString time = m_time . toString ( " yyyyMMdd.HHmmss.zzz " ) ;
return m_values [ " name " ] + " - " + time ;
}
} ;
REGISTER_XMLREPLAYEVENT ( " connection " , ConnectionEvent ) ;
2020-07-09 22:07:03 +00:00
/*
* ConnectionRecorder / ConnectionReplay subclasses of XmlRecorder / XmlReplay
*
* Used by DeviceConnection subclasses to record their activity , such as
* configuration and data sent and received .
*/
2020-06-21 18:19:46 +00:00
class ConnectionRecorder : public XmlRecorder
{
public :
2020-06-29 01:28:25 +00:00
ConnectionRecorder ( XmlRecorder * parent , const ConnectionEvent & event ) : XmlRecorder ( parent , event . id ( ) , event . tag ( ) )
{
Q_ASSERT ( m_xml ) ;
event . writeTag ( * m_xml ) ;
m_xml - > writeAttribute ( " oscar " , getVersion ( ) . toString ( ) ) ;
}
2020-06-21 18:19:46 +00:00
} ;
class ConnectionReplay : public XmlReplay
{
public :
2020-06-29 01:28:25 +00:00
ConnectionReplay ( XmlReplay * parent , const ConnectionEvent & event ) : XmlReplay ( parent , event . id ( ) , event . tag ( ) ) { }
2020-06-21 18:19:46 +00:00
} ;
2020-07-09 22:07:03 +00:00
// Device connection base class
2020-06-15 21:15:37 +00:00
DeviceConnection : : DeviceConnection ( const QString & name , XmlRecorder * record , XmlReplay * replay )
: m_name ( name ) , m_record ( record ) , m_replay ( replay ) , m_opened ( false )
{
}
DeviceConnection : : ~ DeviceConnection ( )
{
}
2020-07-09 22:07:03 +00:00
/*
* Generic get / set events
*/
2020-06-18 20:14:38 +00:00
class SetValueEvent : public XmlReplayBase < SetValueEvent >
2020-06-12 20:44:07 +00:00
{
public :
2020-06-18 20:14:38 +00:00
SetValueEvent ( ) { }
2020-06-12 20:44:07 +00:00
SetValueEvent ( const QString & name , int value )
{
set ( name , value ) ;
}
2020-06-18 20:14:38 +00:00
virtual const QString id ( ) const { return m_keys . first ( ) ; }
2020-06-12 02:29:46 +00:00
} ;
2020-06-18 20:14:38 +00:00
REGISTER_XMLREPLAYEVENT ( " set " , SetValueEvent ) ;
2020-06-12 02:29:46 +00:00
2020-06-18 20:14:38 +00:00
class GetValueEvent : public XmlReplayBase < GetValueEvent >
{
public :
GetValueEvent ( ) { }
GetValueEvent ( const QString & id )
{
set ( id , 0 ) ;
}
virtual const QString id ( ) const { return m_keys . first ( ) ; }
void setValue ( qint64 value )
{
if ( m_keys . isEmpty ( ) ) {
qWarning ( ) < < " setValue: get event missing key " ;
return ;
}
set ( m_keys . first ( ) , value ) ;
}
QString value ( ) const
{
if ( m_keys . isEmpty ( ) ) {
qWarning ( ) < < " getValue: get event missing key " ;
return 0 ;
}
return get ( m_keys . first ( ) ) ;
}
} ;
REGISTER_XMLREPLAYEVENT ( " get " , GetValueEvent ) ;
2020-07-09 22:07:03 +00:00
/*
* Event recorded in the Device Connection Manager XML stream when a
* open connection is closed . This is the complement to a successful
* OpenConnectionEvent ( see above ) , and does not appear when the connection
* failed to open .
*/
2020-06-18 20:14:38 +00:00
class CloseConnectionEvent : public XmlReplayBase < CloseConnectionEvent >
{
public :
CloseConnectionEvent ( ) { }
CloseConnectionEvent ( const QString & type , const QString & name )
{
set ( " type " , type ) ;
set ( " name " , name ) ;
}
virtual const QString id ( ) const { return m_values [ " name " ] ; }
} ;
REGISTER_XMLREPLAYEVENT ( " closeConnection " , CloseConnectionEvent ) ;
class ClearConnectionEvent : public XmlReplayBase < ClearConnectionEvent >
{
} ;
REGISTER_XMLREPLAYEVENT ( " clear " , ClearConnectionEvent ) ;
class FlushConnectionEvent : public XmlReplayBase < FlushConnectionEvent >
{
} ;
REGISTER_XMLREPLAYEVENT ( " flush " , FlushConnectionEvent ) ;
2020-07-09 22:07:03 +00:00
/*
* Event representing data received from a device
*
* The data is stored as hexadecimal data in the XML tag ' s contents .
*/
2020-06-18 20:14:38 +00:00
class ReceiveDataEvent : public XmlReplayBase < ReceiveDataEvent >
{
virtual bool usesData ( ) const { return true ; }
} ;
REGISTER_XMLREPLAYEVENT ( " rx " , ReceiveDataEvent ) ;
2020-07-09 22:07:03 +00:00
/*
* Event representing data sent to a device
*
* The data is stored as hexadecimal data in the XML tag ' s contents .
*
* These events are random - access events ( see discussion above ) , which cause
* subsequent event retrieval to begin searching after the transmission
* event .
*
* Since the data sent is used as the ID for these events , we can treat
* these like distinct " commands " that that elicit a deterministic response ,
* which can be replayed independently of other events if desired .
*
* Of course , for any device that has more complex internal state ( where
* responses to multiple transmissions of a particular " command " depend
* on some intervening event ) , this reordering will not be accurate .
*
* But the intent is that some small changes to client code should still
* work with existing recordings before requiring creation of new ones .
*/
2020-06-18 20:14:38 +00:00
class TransmitDataEvent : public XmlReplayBase < TransmitDataEvent >
{
virtual bool usesData ( ) const { return true ; }
public :
virtual const QString id ( ) const { return m_data ; }
2020-06-19 17:10:43 +00:00
virtual bool randomAccess ( ) const { return true ; }
2020-06-18 20:14:38 +00:00
} ;
REGISTER_XMLREPLAYEVENT ( " tx " , TransmitDataEvent ) ;
2020-06-12 02:29:46 +00:00
2020-07-09 22:07:03 +00:00
/*
* Event representing a " readyRead " signal emitted by a physical device .
*
* These events are marked as " signal " events ( see discussion of m_signal
* above ) so that connections will automatically send them to clients when
* the preceding event ( API call ) is processed .
*/
2020-06-19 02:05:43 +00:00
class ReadyReadEvent : public XmlReplayBase < ReadyReadEvent >
{
public :
2020-07-09 22:07:03 +00:00
// Use the connection's slot that receives readyRead signals.
2020-06-19 02:05:43 +00:00
ReadyReadEvent ( ) { m_signal = " onReadyRead " ; }
} ;
REGISTER_XMLREPLAYEVENT ( " readyRead " , ReadyReadEvent ) ;
2020-06-12 20:44:07 +00:00
2020-06-15 21:15:37 +00:00
// MARK: -
// MARK: Serial port connection
2020-07-09 22:07:03 +00:00
/*
* Serial port connection class
*
* This class wraps calls to an underlying QSerialPort with the logic
* necessary to record and replay arbitrary serial port activity .
* ( or , at least , the serial port functionality currently used by OSCAR ) .
*
* Clients obtain a connection instance via DeviceConnectionManager : : openConnection ( )
* or openSerialPortConnection ( for convenience , if they require a serial port ) .
*/
2020-06-15 21:15:37 +00:00
REGISTER_DEVICECONNECTION ( " serial " , SerialPortConnection ) ;
2020-06-12 02:29:46 +00:00
2020-06-15 21:15:37 +00:00
SerialPortConnection : : SerialPortConnection ( const QString & name , XmlRecorder * record , XmlReplay * replay )
: DeviceConnection ( name , record , replay )
2020-06-12 02:29:46 +00:00
{
connect ( & m_port , SIGNAL ( readyRead ( ) ) , this , SLOT ( onReadyRead ( ) ) ) ;
}
2020-06-13 20:58:46 +00:00
SerialPortConnection : : ~ SerialPortConnection ( )
2020-06-12 02:29:46 +00:00
{
2020-07-09 22:07:03 +00:00
// This will only be false if the connection failed to open immediately after construction.
2020-06-15 21:15:37 +00:00
if ( m_opened ) {
close ( ) ;
DeviceConnectionManager : : getInstance ( ) . connectionClosed ( this ) ;
}
2020-06-12 02:29:46 +00:00
disconnect ( & m_port , SIGNAL ( readyRead ( ) ) , this , SLOT ( onReadyRead ( ) ) ) ;
}
2020-07-09 22:07:03 +00:00
/*
* Try to open the physical port ( or replay a previous attempt ) , returning
* false if the port was not opened .
*
* DeviceConnectionManager : : openConnection calls this immediately after
* creating a connection instance , and will only return open connections
* to clients .
*/
2020-06-15 21:15:37 +00:00
bool SerialPortConnection : : open ( )
2020-06-12 02:29:46 +00:00
{
2020-06-15 21:15:37 +00:00
if ( m_opened ) {
qWarning ( ) < < " serial connection to " < < m_name < < " already opened " ;
return false ;
}
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-29 01:28:25 +00:00
OpenConnectionEvent * replayEvent = nullptr ;
2020-06-18 20:14:38 +00:00
OpenConnectionEvent event ( " serial " , m_name ) ;
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
2020-07-09 22:07:03 +00:00
// TODO: move this into SerialPortConnection::openDevice() and move
// the rest of the logic up to DeviceConnection::open().
2020-06-18 20:14:38 +00:00
m_port . setPortName ( m_name ) ;
checkResult ( m_port . open ( QSerialPort : : ReadWrite ) , event ) ;
} else {
2020-06-29 01:28:25 +00:00
replayEvent = m_replay - > getNextEvent < OpenConnectionEvent > ( event . id ( ) ) ;
2020-06-18 20:14:38 +00:00
if ( replayEvent ) {
event . copyIf ( replayEvent ) ;
} else {
event . set ( " error " , QSerialPort : : DeviceNotFoundError ) ;
}
}
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
event . record ( m_record ) ;
2020-06-15 21:15:37 +00:00
m_opened = event . ok ( ) ;
2020-06-29 01:28:25 +00:00
if ( m_opened ) {
// open a connection substream for connection events
if ( m_record ) {
ConnectionEvent connEvent ( event ) ;
m_record = new ConnectionRecorder ( m_record , connEvent ) ;
}
if ( m_replay ) {
Q_ASSERT ( replayEvent ) ;
ConnectionEvent connEvent ( * replayEvent ) ; // we need to use the replay's timestamp to find the referenced substream
m_replay = new ConnectionReplay ( m_replay , connEvent ) ;
}
}
2020-06-18 20:14:38 +00:00
2020-06-12 20:44:07 +00:00
return event . ok ( ) ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : setBaudRate ( qint32 baudRate , QSerialPort : : Directions directions )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-12 02:29:46 +00:00
SetValueEvent event ( " baudRate " , baudRate ) ;
event . set ( " directions " , directions ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . setBaudRate ( baudRate , directions ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < SetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
return event . ok ( ) ;
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : setDataBits ( QSerialPort : : DataBits dataBits )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-12 02:29:46 +00:00
SetValueEvent event ( " setDataBits " , dataBits ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . setDataBits ( dataBits ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < SetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
return event . ok ( ) ;
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : setParity ( QSerialPort : : Parity parity )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-12 02:29:46 +00:00
SetValueEvent event ( " setParity " , parity ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . setParity ( parity ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < SetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
return event . ok ( ) ;
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : setStopBits ( QSerialPort : : StopBits stopBits )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-12 02:29:46 +00:00
SetValueEvent event ( " setStopBits " , stopBits ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . setStopBits ( stopBits ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < SetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
return event . ok ( ) ;
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : setFlowControl ( QSerialPort : : FlowControl flowControl )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-12 02:29:46 +00:00
SetValueEvent event ( " setFlowControl " , flowControl ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . setFlowControl ( flowControl ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < SetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
return event . ok ( ) ;
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : clear ( QSerialPort : : Directions directions )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
ClearConnectionEvent event ;
2020-06-12 20:44:07 +00:00
event . set ( " directions " , directions ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . clear ( directions ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < ClearConnectionEvent > ( ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 20:44:07 +00:00
return event . ok ( ) ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
qint64 SerialPortConnection : : bytesAvailable ( ) const
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
GetValueEvent event ( " bytesAvailable " ) ;
qint64 result ;
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
result = m_port . bytesAvailable ( ) ;
event . setValue ( result ) ;
checkResult ( result , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < GetValueEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
bool ok ;
result = event . value ( ) . toLong ( & ok ) ;
if ( ! ok ) {
qWarning ( ) < < event . tag ( ) < < event . id ( ) < < " has bad value " ;
}
}
event . record ( m_record ) ;
2020-06-12 20:44:07 +00:00
return result ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
qint64 SerialPortConnection : : read ( char * data , qint64 maxSize )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
qint64 len ;
ReceiveDataEvent event ;
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
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 ) ;
} else {
2020-06-19 02:05:43 +00:00
auto replayEvent = m_replay - > getNextEvent < ReceiveDataEvent > ( ) ;
2020-06-18 20:14:38 +00:00
event . copyIf ( replayEvent ) ;
if ( ! replayEvent ) {
qWarning ( ) < < " reading data past replay " ;
event . set ( " len " , - 1 ) ;
event . set ( " error " , QSerialPort : : ReadError ) ;
}
bool ok ;
len = event . get ( " len " ) . toLong ( & ok ) ;
2020-06-19 02:05:43 +00:00
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 {
2020-06-18 20:14:38 +00:00
qWarning ( ) < < event < < " has bad len " ;
len = - 1 ;
}
2020-06-12 20:44:07 +00:00
}
2020-06-19 02:05:43 +00:00
event . record ( m_record ) ;
2020-06-12 20:44:07 +00:00
return len ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
qint64 SerialPortConnection : : write ( const char * data , qint64 maxSize )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
qint64 len ;
TransmitDataEvent event ;
2020-06-12 20:44:07 +00:00
event . setData ( data , maxSize ) ;
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
len = m_port . write ( data , maxSize ) ;
event . set ( " len " , len ) ;
if ( len ! = maxSize ) {
event . set ( " req " , maxSize ) ;
}
checkResult ( len , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < TransmitDataEvent > ( event . id ( ) ) ;
event . copyIf ( replayEvent ) ;
if ( ! replayEvent ) {
qWarning ( ) < < " writing data past replay " ;
event . set ( " len " , - 1 ) ;
2020-07-10 01:19:54 +00:00
event . set ( " error " , QSerialPort : : WriteError ) ;
2020-06-18 20:14:38 +00:00
}
bool ok ;
len = event . get ( " len " ) . toLong ( & ok ) ;
2020-07-09 22:07:03 +00:00
// No need to copy any data, since the event already contains it.
2020-06-18 20:14:38 +00:00
if ( ! ok ) {
qWarning ( ) < < event < < " has bad len " ;
len = - 1 ;
}
2020-06-12 20:44:07 +00:00
}
2020-06-18 20:14:38 +00:00
event . record ( m_record ) ;
2020-06-12 20:44:07 +00:00
return len ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
bool SerialPortConnection : : flush ( )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
FlushConnectionEvent event ;
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
checkResult ( m_port . flush ( ) , event ) ;
} else {
auto replayEvent = m_replay - > getNextEvent < FlushConnectionEvent > ( ) ;
event . copyIf ( replayEvent ) ;
}
event . record ( m_record ) ;
2020-06-12 20:44:07 +00:00
return event . ok ( ) ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
void SerialPortConnection : : close ( )
2020-06-12 02:29:46 +00:00
{
2020-06-29 01:28:25 +00:00
if ( m_opened ) {
// close event substream first
if ( m_record ) {
2020-07-10 01:19:54 +00:00
m_record = m_record - > closeSubstream ( ) ;
2020-06-29 01:28:25 +00:00
}
if ( m_replay ) {
2020-07-10 01:19:54 +00:00
m_replay = m_replay - > closeSubstream ( ) ;
2020-06-29 01:28:25 +00:00
}
}
2020-06-19 02:05:43 +00:00
XmlReplayLock lock ( this , m_replay ) ;
2020-06-18 20:14:38 +00:00
CloseConnectionEvent event ( " serial " , m_name ) ;
2020-06-12 20:44:07 +00:00
2020-06-29 01:28:25 +00:00
// TODO: We'll also need to include a loader ID and stream version number
2020-06-12 20:44:07 +00:00
// 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.
2020-07-09 22:07:03 +00:00
// NOTE: This may only be required for downloads rather than all connections.
2020-06-12 20:44:07 +00:00
2020-06-18 20:14:38 +00:00
if ( ! m_replay ) {
2020-07-09 22:07:03 +00:00
// TODO: move this into SerialPortConnection::closeDevice() and move
// the remaining logic up to DeviceConnection::close().
2020-06-18 20:14:38 +00:00
m_port . close ( ) ;
checkError ( event ) ;
} else {
2020-06-29 01:28:25 +00:00
auto replayEvent = m_replay - > getNextEvent < CloseConnectionEvent > ( event . id ( ) ) ;
2020-06-18 20:14:38 +00:00
if ( replayEvent ) {
event . copyIf ( replayEvent ) ;
} else {
event . set ( " error " , QSerialPort : : ResourceError ) ;
}
}
event . record ( m_record ) ;
2020-06-12 02:29:46 +00:00
}
2020-06-13 20:58:46 +00:00
void SerialPortConnection : : onReadyRead ( )
2020-06-12 02:29:46 +00:00
{
2020-06-19 02:05:43 +00:00
{
// 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 ) ;
2020-06-19 17:10:43 +00:00
// Unlocking will queue any subsequent signals.
2020-06-19 02:05:43 +00:00
}
2020-06-12 20:44:07 +00:00
2020-06-19 02:05:43 +00:00
// Because clients typically leave this as Qt::AutoConnection, the below emit may
// execute immediately in this thread, so we have to release the lock before sending
// the signal.
2020-06-12 20:44:07 +00:00
2020-06-19 02:05:43 +00:00
// 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.
2020-06-12 02:29:46 +00:00
emit readyRead ( ) ;
}
2020-06-11 20:27:52 +00:00
2020-07-09 22:07:03 +00:00
// Check the boolean returned by a serial port call and the port's error status, and update the event accordingly.
2020-06-18 20:14:38 +00:00
void SerialPortConnection : : checkResult ( bool ok , XmlReplayEvent & event ) const
2020-06-13 20:58:46 +00:00
{
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
}
2020-07-09 22:07:03 +00:00
// Check the length returned by a serial port call and the port's error status, and update the event accordingly.
2020-06-18 20:14:38 +00:00
void SerialPortConnection : : checkResult ( qint64 len , XmlReplayEvent & event ) const
2020-06-13 20:58:46 +00:00
{
QSerialPort : : SerialPortError error = m_port . error ( ) ;
if ( len < 0 | | error ! = QSerialPort : : NoError ) {
event . set ( " error " , error ) ;
}
}
2020-07-09 22:07:03 +00:00
// Check the port's error status, and update the event accordingly.
2020-06-18 20:14:38 +00:00
void SerialPortConnection : : checkError ( XmlReplayEvent & event ) const
2020-06-13 20:58:46 +00:00
{
QSerialPort : : SerialPortError error = m_port . error ( ) ;
if ( error ! = QSerialPort : : NoError ) {
event . set ( " error " , error ) ;
}
}
2020-06-15 21:15:37 +00:00
// MARK: -
// MARK: SerialPort legacy class
2020-07-09 22:07:03 +00:00
/*
* SerialPort drop - in replacement for QSerialPort
*
* This class mimics the interface of QSerialPort for client code , while
* using DeviceConnectionManager to open the SerialPortConnection , allowing
* for transparent recording and replay .
*/
2020-06-15 21:15:37 +00:00
SerialPort : : SerialPort ( )
2020-06-18 20:14:38 +00:00
: m_conn ( nullptr )
2020-06-15 21:15:37 +00:00
{
}
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 ) {
2020-07-09 22:07:03 +00:00
// Listen for readyRead events from the connection so that we can relay them to the client.
2020-06-15 21:15:37 +00:00
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 ( )
{
2020-07-09 22:07:03 +00:00
// Relay readyRead events from the connection on to the client.
2020-06-15 21:15:37 +00:00
emit readyRead ( ) ;
}