nymea-plugins-modbus/nymea-modbus-cli/main.cpp

495 lines
22 KiB
C++

// SPDX-License-Identifier: GPL-3.0-or-later
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*
* Copyright (C) 2013 - 2024, nymea GmbH
* Copyright (C) 2024 - 2025, chargebyte austria GmbH
*
* This file is part of nymea-plugins-modbus.
*
* nymea-plugins-modbus is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* nymea-plugins-modbus 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with nymea-plugins-modbus. If not, see <https://www.gnu.org/licenses/>.
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
#include <QCoreApplication>
#include <QCommandLineParser>
#include <QCommandLineOption>
#include <QDebug>
#include <QObject>
#include <QVariant>
#include <QSerialPort>
#include <QHostAddress>
#include <QSerialPortInfo>
#include <QModbusTcpClient>
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
#include <QModbusRtuSerialClient>
#else
#include <QModbusRtuSerialMaster>
#endif
void sendRequest(quint16 modbusServerAddress, QModbusDataUnit::RegisterType registerType, quint16 registerAddress, quint16 length, const QByteArray &writeData, QModbusClient *client);
QString exceptionCodeToString(QModbusPdu::ExceptionCode exception);
int main(int argc, char *argv[])
{
QCoreApplication application(argc, argv);
application.setApplicationName("nymea-modbus-cli");
application.setOrganizationName("nymea");
application.setApplicationVersion("1.3.0");
QString description = QString("\nTool for testing and reading Modbus TCP or RTU registers.\n\n");
description.append(QString("Copyright %1 2016 - 2025 nymea GmbH <contact@nymea.io>\n\n").arg(QChar(0xA9)));
description.append("TCP\n");
description.append("-----------------------------------------\n");
description.append("Example reading 2 holding registers from address 1000:\n");
description.append("nymea-modbus-cli -a 192.168.0.10 -p 502 -r 1000 -l 2\n\n");
description.append("RTU\n");
description.append("-----------------------------------------\n\n");
description.append("Typical baudrates:\n");
description.append("- 1200\n");
description.append("- 2400\n");
description.append("- 4800\n");
description.append("- 9600\n");
description.append("- 19200\n");
description.append("- 38400\n");
description.append("- 57600\n");
description.append("- 115200\n\n");
description.append("Example reading 2 holding registers from address 1000:\n");
description.append("nymea-modbus-cli --serial /dev/ttyUSB0 --baudrate 9600 -r 1000 -l 2\n\n");
QCommandLineParser parser;
parser.addHelpOption();
parser.addVersionOption();
parser.setApplicationDescription(description);
// TCP
QCommandLineOption addressOption(QStringList() << "a" << "address", QString("TCP: The IP address of the modbus TCP server."), "address");
parser.addOption(addressOption);
QCommandLineOption portOption(QStringList() << "p" << "port", QString("TCP: The port of the modbus TCP server. Default is 502."), "port");
portOption.setDefaultValue("502");
parser.addOption(portOption);
// RTU
QCommandLineOption serialPortOption(QStringList() << "serial", QString("RTU: The serial port to use for the RTU communication."), "port");
parser.addOption(serialPortOption);
QCommandLineOption baudrateOption(QStringList() << "baudrate", QString("RTU: The baudrate for the RTU communication. Default is 19200."), "baudrate");
baudrateOption.setDefaultValue("19200");
parser.addOption(baudrateOption);
QCommandLineOption parityOption(QStringList() << "parity", QString("RTU: The parity for the RTU communication. Allowed values are [none, even, odd, space, mark]. Default is none."), "parity");
parityOption.setDefaultValue("none");
parser.addOption(parityOption);
QCommandLineOption dataBitsOption(QStringList() << "databits", QString("RTU: The amount of data bits for the RTU communication. Allowed values are [5, 6, 7, 8]. Default is 8."), "databits");
dataBitsOption.setDefaultValue("8");
parser.addOption(dataBitsOption);
QCommandLineOption stopBitsOption(QStringList() << "stopbits", QString("RTU: The amount of stop bits for the RTU communication. Allowed values are [1, 1.5, 2]. Default is 1."), "stopbits");
stopBitsOption.setDefaultValue("1");
parser.addOption(stopBitsOption);
QCommandLineOption listSerialPortsOption(QStringList() << "list-serials", QString("List the available serial ports on this host."));
parser.addOption(listSerialPortsOption);
// General
QCommandLineOption modbusServerAddressOption(QStringList() << "m" << "modbus-address", QString("The modbus server address on the bus (slave ID). Default is 1."), "id");
modbusServerAddressOption.setDefaultValue("1");
parser.addOption(modbusServerAddressOption);
QCommandLineOption registerTypeOption(QStringList() << "t" << "type", QString("The type of the modbus register. Default is holding."), "input, holding, discrete, coils");
registerTypeOption.setDefaultValue("holding");
parser.addOption(registerTypeOption);
QCommandLineOption registerOption(QStringList() << "r" << "register", QString("The number of the modbus register."), "register");
parser.addOption(registerOption);
QCommandLineOption lengthOption(QStringList() << "l" << "length", QString("The number of registers to read. Default is 1."), "length");
lengthOption.setDefaultValue("1");
parser.addOption(lengthOption);
QCommandLineOption writeOption(QStringList() << "w" << "write", QString("The data to be written to the given register."), "data");
parser.addOption(writeOption);
QCommandLineOption debugOption(QStringList() << "d" << "debug", QString("Print more information."));
parser.addOption(debugOption);
QCommandLineOption broadcastOption(QStringList() << "broadcast", QString("Send the request to modbus slave ID 0 aka broadcast"));
parser.addOption(broadcastOption);
parser.process(application);
bool verbose = parser.isSet(debugOption);
if (verbose) qDebug() << "Verbose debug print enabled";
if (parser.isSet(listSerialPortsOption)) {
foreach (const QSerialPortInfo &serialPortInfo, QSerialPortInfo::availablePorts())
qInfo().noquote() << serialPortInfo.systemLocation() << "|" << serialPortInfo.description() << "|" << serialPortInfo.serialNumber() << "|" << serialPortInfo.manufacturer();
exit(EXIT_SUCCESS);
}
// Make sure we have either RTU, or TCP, not both or none
if (parser.isSet(addressOption) && parser.isSet(serialPortOption)) {
qCritical() << "Error: invalid paramter combination. Use either TCP connection by defining the \"address\" or RTU by defining the \"serial\" paramter, not both.";
exit(EXIT_FAILURE);
}
if (!parser.isSet(addressOption) && !parser.isSet(serialPortOption)) {
qCritical() << "Error: unknown protocol. Use either TCP connection by specifying the \"address\" or RTU by specifying the \"serial\" port.";
exit(EXIT_FAILURE);
}
QModbusDataUnit::RegisterType registerType = QModbusDataUnit::RegisterType::Invalid;
QString registerTypeString = parser.value(registerTypeOption);
if (registerTypeString.toLower() == "input") {
registerType = QModbusDataUnit::RegisterType::InputRegisters;
} else if (registerTypeString.toLower() == "holding") {
registerType = QModbusDataUnit::RegisterType::HoldingRegisters;
} else if (registerTypeString.toLower() == "discrete") {
registerType = QModbusDataUnit::RegisterType::DiscreteInputs;
} else if (registerTypeString.toLower() == "coils") {
registerType = QModbusDataUnit::RegisterType::Coils;
} else {
qCritical() << "Error: invalid register type:" << parser.value(registerTypeOption) << "Please select on of the valid register types: input, holding, discrete, coils";
exit(EXIT_FAILURE);
}
if (parser.isSet(broadcastOption) && parser.isSet(modbusServerAddressOption)) {
qCritical() << "Error: broadcast and modbus address specified. Please use one or the other option.";
exit(EXIT_FAILURE);
}
bool valueOk = false;
quint16 modbusServerAddress = parser.value(modbusServerAddressOption).toUInt(&valueOk);
if (parser.isSet(broadcastOption)) {
modbusServerAddress = 0;
} else {
modbusServerAddress = parser.value(modbusServerAddressOption).toUInt(&valueOk);
if (!valueOk) {
qCritical() << "Error: invalid modbus server address (slave ID):" << parser.value(modbusServerAddressOption);
exit(EXIT_FAILURE);
} else if (modbusServerAddress == 0) {
qCritical() << "Error: invalid modbus server address (slave ID):" << parser.value(modbusServerAddressOption);
qCritical() << "Please use the broadcast parameter for sending broadcast requests.";
exit(EXIT_FAILURE);
}
}
quint16 registerAddress = parser.value(registerOption).toUInt(&valueOk);
if (!valueOk) {
qCritical() << "Error: invalid register number:" << parser.value(registerOption);
exit(EXIT_FAILURE);
}
quint16 length = parser.value(lengthOption).toUInt(&valueOk);
if (!valueOk) {
qCritical() << "Error: invalid register length number:" << parser.value(lengthOption);
exit(EXIT_FAILURE);
}
QByteArray writeData;
if (parser.isSet(writeOption)) {
writeData = parser.value(writeOption).toLocal8Bit();
qDebug() << "Write data:" << writeData;
}
// TCP
if (parser.isSet(addressOption)) {
// TCP connection
QHostAddress address = QHostAddress(parser.value(addressOption));
if (address.isNull()) {
qCritical() << "Error: invalid address:" << parser.value(addressOption);
exit(EXIT_FAILURE);
}
quint16 port = parser.value(portOption).toUInt();
qInfo().noquote() << "Connecting to" << QString("%1:%2").arg(address.toString()).arg(port) << "modbus server address:" << modbusServerAddress;
QModbusTcpClient *client = new QModbusTcpClient(nullptr);
client->setConnectionParameter(QModbusDevice::NetworkAddressParameter, address.toString());
client->setConnectionParameter(QModbusDevice::NetworkPortParameter, port);
client->setTimeout(3000);
client->setNumberOfRetries(3);
QObject::connect(client, &QModbusTcpClient::stateChanged, &application, [=](QModbusDevice::State state){
if (verbose) qDebug() << "Connection state changed" << state;
if (state != QModbusDevice::ConnectedState)
return;
qDebug() << "Connected successfully to" << QString("%1:%2").arg(address.toString()).arg(port);
sendRequest(modbusServerAddress, registerType, registerAddress, length, writeData, client);
});
QObject::connect(client, &QModbusTcpClient::errorOccurred, &application, [=](QModbusDevice::Error error){
qWarning() << "Modbus error occurred:" << error << client->errorString();
exit(EXIT_FAILURE);
});
if (!client->connectDevice()) {
qWarning() << "Error: could not connect to" << QString("%1:%2").arg(address.toString()).arg(port);
exit(EXIT_FAILURE);
}
}
if (parser.isSet(serialPortOption)) {
QString serialPortName = parser.value(serialPortOption);
quint32 baudrate = parser.value(baudrateOption).toUInt();
QSerialPort::Parity parity = QSerialPort::NoParity;
QString parityString = parser.value(parityOption);
if (parityString.toLower() == "none") {
parity = QSerialPort::NoParity;
} else if (parityString.toLower() == "even") {
parity = QSerialPort::EvenParity;
} else if (parityString.toLower() == "odd") {
parity = QSerialPort::OddParity;
} else if (parityString.toLower() == "space") {
parity = QSerialPort::SpaceParity;
} else if (parityString.toLower() == "mark") {
parity = QSerialPort::MarkParity;
} else {
qCritical() << "Error: invalid parit type:" << parser.value(parityOption) << "Please select on of the valid values: [none, even, odd, space, mark].";
exit(EXIT_FAILURE);
}
QSerialPort::StopBits stopBits = QSerialPort::OneStop;
QString stopBitsString = parser.value(stopBitsOption);
if (stopBitsString == "1") {
stopBits = QSerialPort::OneStop;
} else if (stopBitsString == "1.5") {
stopBits = QSerialPort::OneAndHalfStop;
} else if (stopBitsString == "2") {
stopBits = QSerialPort::TwoStop;
} else {
qCritical() << "Error: invalid stop bits:" << parser.value(stopBitsOption) << "Please select on of the valid values: [1, 1.5, 2].";
exit(EXIT_FAILURE);
}
QSerialPort::DataBits dataBits = QSerialPort::Data8;
QString dataBitsString = parser.value(dataBitsOption);
if (dataBitsString == "5") {
dataBits = QSerialPort::Data5;
} else if (dataBitsString == "6") {
dataBits = QSerialPort::Data6;
} else if (dataBitsString == "7") {
dataBits = QSerialPort::Data7;
} else if (dataBitsString == "8") {
dataBits = QSerialPort::Data8;
} else {
qCritical() << "Error: invalid data bits:" << parser.value(dataBitsOption) << "Please select on of the valid values: [5, 6, 7, 8].";
exit(EXIT_FAILURE);
}
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
QModbusRtuSerialClient *client = new QModbusRtuSerialClient(nullptr);
#else
QModbusRtuSerialMaster *client = new QModbusRtuSerialMaster(nullptr);
#endif
client->setConnectionParameter(QModbusDevice::SerialPortNameParameter, serialPortName);
client->setConnectionParameter(QModbusDevice::SerialBaudRateParameter, baudrate);
client->setConnectionParameter(QModbusDevice::SerialDataBitsParameter, dataBits);
client->setConnectionParameter(QModbusDevice::SerialStopBitsParameter, stopBits);
client->setConnectionParameter(QModbusDevice::SerialParityParameter, parity);
client->setNumberOfRetries(3);
client->setTimeout(500);
QObject::connect(client, &QModbusTcpClient::stateChanged, &application, [=](QModbusDevice::State state){
qDebug() << "Connection state changed" << state;
if (state != QModbusDevice::ConnectedState)
return;
qDebug() << "Connected successfully to" << serialPortName << baudrate << dataBits << stopBits << parity << "modbus server address:" << modbusServerAddress;
sendRequest(modbusServerAddress, registerType, registerAddress, length, writeData, client);
});
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
QObject::connect(client, &QModbusRtuSerialClient::errorOccurred, &application, [=](QModbusDevice::Error error){
#else
QObject::connect(client, &QModbusRtuSerialMaster::errorOccurred, &application, [=](QModbusDevice::Error error){
#endif
if (error != QModbusDevice::NoError) {
exit(EXIT_FAILURE);
}
});
if (!client->connectDevice()) {
qWarning() << "Error: failed not connect to" << serialPortName << client->errorString();
exit(EXIT_FAILURE);
}
}
return application.exec();
}
void sendRequest(quint16 modbusServerAddress, QModbusDataUnit::RegisterType registerType, quint16 registerAddress, quint16 length, const QByteArray &writeData, QModbusClient *client)
{
if (writeData.isEmpty()) {
qDebug() << "Reading from modbus server address" << modbusServerAddress << registerType << "register:" << registerAddress << "Length:" << length;
QModbusDataUnit request = QModbusDataUnit(registerType, registerAddress, length);
QModbusReply *reply = client->sendReadRequest(request, modbusServerAddress);
if (!reply) {
qCritical() << "Failed to read register" << client->errorString();
exit(EXIT_FAILURE);
}
if (reply->isFinished()) {
reply->deleteLater(); // broadcast replies return immediately
qCritical() << "Reply finished immediatly. Something might have gone wrong:" << reply->errorString();
exit(EXIT_FAILURE);
}
QObject::connect(reply, &QModbusReply::finished, reply, &QModbusReply::deleteLater);
QObject::connect(reply, &QModbusReply::finished, client, [=]() {
if (reply->error() != QModbusDevice::NoError) {
QModbusResponse response = reply->rawResult();
if (reply->error() == QModbusDevice::ProtocolError && response.isException()) {
qCritical() << "Modbus reply finished with error" << reply->error() << reply->errorString() << exceptionCodeToString(response.exceptionCode());
} else {
qCritical() << "Modbus reply finished with error" << reply->error() << reply->errorString();
}
exit(EXIT_FAILURE);
}
const QModbusDataUnit unit = reply->result();
// Note: we need the cast in since the valueCount() type changes with different Qt versions
for (int i = 0; i < static_cast<int>(unit.valueCount()); i++) {
quint16 registerValue = unit.values().at(i);
quint16 registerNumber = unit.startAddress() + i;
qInfo() << "-->" << registerNumber << ":" << QString("0x%1").arg(registerValue, 4, 16, QLatin1Char('0')) << registerValue;
}
exit(EXIT_SUCCESS);
});
QObject::connect(reply, &QModbusReply::errorOccurred, client, [=] (QModbusDevice::Error error){
QModbusResponse response = reply->rawResult();
if (reply->error() == QModbusDevice::ProtocolError && response.isException()) {
qCritical() << "Modbus reply error occurred" << error << reply->errorString() << exceptionCodeToString(response.exceptionCode());
} else {
qCritical() << "Modbus reply error occurred" << error << reply->errorString();
}
});
} else {
QModbusDataUnit request = QModbusDataUnit(registerType, registerAddress, length);
QDataStream stream(writeData);
qDebug() << "Reading write data" << writeData;
quint16 data = writeData.toUInt();
request.setValues({data});
qDebug() << "Writing" << request.values();
QModbusReply *reply = client->sendWriteRequest(request, modbusServerAddress);
if (!reply) {
qCritical() << "Failed to read register" << client->errorString();
exit(EXIT_FAILURE);
}
if (reply->isFinished()) {
reply->deleteLater(); // broadcast replies return immediately
qCritical() << "Reply finished immediatly. Something might have gone wrong:" << reply->errorString();
exit(EXIT_FAILURE);
}
QObject::connect(reply, &QModbusReply::finished, reply, &QModbusReply::deleteLater);
QObject::connect(reply, &QModbusReply::finished, client, [=]() {
if (reply->error() != QModbusDevice::NoError) {
QModbusResponse response = reply->rawResult();
if (reply->error() == QModbusDevice::ProtocolError && response.isException()) {
qCritical() << "Modbus reply finished with error" << reply->error() << reply->errorString() << exceptionCodeToString(response.exceptionCode());
} else {
qCritical() << "Modbus reply finished with error" << reply->error() << reply->errorString();
}
exit(EXIT_FAILURE);
}
const QModbusDataUnit unit = reply->result();
// Note: we need the cast in since the valueCount() type changes with different Qt versions
for (int i = 0; i < static_cast<int>(unit.valueCount()); i++) {
quint16 registerValue = unit.values().at(i);
quint16 registerNumber = unit.startAddress() + i;
qInfo() << "-->" << registerNumber << ":" << QString("0x%1").arg(registerValue, 4, 16, QLatin1Char('0')) << registerValue;
}
exit(EXIT_SUCCESS);
});
QObject::connect(reply, &QModbusReply::errorOccurred, client, [=] (QModbusDevice::Error error){
QModbusResponse response = reply->rawResult();
if (reply->error() == QModbusDevice::ProtocolError && response.isException()) {
qCritical() << "Modbus reply error occurred" << error << reply->errorString() << exceptionCodeToString(response.exceptionCode());
} else {
qCritical() << "Modbus reply error occurred" << error << reply->errorString();
}
});
}
}
QString exceptionCodeToString(QModbusPdu::ExceptionCode exception)
{
QString exceptionString;
switch (exception) {
case QModbusPdu::IllegalFunction:
exceptionString = "Illegal function";
break;
case QModbusPdu::IllegalDataAddress:
exceptionString = "Illegal data address";
break;
case QModbusPdu::IllegalDataValue:
exceptionString = "Illegal data value";
break;
case QModbusPdu::ServerDeviceFailure:
exceptionString = "Server device failure";
break;
case QModbusPdu::Acknowledge:
exceptionString = "Acknowledge";
break;
case QModbusPdu::ServerDeviceBusy:
exceptionString = "Server device busy";
break;
case QModbusPdu::NegativeAcknowledge:
exceptionString = "Negative acknowledge";
break;
case QModbusPdu::MemoryParityError:
exceptionString = "Memory parity error";
break;
case QModbusPdu::GatewayPathUnavailable:
exceptionString = "Gateway path unavailable";
break;
case QModbusPdu::GatewayTargetDeviceFailedToRespond:
exceptionString = "Gateway target device failed to respond";
break;
case QModbusPdu::ExtendedException:
exceptionString = "Extended exception";
break;
}
return exceptionString;
}