2020-07-09 22:07:03 +00:00
/* Device Connection Manager
2020-06-04 18:32:03 +00:00
*
2024-01-13 20:27:48 +00:00
* Copyright ( c ) 2020 - 2024 The OSCAR Team
2020-06-04 18:32:03 +00:00
*
* 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 . */
# ifndef DEVICECONNECTION_H
# define DEVICECONNECTION_H
// TODO: This file will eventually abstract serial port or bluetooth (or other)
// connections to devices. For now it just supports serial ports.
# include <QtSerialPort/QSerialPort>
2020-06-05 21:01:58 +00:00
# include <QHash>
# include <QVariant>
2020-06-15 21:15:37 +00:00
2020-07-09 22:07:03 +00:00
/*
* Device connection base class
*
* Clients obtain a connection instance via DeviceConnectionManager : : openConnection ( ) .
* See SerialPortConnection for the only current concrete implementation .
*
* See DeviceConnectionManager for the primary interface to device
* connections .
*/
2020-06-15 21:15:37 +00:00
class DeviceConnection : public QObject
{
Q_OBJECT
protected :
2020-07-09 22:07:03 +00:00
// Constructor is protected so that only subclasses and DeviceConnectionManager can call it.
2020-06-30 15:35:23 +00:00
DeviceConnection ( const QString & name , class XmlRecorder * record , class XmlReplay * replay ) ;
2020-06-15 21:15:37 +00:00
2020-07-09 22:07:03 +00:00
const QString & m_name ; // port/device identifier used to open the connection
XmlRecorder * m_record ; // nullptr or pointer to recorder instance
XmlReplay * m_replay ; // nullptr or pointer to replay instance
bool m_opened ; // true if open() succeeded
2020-06-15 21:15:37 +00:00
virtual bool open ( ) = 0 ;
friend class DeviceConnectionManager ;
public :
// See DeviceConnectionManager::openConnection() to create connections.
virtual ~ DeviceConnection ( ) ;
virtual const QString & type ( ) const = 0 ;
const QString & name ( ) const { return m_name ; }
typedef DeviceConnection * ( * FactoryMethod ) ( const QString & name , XmlRecorder * record , XmlReplay * replay ) ;
} ;
2020-07-09 22:07:03 +00:00
/*
* Device connection manager
*
* Principal class used to abstract direct connections to devices ,
* eventually encompassing serial port , Bluetooth , and BLE . This class not
* only provides an abstraction for the specific connection type ( where
* possible ) , but it also provides the capability to record and replay
* connections transparently to clients .
*
* Clients obtain the singleton instance via DeviceConnectionManager : : getInstance ( ) .
*
* TODO : Eventually they will be able to connect to signals when a device
* becomes available or is removed . For now they need to call instance - >
* getAvailableSerialPorts ( ) to poll .
*
* When a device becomes available , clients call instance - > openSerialPortConnection ( ) .
* TODO : This will eventually probably be openConnection ( ) once Bluetooth is
* supported .
*
* To enable recording and replay of connections , call instance - > record ( )
* and / or instance - > replay ( ) , which will cause all subsequent connections to
* be recorded or replayed , respectively . Passing nullptr to record ( ) or
* replay ( ) will turn off recording / replaying for subsequent connections .
* This allows an application to record or replay connection data
* transparently to client code that assumes it is talking directly to a
* real device .
*/
2020-06-06 20:53:47 +00:00
class DeviceConnectionManager : public QObject
{
Q_OBJECT
private :
2020-07-09 22:07:03 +00:00
// See getInstance() for creating/using the device connection manager.
2020-06-06 20:53:47 +00:00
DeviceConnectionManager ( ) ;
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
XmlRecorder * m_record ; // nullptr or pointer to recorder instance
XmlReplay * m_replay ; // nullptr or pointer to replay instance
2020-06-11 19:58:34 +00:00
2020-07-09 22:07:03 +00:00
QList < class SerialPortInfo > m_serialPorts ; // currently available serial ports
2020-06-11 19:58:34 +00:00
void reset ( ) { // clear state
m_serialPorts . clear ( ) ;
}
2020-06-06 20:53:47 +00:00
2020-07-09 22:07:03 +00:00
QHash < QString , DeviceConnection * > m_connections ; // currently open connections
2020-06-15 21:15:37 +00:00
2020-06-06 20:53:47 +00:00
public :
2020-07-09 22:07:03 +00:00
//! \brief Obtain pointer to global DeviceConnectionManager instance, creating it if necessary.
2020-06-06 20:53:47 +00:00
static DeviceConnectionManager & getInstance ( ) ;
2020-07-09 22:07:03 +00:00
//! \brief Open a connection to a device, returning an instance of the appropriate type, or nullptr if the connection couldn't be opened.
2020-06-15 21:15:37 +00:00
class DeviceConnection * openConnection ( const QString & type , const QString & name ) ;
2020-07-09 22:07:03 +00:00
//! \brief Open a serial port connection (convenience function, hopefully temporary), returning nullptr if the connection couldn't be opened.
2020-06-15 21:15:37 +00:00
static class SerialPortConnection * openSerialPortConnection ( const QString & portName ) ; // temporary
2020-06-06 20:53:47 +00:00
2020-07-09 22:07:03 +00:00
//! \brief Return the list of currently available serial ports.
2020-06-16 16:19:32 +00:00
QList < class SerialPortInfo > getAvailableSerialPorts ( ) ;
2020-07-09 22:07:03 +00:00
2020-06-06 20:53:47 +00:00
// TODO: method to start a polling thread that maintains the list of ports
2020-07-09 22:07:03 +00:00
// TODO: emit signal when new port is detected (or removed)
2020-06-06 20:53:47 +00:00
2020-07-09 22:07:03 +00:00
//! \brief Record all subsequent device activity to the given file, and subsequent connections to separate files alongside it. Passing nullptr turns off recording.
2020-06-11 19:58:34 +00:00
void record ( class QFile * stream ) ;
2020-07-09 22:07:03 +00:00
// Record all subsequent device activity to the given string. Primarily for testing; connection recordings are not supported.
2020-06-11 19:58:34 +00:00
void record ( QString & string ) ;
2020-07-09 22:07:03 +00:00
//! \brief Replay the activity previously recorded in the given file, allowing for some simple variation in the order of API calls. Passing nullptr turns off replay.
2020-06-11 19:58:34 +00:00
void replay ( class QFile * stream ) ;
2020-07-09 22:07:03 +00:00
// Replay the activity represented by the given string. Primarily for testing; connection replay is not supported.
2020-06-11 19:58:34 +00:00
void replay ( const QString & string ) ;
2020-06-06 20:53:47 +00:00
2020-07-09 22:07:03 +00:00
// DeviceConnection subclasses registration, not intended for client use.
2020-06-15 21:15:37 +00:00
protected :
2020-07-10 16:50:34 +00:00
static QHash < QString , DeviceConnection : : FactoryMethod > & factories ( ) ;
2020-06-15 21:15:37 +00:00
public :
static bool registerClass ( const QString & type , DeviceConnection : : FactoryMethod factory ) ;
static class DeviceConnection * createInstance ( const QString & type ) ;
2020-06-06 20:53:47 +00:00
2020-07-09 22:07:03 +00:00
// Currently public only so that connections can deregister themselves.
// Eventually this could move to protected if that gets handled by the
// DeviceConnection destructor and DeviceConnection is declared a friend.
2020-06-15 21:15:37 +00:00
void connectionClosed ( DeviceConnection * conn ) ;
2020-06-13 20:58:46 +00:00
} ;
2020-07-09 22:07:03 +00:00
/*
* Serial port connection class
*
* This class provides functionality equivalent to QSerialPort , but
* specifically represents an opened connection rather than the port itself .
* ( See the SerialPort class for the QSerialPort equivalent . ) This class
* also provides support for recording and replay of an opened serial port
* connection .
*
* TODO : This class may eventually be internal to DeviceConnection , if its
* interface shares enough in common with Bluetooth and / or BLE .
*/
2020-06-13 20:58:46 +00:00
class SerialPortConnection : public DeviceConnection
2020-06-04 18:32:03 +00:00
{
2020-06-12 02:29:46 +00:00
Q_OBJECT
private :
2020-07-09 22:07:03 +00:00
QSerialPort m_port ; // physical port used by connection
2020-06-30 15:35:23 +00:00
void checkResult ( bool ok , class XmlReplayEvent & event ) const ;
2020-06-18 20:14:38 +00:00
void checkResult ( qint64 len , XmlReplayEvent & event ) const ;
void checkError ( XmlReplayEvent & event ) const ;
2020-06-15 21:15:37 +00:00
void close ( ) ;
2020-06-12 02:29:46 +00:00
private slots :
void onReadyRead ( ) ;
signals :
2020-07-09 22:07:03 +00:00
// The readyRead() signal is emitted with the same semantics as QSerialPort::readyRead().
2020-06-12 02:29:46 +00:00
void readyRead ( ) ;
2020-06-15 21:15:37 +00:00
protected :
SerialPortConnection ( const QString & name , XmlRecorder * record , XmlReplay * replay ) ;
virtual bool open ( ) ;
2020-06-13 20:58:46 +00:00
2020-06-15 21:15:37 +00:00
public :
2020-07-09 22:07:03 +00:00
// See DeviceConnectionManager::openConnection() or openSerialPortConnection() to create connections.
2020-06-13 20:58:46 +00:00
virtual ~ SerialPortConnection ( ) ;
2020-07-09 22:07:03 +00:00
// See QSerialPort for semantics of the below functions.
2020-06-12 02:29:46 +00:00
bool setBaudRate ( qint32 baudRate , QSerialPort : : Directions directions = QSerialPort : : AllDirections ) ;
bool setDataBits ( QSerialPort : : DataBits dataBits ) ;
bool setParity ( QSerialPort : : Parity parity ) ;
bool setStopBits ( QSerialPort : : StopBits stopBits ) ;
bool setFlowControl ( QSerialPort : : FlowControl flowControl ) ;
bool clear ( QSerialPort : : Directions directions = QSerialPort : : AllDirections ) ;
qint64 bytesAvailable ( ) const ;
qint64 read ( char * data , qint64 maxSize ) ;
qint64 write ( const char * data , qint64 maxSize ) ;
bool flush ( ) ;
2020-07-09 22:07:03 +00:00
// Subclass registration with DeviceConnectionManager, not intended for client use.
2020-06-15 21:15:37 +00:00
public :
static DeviceConnection * createInstance ( const QString & name , XmlRecorder * record , XmlReplay * replay ) ;
static const QString TYPE ;
static const bool registered ;
virtual const QString & type ( ) const { return TYPE ; }
2020-06-04 18:32:03 +00:00
} ;
2020-07-09 22:07:03 +00:00
/*
* SerialPort temporary class for legacy compatibility
*
* This class is a temporary drop - in replacement for QSerialPort for code
* that currently assumes serial port connectivity . Using this class instead
* of QSerialPort allows for recording and replay of connection data .
*
* See QSerialPort documentation for interface details . See
* DeviceConnectionManager : : record ( ) and replay ( ) for enabling recording
* and replay .
*
* See SerialPortConnection for implementation details .
*/
2020-06-15 21:15:37 +00:00
class SerialPort : public QObject
2020-06-13 20:58:46 +00:00
{
2020-06-15 21:15:37 +00:00
Q_OBJECT
private :
SerialPortConnection * m_conn ;
QString m_portName ;
private slots :
void onReadyRead ( ) ;
signals :
void readyRead ( ) ;
public :
SerialPort ( ) ;
virtual ~ SerialPort ( ) ;
void setPortName ( const QString & name ) ;
bool open ( QIODevice : : OpenMode mode ) ;
bool setBaudRate ( qint32 baudRate , QSerialPort : : Directions directions = QSerialPort : : AllDirections ) ;
bool setDataBits ( QSerialPort : : DataBits dataBits ) ;
bool setParity ( QSerialPort : : Parity parity ) ;
bool setStopBits ( QSerialPort : : StopBits stopBits ) ;
bool setFlowControl ( QSerialPort : : FlowControl flowControl ) ;
bool clear ( QSerialPort : : Directions directions = QSerialPort : : AllDirections ) ;
qint64 bytesAvailable ( ) const ;
qint64 read ( char * data , qint64 maxSize ) ;
qint64 write ( const char * data , qint64 maxSize ) ;
bool flush ( ) ;
void close ( ) ;
2020-06-13 20:58:46 +00:00
} ;
2020-07-09 22:07:03 +00:00
/*
* SerialPortInfo temporary class for legacy compatibility
*
* This class is a temporary drop - in replacement for QSerialPortInfo for
* code that currently assumes serial port connectivity . Using this class
* instead of QSerialPortInfo allows for recording and replay of port
* scanning .
*
* See QSerialPortInfo documentation for interface details . See
* DeviceConnectionManager : : record ( ) and replay ( ) for enabling recording
* and replay .
*
* TODO : This class ' s functionality may either become internal to
* DeviceConnection or may be moved to a generic port info class that
* supports Bluetooth and BLE as well as serial . Such a class might then be
* used instead of port " name " between DeviceConnectionManager and clients .
*/
2020-07-21 17:51:20 +00:00
class QXmlStreamWriter ;
class QXmlStreamReader ;
2020-06-05 21:01:58 +00:00
class SerialPortInfo
2020-06-04 18:32:03 +00:00
{
public :
static QList < SerialPortInfo > availablePorts ( ) ;
SerialPortInfo ( const SerialPortInfo & other ) ;
2020-06-05 21:01:58 +00:00
SerialPortInfo ( const QString & data ) ;
2020-06-11 19:58:34 +00:00
SerialPortInfo ( ) ;
2020-06-05 21:01:58 +00:00
inline QString portName ( ) const { return m_info [ " portName " ] . toString ( ) ; }
inline QString systemLocation ( ) const { return m_info [ " systemLocation " ] . toString ( ) ; }
inline QString description ( ) const { return m_info [ " description " ] . toString ( ) ; }
inline QString manufacturer ( ) const { return m_info [ " manufacturer " ] . toString ( ) ; }
inline QString serialNumber ( ) const { return m_info [ " serialNumber " ] . toString ( ) ; }
inline quint16 vendorIdentifier ( ) const { return m_info [ " vendorIdentifier " ] . toInt ( ) ; }
inline quint16 productIdentifier ( ) const { return m_info [ " productIdentifier " ] . toInt ( ) ; }
inline bool hasVendorIdentifier ( ) const { return m_info . contains ( " vendorIdentifier " ) ; }
inline bool hasProductIdentifier ( ) const { return m_info . contains ( " productIdentifier " ) ; }
inline bool isNull ( ) const { return m_info . isEmpty ( ) ; }
operator QString ( ) const ;
2020-06-30 15:35:23 +00:00
friend class QXmlStreamWriter & operator < < ( QXmlStreamWriter & xml , const SerialPortInfo & info ) ;
friend class QXmlStreamReader & operator > > ( QXmlStreamReader & xml , SerialPortInfo & info ) ;
2020-06-11 19:58:34 +00:00
bool operator = = ( const SerialPortInfo & other ) const ;
2023-03-30 21:11:28 +00:00
SerialPortInfo & operator = ( const SerialPortInfo & other ) = default ;
2020-06-04 18:32:03 +00:00
protected :
2020-06-05 21:01:58 +00:00
SerialPortInfo ( const class QSerialPortInfo & other ) ;
QHash < QString , QVariant > m_info ;
2020-06-06 20:53:47 +00:00
friend class DeviceConnectionManager ;
2020-06-04 18:32:03 +00:00
} ;
# endif // DEVICECONNECTION_H