// Copyright (C) 2017 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #include "qmodbusreply.h" #include #include QT_BEGIN_NAMESPACE class QModbusReplyPrivate : public QObjectPrivate { Q_DECLARE_PUBLIC(QModbusReply) public: QModbusDataUnit m_unit; int m_serverAddress = 1; bool m_finished = false; QModbusDevice::Error m_error = QModbusDevice::NoError; QString m_errorText; QModbusResponse m_response; QModbusReply::ReplyType m_type; QList m_intermediateErrors; }; /*! \class QModbusReply \inmodule QtSerialBus \since 5.8 \brief The QModbusReply class contains the data for a request sent with a \l QModbusClient derived class. */ /*! \enum QModbusReply::ReplyType This enum describes the possible reply type. \value Raw The reply originates from a raw Modbus request. See \l QModbusClient::sendRawRequest \value Common The reply originates from a common read, write or read/write request. See \l QModbusClient::sendReadRequest, \l QModbusClient::sendWriteRequest and \l QModbusClient::sendReadWriteRequest \value Broadcast The reply originates from a Modbus broadcast request. The \l serverAddress() will return \c 0 and the \l finished() signal will be emitted immediately. */ /*! Constructs a QModbusReply object with a given \a type and the specified \a parent. The reply will be sent to the Modbus client represented by \a serverAddress. */ QModbusReply::QModbusReply(ReplyType type, int serverAddress, QObject *parent) : QObject(*new QModbusReplyPrivate, parent) { Q_D(QModbusReply); d->m_type = type; d->m_serverAddress = serverAddress; } /*! Returns \c true when the reply has finished or was aborted. \sa finished(), error() */ bool QModbusReply::isFinished() const { Q_D(const QModbusReply); return d->m_finished; } /*! \internal Sets whether or not this reply has finished to \a isFinished. If \a isFinished is \c true, this will cause the \l finished() signal to be emitted. If the operation completed successfully, \l setResult() should be called before this function. If an error occurred, \l setError() should be used instead. */ void QModbusReply::setFinished(bool isFinished) { Q_D(QModbusReply); d->m_finished = isFinished; if (isFinished) emit finished(); } /*! \fn void QModbusReply::finished() This signal is emitted when the reply has finished processing. The reply may still have returned with an error. After this signal is emitted, there will be no more updates to the reply's data. \note Do not delete the object in the slot connected to this signal. Use deleteLater(). You can also use \l isFinished() to check if a QNetworkReply has finished even before you receive the \l finished() signal. \sa isFinished(), error() */ /*! Returns the preprocessed result of a Modbus request. For read requests as well as combined read/write requests send via \l QModbusClient::sendReadWriteRequest() it contains the values read from the server instance. If the request has not finished, has failed with an error or was a write request then the returned \l QModbusDataUnit instance is invalid. \note If the \l type() of the reply is \l QModbusReply::Broadcast, the return value will always be invalid. If the \l type() of the reply is \l QModbusReply::Raw, the return value might be invalid depending on the implementation of \l QModbusClient::processPrivateResponse(). \sa type(), rawResult(), QModbusClient::processPrivateResponse() */ QModbusDataUnit QModbusReply::result() const { Q_D(const QModbusReply); if (type() != QModbusReply::Broadcast) return d->m_unit; return QModbusDataUnit(); } /*! \internal Sets the results of a read/write request to a Modbus register data \a unit. */ void QModbusReply::setResult(const QModbusDataUnit &unit) { Q_D(QModbusReply); d->m_unit = unit; } /*! Returns the server address that this reply object targets. */ int QModbusReply::serverAddress() const { Q_D(const QModbusReply); return d->m_serverAddress; } /*! \fn void QModbusReply::errorOccurred(QModbusDevice::Error error) This signal is emitted when an error has been detected in the processing of this reply. The \l finished() signal will probably follow. The error will be described by the error code \a error. If errorString is not empty it will contain a textual description of the error. In case of a \l QModbusDevice::ProtocolError the \l rawResult() function can be used to obtain the original Modbus exception response to get the exception code. Note: Do not delete this reply object in the slot connected to this signal. Use \l deleteLater() instead. \sa error(), errorString() */ /*! Returns the error state of this reply. \sa errorString(), errorOccurred() */ QModbusDevice::Error QModbusReply::error() const { Q_D(const QModbusReply); return d->m_error; } /*! \internal Sets the error state of this reply to \a error and the textual representation of the error to \a errorText. This will also cause the \l errorOccurred() and \l finished() signals to be emitted, in that order. */ void QModbusReply::setError(QModbusDevice::Error error, const QString &errorText) { Q_D(QModbusReply); d->m_error = error; d->m_errorText = errorText; emit errorOccurred(error); setFinished(true); } /*! Returns the textual representation of the error state of this reply. If no error has occurred this will return an empty string. It is possible that an error occurred which has no associated textual representation, in which case this will also return an empty string. \sa error(), errorOccurred() */ QString QModbusReply::errorString() const { Q_D(const QModbusReply); return d->m_errorText; } /*! Returns the type of the reply. \note If the type of the reply is \l QModbusReply::Raw, the return value of \l result() will always be invalid. \sa result(), rawResult() */ QModbusReply::ReplyType QModbusReply::type() const { Q_D(const QModbusReply); return d->m_type; } /*! Returns the raw response of a Modbus request. If the request has not finished then the returned \l QModbusResponse instance is invalid. \sa type(), result() */ QModbusResponse QModbusReply::rawResult() const { Q_D(const QModbusReply); return d->m_response; } /*! \internal Sets the result of a Modbus request to a Modbus \a response. */ void QModbusReply::setRawResult(const QModbusResponse &response) { Q_D(QModbusReply); d->m_response = response; } /*! \since 6.0 \fn void intermediateErrorOccurred(QModbusDevice::IntermediateError error) This signal is emitted when an error has been detected in the processing of this reply. The error will be described by the error code \a error. */ /*! \since 6.0 Returns the list of intermediate errors that might have happened during the send-receive cycle of a Modbus request until the QModbusReply reports to be finished. */ QList QModbusReply::intermediateErrors() const { Q_D(const QModbusReply); return d->m_intermediateErrors; } /*! \internal \since 6.0 Adds an intermediate error to the list of intermediate errors. This will also cause the \l intermediateErrorOccurred() signal to be emitted. */ void QModbusReply::addIntermediateError(QModbusDevice::IntermediateError error) { Q_D(QModbusReply); d->m_intermediateErrors.append(error); emit intermediateErrorOccurred(error); } QT_END_NAMESPACE #include "moc_qmodbusreply.cpp"