/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * Copyright 2013 - 2024, nymea GmbH * Contact: contact@nymea.io * * This file is part of nymea. * This project including source code and documentation is protected by * copyright law, and remains the property of nymea GmbH. All rights, including * reproduction, publication, editing and translation, are reserved. The use of * this project is subject to the terms of a license agreement to be concluded * with nymea GmbH in accordance with the terms of use of nymea GmbH, available * under https://nymea.io/license * * GNU Lesser General Public License Usage * Alternatively, this project may be redistributed and/or modified under the * terms of the GNU Lesser General Public License as published by the Free * Software Foundation; version 3. This project is distributed in the hope that * it will be useful, but WITHOUT ANY WARRANTY; without even the implied * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this project. If not, see . * * For any further details and any questions please contact us under * contact@nymea.io or see our FAQ/Licensing Information on * https://nymea.io/license/faq * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */ #include "modbusrtumanager.h" #include "nymeasettings.h" #include "loggingcategories.h" #include "modbusrtumasterimpl.h" #include "hardware/serialport/serialportmonitor.h" #include "qglobal.h" #include #include #include #include NYMEA_LOGGING_CATEGORY(dcModbusRtu, "ModbusRtu") namespace nymeaserver { ModbusRtuManager::ModbusRtuManager(SerialPortMonitor *serialPortMonitor, QObject *parent) : QObject(parent), m_serialPortMonitor(serialPortMonitor) { if (!supported()) return; // Load the platform config loadPlatformConfiguration(); // Load uart configurations loadRtuMasters(); // Initialize the list of serial ports already available in the system... foreach (const SerialPort &serialPort, m_serialPortMonitor->serialPorts()) { onSerialPortAdded(serialPort); } connect(m_serialPortMonitor, &SerialPortMonitor::serialPortAdded, this, &ModbusRtuManager::onSerialPortAdded); connect(m_serialPortMonitor, &SerialPortMonitor::serialPortRemoved, this, &ModbusRtuManager::onSerialPortRemoved); // Try to connect the modbus RTU masters foreach (ModbusRtuMaster *modbusMaster, m_modbusRtuMasters.values()) { ModbusRtuMasterImpl *modbusMasterImpl = qobject_cast(modbusMaster); if (!modbusMasterImpl->connectDevice()) { qCWarning(dcModbusRtu()) << "Failed to connect modbus RTU master. Could not connect to" << modbusMaster; } } } SerialPorts ModbusRtuManager::serialPorts() const { return m_serialPorts.values(); } bool ModbusRtuManager::serialPortAvailable(const QString &systemLocation) const { return m_serialPorts.keys().contains(systemLocation); } void ModbusRtuManager::onSerialPortAdded(const SerialPort &serialPort) { if (m_serialPorts.contains(serialPort.systemLocation())) return; // Depending on the platform configuration we have to filter out those serial ports which are not suitable for modbus RTU communication. foreach (const ModbusRtuPlatformConfiguration &platformConfig, m_platformConfigurations) { if (platformConfig.serialPort == serialPort.systemLocation()) { if (platformConfig.usable) { SerialPort internalPort = serialPort; internalPort.setCustomDescription(platformConfig.description); m_serialPorts.insert(internalPort.systemLocation(), internalPort); emit serialPortAdded(internalPort); } else { qCDebug(dcModbusRtu()) << "Serial port" << serialPort.systemLocation() << "added but not usable for modbus RTU according to the platorm configuration."; return; } } } if (!m_serialPorts.contains(serialPort.systemLocation())) { m_serialPorts.insert(serialPort.systemLocation(), serialPort); emit serialPortAdded(serialPort); } qCDebug(dcModbusRtu()) << "Serial port" << serialPort.systemLocation() << "added. Evaluate modbus RTU masters..."; // Check if we have to reconnect any modbus RTU masters foreach (ModbusRtuMaster *modbusMaster, m_modbusRtuMasters.values()) { ModbusRtuMasterImpl *modbusMasterImpl = qobject_cast(modbusMaster); // Try only to reconnect if the added serial port matches a disconnected modbus RTU master if (!modbusMasterImpl->connected() && modbusMasterImpl->serialPort() == serialPort.systemLocation()) { if (!modbusMasterImpl->connectDevice()) { qCDebug(dcModbusRtu()) << "Reconnect" << modbusMaster << "failed."; } else { qCDebug(dcModbusRtu()) << "Reconnected" << modbusMaster << "successfully."; } } } } void ModbusRtuManager::onSerialPortRemoved(const SerialPort &serialPort) { if (m_serialPorts.contains(serialPort.systemLocation())) { qCDebug(dcModbusRtu()) << "Serial port removed" << serialPort.systemLocation(); m_serialPorts.remove(serialPort.systemLocation()); emit serialPortRemoved(serialPort); } } bool ModbusRtuManager::supported() const { #ifdef WITH_QTSERIALBUS return true; #else return false; #endif } QList ModbusRtuManager::modbusRtuMasters() const { return m_modbusRtuMasters.values(); } bool ModbusRtuManager::hasModbusRtuMaster(const QUuid &modbusUuid) const { return m_modbusRtuMasters.value(modbusUuid) != nullptr; } ModbusRtuMaster *ModbusRtuManager::getModbusRtuMaster(const QUuid &modbusUuid) { if (hasModbusRtuMaster(modbusUuid)) { return m_modbusRtuMasters.value(modbusUuid); } return nullptr; } QPair ModbusRtuManager::addNewModbusRtuMaster(const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits, int numberOfRetries, int timeout) { if (!supported()) { qCWarning(dcModbusRtu()) << "Cannot add new modbus RTU master because serialbus is not suppoerted on this platform."; return QPair(ModbusRtuErrorNotSupported, QUuid()); } // Check if the serial port exists if (!m_serialPortMonitor->serialPortAvailable(serialPort)) { qCWarning(dcModbusRtu()) << "Cannot add new modbus RTU master because the serial port" << serialPort << "is not available any more"; return QPair(ModbusRtuErrorHardwareNotFound, QUuid()); } QUuid modbusUuid = QUuid::createUuid(); ModbusRtuMasterImpl *modbusMaster = new ModbusRtuMasterImpl(modbusUuid, serialPort, baudrate, parity, dataBits, stopBits, numberOfRetries, timeout, this); ModbusRtuMaster *modbus = qobject_cast(modbusMaster); qCDebug(dcModbusRtu()) << "Adding new" << modbus << parity << dataBits << stopBits; // Note: We could add the modbus master event if a connection is currently not possible...not sure yet if (!modbusMaster->connectDevice()) { qCWarning(dcModbusRtu()) << "Failed to add new modbus RTU master. Could not connect to" << modbus << parity << dataBits << stopBits; modbusMaster->deleteLater(); return QPair(ModbusRtuErrorConnectionFailed, QUuid()); } addModbusRtuMasterInternally(modbusMaster); saveModbusRtuMaster(modbus); return QPair(ModbusRtuErrorNoError, modbusUuid); } ModbusRtuManager::ModbusRtuError ModbusRtuManager::reconfigureModbusRtuMaster(const QUuid &modbusUuid, const QString &serialPort, qint32 baudrate, QSerialPort::Parity parity, QSerialPort::DataBits dataBits, QSerialPort::StopBits stopBits, int numberOfRetries, int timeout) { if (!supported()) { qCWarning(dcModbusRtu()) << "Cannot reconfigure modbus RTU master because serialbus is not suppoerted on this platform."; return ModbusRtuErrorNotSupported; } if (!m_modbusRtuMasters.contains(modbusUuid)) { qCWarning(dcModbusRtu()) << "Could not reconfigure modbus RTU master because no resource could be found with uuid" << modbusUuid.toString(); return ModbusRtuErrorUuidNotFound; } // Take the modbus masters ModbusRtuMasterImpl *modbusMaster = qobject_cast(m_modbusRtuMasters.value(modbusUuid)); // Disconnect modbusMaster->disconnectDevice(); // Reconfigure modbusMaster->setSerialPort(serialPort); modbusMaster->setBaudrate(baudrate); modbusMaster->setParity(parity); modbusMaster->setDataBits(dataBits); modbusMaster->setStopBits(stopBits); modbusMaster->setNumberOfRetries(numberOfRetries); modbusMaster->setTimeout(timeout); // Connect again if (!modbusMaster->connectDevice()) { qCWarning(dcModbusRtu()) << "Failed to connect to" << m_modbusRtuMasters.value(modbusUuid); // FIXME: check if we should reload the old configuration emit modbusRtuMasterChanged(m_modbusRtuMasters.value(modbusUuid)); return ModbusRtuErrorConnectionFailed; } emit modbusRtuMasterChanged(m_modbusRtuMasters.value(modbusUuid)); qCDebug(dcModbusRtu()) << "Reconfigured successfully" << m_modbusRtuMasters.value(modbusUuid); saveModbusRtuMaster(modbusMaster); return ModbusRtuErrorNoError; } ModbusRtuManager::ModbusRtuError ModbusRtuManager::removeModbusRtuMaster(const QUuid &modbusUuid) { if (!supported()) { qCWarning(dcModbusRtu()) << "Cannot remove modbus RTU master because serialbus is not suppoerted on this platform."; return ModbusRtuErrorNotSupported; } if (!m_modbusRtuMasters.contains(modbusUuid)) { qCWarning(dcModbusRtu()) << "Could not remove modbus RTU master because no resource could be found with uuid" << modbusUuid.toString(); return ModbusRtuErrorUuidNotFound; } ModbusRtuMasterImpl *modbusMaster = qobject_cast(m_modbusRtuMasters.take(modbusUuid)); qCDebug(dcModbusRtu()) << "Removing modbus RTU master" << qobject_cast(modbusMaster); modbusMaster->disconnectDevice(); modbusMaster->deleteLater(); // Remove from settings NymeaSettings settings(NymeaSettings::SettingsRoleModbusRtu); settings.beginGroup("ModbusRtuMasters"); settings.beginGroup(modbusUuid.toString()); settings.remove(""); settings.endGroup(); // modbusUuid settings.endGroup(); // ModbusRtuMasters emit modbusRtuMasterRemoved(modbusMaster); return ModbusRtuErrorNoError; } void ModbusRtuManager::loadPlatformConfiguration() { QFileInfo platformConfigurationFileInfo(NymeaSettings::settingsPath() + QDir::separator() + "modbus-rtu-platform.conf"); if (platformConfigurationFileInfo.exists()) { qCDebug(dcModbusRtu()) << "Loading modbus RTU platform configuration from" << platformConfigurationFileInfo.absoluteFilePath(); QFile platformConfigurationFile(platformConfigurationFileInfo.absoluteFilePath()); if (!platformConfigurationFile.open(QIODevice::ReadOnly | QIODevice::Text)) { qCWarning(dcModbusRtu()) << "Failed to open modbus RTU platform configuration file" << platformConfigurationFileInfo.absoluteFilePath() << ":" << platformConfigurationFile.errorString(); return; } QByteArray data = platformConfigurationFile.readAll(); platformConfigurationFile.close(); QJsonParseError jsonError; QJsonDocument jsonDoc = QJsonDocument::fromJson(data, &jsonError); if (jsonError.error != QJsonParseError::NoError) { qCWarning(dcModbusRtu()) << "Failed to parse JSON data from modbus RTU platform configuration file" << platformConfigurationFileInfo.absoluteFilePath() << ":" << jsonError.errorString(); return; } // Make sure we have a clean start m_platformConfigurations.clear(); QVariantMap platformConfigMap = jsonDoc.toVariant().toMap(); foreach (const QVariant &configVariant, platformConfigMap.value("interfaces").toList()) { QVariantMap configMap = configVariant.toMap(); ModbusRtuPlatformConfiguration config; config.name = configMap.value("name").toString(); config.description = configMap.value("description").toString(); config.serialPort = configMap.value("serialPort").toString(); config.usable = configMap.value("usable").toBool(); m_platformConfigurations.append(config); } if (m_platformConfigurations.isEmpty()) { qCDebug(dcModbusRtu()) << "No platform configurations available"; } else { qCDebug(dcModbusRtu()) << "Loaded successfully" << m_platformConfigurations.count() << "platform configurations"; foreach(const ModbusRtuPlatformConfiguration &config, m_platformConfigurations) { qCDebug(dcModbusRtu()).nospace() << "- Platform configuration: " << config.description << " (" << config.name << ") " << "Serial port: " << config.serialPort << " usable: " << config.usable; } } } } void ModbusRtuManager::loadRtuMasters() { if (!supported()) return; NymeaSettings settings(NymeaSettings::SettingsRoleModbusRtu); qCDebug(dcModbusRtu()) << "Loading modbus RTU resources from" << settings.fileName(); settings.beginGroup("ModbusRtuMasters"); foreach (const QString &uuidString, settings.childGroups()) { settings.beginGroup(uuidString); QString serialPort = settings.value("serialPort").toString(); quint32 baudrate = settings.value("baudrate").toUInt(); QSerialPort::Parity parity = static_cast(settings.value("parity").toInt()); QSerialPort::DataBits dataBits = static_cast(settings.value("dataBits").toInt()); QSerialPort::StopBits stopBits = static_cast(settings.value("stopBits").toInt()); int numberOfRetries = settings.value("numberOfRetries").toInt(); int timeout = settings.value("timeout").toInt(); settings.endGroup(); // uuid addModbusRtuMasterInternally(new ModbusRtuMasterImpl(QUuid(uuidString), serialPort, baudrate, parity, dataBits, stopBits, numberOfRetries, timeout, this)); } settings.endGroup(); // ModbusRtuMasters } void ModbusRtuManager::saveModbusRtuMaster(ModbusRtuMaster *modbusRtuMaster) { NymeaSettings settings(NymeaSettings::SettingsRoleModbusRtu); qCDebug(dcModbusRtu()) << "Saving" << modbusRtuMaster << "to" << settings.fileName(); settings.beginGroup("ModbusRtuMasters"); settings.beginGroup(modbusRtuMaster->modbusUuid().toString()); settings.setValue("serialPort", modbusRtuMaster->serialPort()); settings.setValue("baudrate", modbusRtuMaster->baudrate()); settings.setValue("parity", static_cast(modbusRtuMaster->parity())); settings.setValue("dataBits", static_cast(modbusRtuMaster->dataBits())); settings.setValue("stopBits", static_cast(modbusRtuMaster->stopBits())); settings.setValue("numberOfRetries", modbusRtuMaster->numberOfRetries()); settings.setValue("timeout", modbusRtuMaster->timeout()); settings.endGroup(); // uuid settings.endGroup(); // ModbusRtuMasters } void ModbusRtuManager::addModbusRtuMasterInternally(ModbusRtuMasterImpl *modbusRtuMaster) { ModbusRtuMaster *modbusMaster = qobject_cast(modbusRtuMaster); qCDebug(dcModbusRtu()) << "Adding" << modbusMaster; m_modbusRtuMasters.insert(modbusMaster->modbusUuid(), modbusMaster); connect(modbusMaster, &ModbusRtuMaster::connectedChanged, this, [=](bool connected){ qCDebug(dcModbusRtu()) << modbusMaster << (connected ? "connected" : "disconnected"); emit modbusRtuMasterChanged(modbusMaster); }); emit modbusRtuMasterAdded(modbusMaster); } }