Implement add/remove/reconfigure for modbus rtu manager

This commit is contained in:
Simon Stürz 2021-02-08 17:55:33 +01:00
parent ab7579de42
commit af7948dc1e
5 changed files with 191 additions and 6 deletions

View File

@ -40,13 +40,35 @@ namespace nymeaserver {
ModbusRtuManager::ModbusRtuManager(QObject *parent) : QObject(parent)
{
// Load available serial ports
updateSerialPorts();
// Load uart configurations
loadRtuMasters();
// Connect signals
// Enable autoconnect for each modbus rtu master
m_timer = new QTimer(this);
m_timer->setInterval(5000);
m_timer->setSingleShot(false);
connect(m_timer, &QTimer::timeout, this, [=](){
// Update serial port list
updateSerialPorts();
// Check if we have to reconnect a device
foreach (ModbusRtuMaster *modbusMaster, m_modbusRtuMasters.values()) {
ModbusRtuMasterImpl *modbusMasterImpl = qobject_cast<ModbusRtuMasterImpl *>(modbusMaster);
if (!modbusMasterImpl->connected()) {
if (!modbusMasterImpl->connectDevice()) {
qCDebug(dcModbusRtu()) << "Reconnect" << modbusMaster << "failed.";
} else {
qCDebug(dcModbusRtu()) << "Reconnected" << modbusMaster << "sucessfully.";
}
}
}
});
m_timer->start();
}
QList<ModbusRtuMaster *> ModbusRtuManager::modbusRtuMasters() const
@ -68,6 +90,79 @@ ModbusRtuMaster *ModbusRtuManager::getModbusRtuMaster(const QUuid &modbusUuid)
return nullptr;
}
QPair<ModbusRtuManager::Error, QUuid> ModbusRtuManager::addNewModbusRtuMaster(const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits)
{
// Check if the serial port exists
// Check if the serial port is not occupied
QUuid modbusUuid = QUuid::createUuid();
ModbusRtuMasterImpl *modbusMaster = new ModbusRtuMasterImpl(modbusUuid, serialPort, baudrate, parity, dataBits, stopBits, this);
ModbusRtuMaster *modbus = qobject_cast<ModbusRtuMaster *>(modbusMaster);
qCDebug(dcModbusRtu()) << "Adding new" << qobject_cast<ModbusRtuMaster *>(modbusMaster) << parity << dataBits << stopBits;
// Connect the modbus master bus;
m_modbusRtuMasters.insert(modbusUuid, modbus);
emit modbusRtuMasterAdded(modbus);
saveModbusRtuMaster(modbus);
return QPair<Error, QUuid>(ErrorNoError, modbusUuid);
}
ModbusRtuManager::Error ModbusRtuManager::reconfigureRtuMaster(const QUuid &modbusUuid, const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits)
{
if (!m_modbusRtuMasters.contains(modbusUuid)) {
qCWarning(dcModbusRtu()) << "Could not reconfigure modbus RTU master because no resource could be found with uuid" << modbusUuid.toString();
return ErrorNotFound;
}
ModbusRtuMasterImpl *modbusMaster = qobject_cast<ModbusRtuMasterImpl *>(m_modbusRtuMasters.value(modbusUuid));
// Disconnect
modbusMaster->disconnectDevice();
// Reconfigure
modbusMaster->setSerialPort(serialPort);
modbusMaster->setBaudrate(baudrate);
modbusMaster->setParity(parity);
modbusMaster->setDataBits(dataBits);
modbusMaster->setStopBits(stopBits);
// Connect again
if (!modbusMaster->connectDevice()) {
qCWarning(dcModbusRtu()) << "Failed to connect to" << m_modbusRtuMasters.value(modbusUuid);
return ErrorConnectionFailed;
}
emit modbusRtuMasterChanged(m_modbusRtuMasters.value(modbusUuid));
qCDebug(dcModbusRtu()) << "Reconfigured successfully" << m_modbusRtuMasters.value(modbusUuid);
return ErrorNoError;
}
ModbusRtuManager::Error ModbusRtuManager::removeModbusRtuMaster(const QUuid &modbusUuid)
{
ModbusRtuMasterImpl *modbusMaster = qobject_cast<ModbusRtuMasterImpl *>(m_modbusRtuMasters.value(modbusUuid));
if (!modbusMaster) {
qCWarning(dcModbusRtu()) << "Could not remove modbus RTU master because no resource could be found with uuid" << modbusUuid.toString();
return ErrorNotFound;
}
qCDebug(dcModbusRtu()) << "Removing modbus RTU master" << qobject_cast<ModbusRtuMaster *>(modbusMaster);
emit modbusRtuMasterRemoved(modbusMaster);
modbusMaster->deleteLater();
return ErrorNoError;
}
void ModbusRtuManager::updateSerialPorts()
{
// Check if serial port added or removed
}
void ModbusRtuManager::loadRtuMasters()
{
NymeaSettings settings(NymeaSettings::SettingsRoleModbusRtu);
@ -87,7 +182,7 @@ void ModbusRtuManager::loadRtuMasters()
ModbusRtuMaster *modbusRtuMaster = qobject_cast<ModbusRtuMaster *>(modbus);
qCDebug(dcModbusRtu()) << "Loaded" << modbusRtuMaster;
m_modbusRtuMasters.insert(modbusRtuMaster->modbusUuid(), modbusRtuMaster);
emit modbusRtuMasterAdded(modbusRtuMaster->modbusUuid());
emit modbusRtuMasterAdded(modbusRtuMaster);
}
settings.endGroup(); // ModbusRtuMasters

