/**************************************************************************** * * (c) 2009-2020 QGROUNDCONTROL PROJECT * * QGroundControl is licensed according to the terms in the file * COPYING.md in the root of the source code directory. * ****************************************************************************/ #pragma once #include #include #include #include #ifdef __android__ #include "qserialport.h" #else #include #endif #include #include // We use QSerialPort::SerialPortError in a signal so we must declare it as a meta type Q_DECLARE_METATYPE(QSerialPort::SerialPortError) #include "QGCConfig.h" #include "LinkConfiguration.h" #include "LinkInterface.h" Q_DECLARE_LOGGING_CATEGORY(SerialLinkLog) class LinkManager; /// SerialLink configuration class SerialConfiguration : public LinkConfiguration { Q_OBJECT public: SerialConfiguration(const QString& name); SerialConfiguration(SerialConfiguration* copy); Q_PROPERTY(int baud READ baud WRITE setBaud NOTIFY baudChanged) Q_PROPERTY(int dataBits READ dataBits WRITE setDataBits NOTIFY dataBitsChanged) Q_PROPERTY(int flowControl READ flowControl WRITE setFlowControl NOTIFY flowControlChanged) Q_PROPERTY(int stopBits READ stopBits WRITE setStopBits NOTIFY stopBitsChanged) Q_PROPERTY(int parity READ parity WRITE setParity NOTIFY parityChanged) Q_PROPERTY(QString portName READ portName WRITE setPortName NOTIFY portNameChanged) Q_PROPERTY(QString portDisplayName READ portDisplayName NOTIFY portDisplayNameChanged) Q_PROPERTY(bool usbDirect READ usbDirect WRITE setUsbDirect NOTIFY usbDirectChanged) ///< true: direct usb connection to board int baud() { return _baud; } int dataBits() { return _dataBits; } int flowControl() { return _flowControl; } ///< QSerialPort Enums int stopBits() { return _stopBits; } int parity() { return _parity; } ///< QSerialPort Enums bool usbDirect() { return _usbDirect; } const QString portName () { return _portName; } const QString portDisplayName () { return _portDisplayName; } void setBaud (int baud); void setDataBits (int databits); void setFlowControl (int flowControl); ///< QSerialPort Enums void setStopBits (int stopBits); void setParity (int parity); ///< QSerialPort Enums void setPortName (const QString& portName); void setUsbDirect (bool usbDirect); static QStringList supportedBaudRates(); static QString cleanPortDisplayname(const QString name); /// From LinkConfiguration LinkType type () { return LinkConfiguration::TypeSerial; } void copyFrom (LinkConfiguration* source); bool isHighLatencyAllowed () { return true; } void loadSettings (QSettings& settings, const QString& root); void saveSettings (QSettings& settings, const QString& root); void updateSettings (); QString settingsURL () { return "SerialSettings.qml"; } QString settingsTitle () { return tr("Serial Link Settings"); } signals: void baudChanged (); void dataBitsChanged (); void flowControlChanged (); void stopBitsChanged (); void parityChanged (); void portNameChanged (); void portDisplayNameChanged (); void usbDirectChanged (bool usbDirect); private: static void _initBaudRates(); int _baud; int _dataBits; int _flowControl; int _stopBits; int _parity; QString _portName; QString _portDisplayName; bool _usbDirect; }; class SerialLink : public LinkInterface { Q_OBJECT friend class SerialConfiguration; friend class LinkManager; public: // LinkInterface overrides QString getName (void) const override; bool isConnected (void) const override; void disconnect (void) override; /// Don't even think of calling this method! QSerialPort* _hackAccessToPort(void) { return _port; } private slots: void _writeBytes(const QByteArray data) override; public slots: void linkError(QSerialPort::SerialPortError error); private slots: void _readBytes (void); private: // Links are only created/destroyed by LinkManager so constructor/destructor is not public SerialLink(SharedLinkConfigurationPtr& config, bool isPX4Flow = false); ~SerialLink(); // LinkInterface overrides bool _connect(void) override; void _emitLinkError (const QString& errorMsg); bool _hardwareConnect (QSerialPort::SerialPortError& error, QString& errorString); bool _isBootloader (void); QSerialPort* _port = nullptr; quint64 _bytesRead = 0; int _timeout; QMutex _dataMutex; ///< Mutex for reading data from _port QMutex _writeMutex; ///< Mutex for accessing the _transmitBuffer. volatile bool _stopp = false; QMutex _stoppMutex; ///< Mutex for accessing _stopp QByteArray _transmitBuffer; ///< An internal buffer for receiving data from member functions and actually transmitting them via the serial port. SerialConfiguration* _serialConfig = nullptr; };