View File

@ -34,6 +34,7 @@
#include <QHash>
#include <QUuid>
#include <QObject>
#include <QTimer>
#include "hardware/modbus/modbusrtumaster.h"
@ -43,6 +44,13 @@ class ModbusRtuManager : public QObject
{
Q_OBJECT
public:
enum Error {
ErrorNoError,
ErrorNotFound,
ErrorConnectionFailed
};
Q_ENUM(Error)
explicit ModbusRtuManager(QObject *parent = nullptr);
~ModbusRtuManager() = default;
@ -50,10 +58,14 @@ public:
bool hasModbusRtuMaster(const QUuid &modbusUuid) const;
ModbusRtuMaster *getModbusRtuMaster(const QUuid &modbusUuid);
QPair<Error, QUuid> addNewModbusRtuMaster(const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits);
Error reconfigureRtuMaster(const QUuid &modbusUuid, const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits);
Error removeModbusRtuMaster(const QUuid &modbusUuid);
signals:
void modbusRtuMasterAdded(const QUuid &modbusUuid);
void modbusRtuMasterRemoved(const QUuid &modbusUuid);
void modbusRtuMasterChanged(const QUuid &modbusUuid);
void modbusRtuMasterAdded(ModbusRtuMaster *modbusRtuMaster);
void modbusRtuMasterRemoved(ModbusRtuMaster *modbusRtuMaster);
void modbusRtuMasterChanged(ModbusRtuMaster *modbusRtuMaster);
private:
QHash<QUuid, ModbusRtuMaster *> m_modbusRtuMasters;

View File

@ -85,31 +85,91 @@ QString ModbusRtuMasterImpl::serialPort() const
return m_serialPort;
}
void ModbusRtuMasterImpl::setSerialPort(const QString &serialPort)
{
if (m_serialPort == serialPort)
return;
m_serialPort = serialPort;
emit serialPortChanged(m_serialPort);
}
qint32 ModbusRtuMasterImpl::baudrate() const
{
return m_baudrate;
}
void ModbusRtuMasterImpl::setBaudrate(qint32 baudrate)
{
if (m_baudrate == baudrate)
return;
m_baudrate = baudrate;
emit baudrateChanged(m_baudrate);
}
QSerialPort::Parity ModbusRtuMasterImpl::parity() const
{
return m_parity;
}
void ModbusRtuMasterImpl::setParity(QSerialPort::Parity parity)
{
if (m_parity == parity)
return;
m_parity = parity;
emit parityChanged(m_parity);
}
QSerialPort::DataBits ModbusRtuMasterImpl::dataBits() const
{
return m_dataBits;
}
void ModbusRtuMasterImpl::setDataBits(QSerialPort::DataBits dataBits)
{
if (m_dataBits == dataBits)
return;
m_dataBits = dataBits;
emit dataBitsChanged(m_dataBits);
}
QSerialPort::StopBits ModbusRtuMasterImpl::stopBits()
{
return m_stopBits;
}
void ModbusRtuMasterImpl::setStopBits(QSerialPort::StopBits stopBits)
{
if (m_stopBits == stopBits)
return;
m_stopBits = stopBits;
emit stopBitsChanged(m_stopBits);
}
bool ModbusRtuMasterImpl::connected() const
{
return m_connected;
}
bool ModbusRtuMasterImpl::connectDevice()
{
m_modbus->setConnectionParameter(QModbusDevice::SerialPortNameParameter, m_serialPort);
m_modbus->setConnectionParameter(QModbusDevice::SerialBaudRateParameter, m_baudrate);
m_modbus->setConnectionParameter(QModbusDevice::SerialDataBitsParameter, m_dataBits);
m_modbus->setConnectionParameter(QModbusDevice::SerialStopBitsParameter, m_stopBits);
m_modbus->setConnectionParameter(QModbusDevice::SerialParityParameter, m_parity);
return m_modbus->connectDevice();
}
void ModbusRtuMasterImpl::disconnectDevice()
{
m_modbus->disconnectDevice();
}
ModbusRtuReply *ModbusRtuMasterImpl::readCoil(int slaveAddress, int registerAddress, quint16 size)
{
#ifdef WITH_QTSERIALBUS

View File

@ -50,14 +50,27 @@ public:
~ModbusRtuMasterImpl() override = default;
QUuid modbusUuid() const override;
QString serialPort() const override;
void setSerialPort(const QString &serialPort);
qint32 baudrate() const override;
void setBaudrate(qint32 baudrate);
QSerialPort::Parity parity() const override;
void setParity(QSerialPort::Parity parity);
QSerialPort::DataBits dataBits() const override;
void setDataBits(QSerialPort::DataBits dataBits);
QSerialPort::StopBits stopBits() override;
void setStopBits(QSerialPort::StopBits stopBits);
bool connected() const override;
bool connectDevice();
void disconnectDevice();
// Requests
ModbusRtuReply *readCoil(int slaveAddress, int registerAddress, quint16 size = 1) override;
ModbusRtuReply *readDiscreteInput(int slaveAddress, int registerAddress, quint16 size = 1) override;

View File

@ -67,6 +67,11 @@ protected:
signals:
void connectedChanged(bool connected);
void serialPortChanged(const QString &serialPort);
void baudrateChanged(quint32 baudrate);
void parityChanged(QSerialPort::Parity parity);
void dataBitsChanged(QSerialPort::DataBits dataBits);
void stopBitsChanged(QSerialPort::StopBits stopBits);
};