-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathmodbusResponse.hpp
165 lines (144 loc) · 5.39 KB
/
modbusResponse.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
// Modbus for c++ <https://github.com/Mazurel/Modbus>
// Copyright (c) 2024 Mateusz Mazur aka Mazurel
// Licensed under: MIT License <http://opensource.org/licenses/MIT>
#pragma once
#include <cassert>
#include <cstdint>
#include <stdexcept>
#include <string>
#include <vector>
#include "modbusCell.hpp"
#include "modbusException.hpp"
#include "modbusRequest.hpp"
#include "modbusUtils.hpp"
/**
* Namespace that contains whole project
*/
namespace MB {
/**
* This class represents Modbus response and allows
* user to manipulate and represent it in various ways
*/
class ModbusResponse {
private:
uint8_t _slaveID;
utils::MBFunctionCode _functionCode;
uint16_t _address;
uint16_t _registersNumber;
std::vector<ModbusCell> _values;
public:
// We do not allow default CTORs: https://github.com/Mazurel/Modbus/issues/6
ModbusResponse() = delete;
/**
* @brief
* Constructs Response from raw data
* @note
* 1) if CRC = true input data needs to contain 2 CRC bytes on back (used in
*RS)
* @note
* 2) This is private constructor, you need to use fromRaw or fromRawCRC
* @param inputData - Is vector of bytes that will be be interpreted,
*whereas based on CRC parameter method performs CRC calculation and throws
*exception if it is invalid
* @throws ModbusException
**/
explicit ModbusResponse(std::vector<uint8_t> inputData, bool CRC = false);
/*
* @description Constructs Response from raw data
* @params inputData is a vector of bytes that will be interpreted
* @throws ModbusException
**/
static ModbusResponse fromRaw(std::vector<uint8_t> inputData) {
return ModbusResponse(inputData);
}
/*
* @description Constructs Request from raw data and checks it's CRC
* @params inputData is a vector of bytes that will be interpreted
* @throws ModbusException
* @note This methods performs CRC check that may throw ModbusException on
* invalid CRC
**/
static ModbusResponse fromRawCRC(std::vector<uint8_t> inputData) {
return ModbusResponse(inputData, true);
}
/**
* Simple constructor, that allows to create "dummy" ModbusResponse
* object. May be useful in some cases.
*/
ModbusResponse(
uint8_t slaveId = 0,
utils::MBFunctionCode functionCode = static_cast<utils::MBFunctionCode>(0),
uint16_t address = 0, uint16_t registersNumber = 0,
std::vector<ModbusCell> values = {});
/**
* Copy constructor for the response.
*/
ModbusResponse(const ModbusResponse &);
/**
* Equal operator for the response.
*/
ModbusResponse &operator=(const ModbusResponse &);
//! Converts object to it's string representation
[[nodiscard]] std::string toString() const;
/**
* @brief Convert Modbus response object into the vector of bytes,
* representing modbus frame, ready to be sent over HW interface
*
* @throws ModbusException - if modbus data in the object is invalid
*/
[[nodiscard]] std::vector<uint8_t> toRaw() const;
/*
* @description Constructs response based on input modbus request
* @note Resulting Modbus response is not guaranteed to be correct
**/
static ModbusResponse from(const ModbusRequest &request) {
return ModbusResponse(request.slaveID(), request.functionCode(),
request.registerAddress(), request.numberOfRegisters(),
request.registerValues());
}
[[nodiscard]] utils::MBFunctionType functionType() const {
return utils::functionType(_functionCode);
}
[[nodiscard]] utils::MBFunctionRegisters functionRegisters() const {
return utils::functionRegister(_functionCode);
}
[[nodiscard]] uint8_t slaveID() const { return _slaveID; }
[[nodiscard]] utils::MBFunctionCode functionCode() const { return _functionCode; }
[[nodiscard]] uint16_t registerAddress() const { return _address; }
[[nodiscard]] uint16_t numberOfRegisters() const { return _registersNumber; }
[[nodiscard]] const std::vector<ModbusCell> ®isterValues() const {
if (this->_values.size() <= 0) {
throw ModbusException(utils::NumberOfValuesInvalid);
}
return _values;
}
[[nodiscard]] uint16_t numberOfBytesToFollow() const {
if (this->functionType() == utils::Read) {
if ((*this->registerValues().begin()).isCoil()) {
// Coils
return (this->numberOfRegisters() / 8) +
(this->numberOfRegisters() % 8 == 0 ? 0 : 1);
} else {
// Regs
return this->numberOfRegisters() * 2;
}
} else {
if (this->functionType() == utils::WriteSingle) {
return static_cast<uint16_t>(2);
} else {
return this->numberOfRegisters();
}
}
}
void setSlaveId(uint8_t slaveId) { _slaveID = slaveId; }
void setFunctionCode(utils::MBFunctionCode functionCode) {
_functionCode = functionCode;
}
void setAddress(uint16_t address) { _address = address; }
void setRegistersNumber(uint16_t registersNumber) {
_registersNumber = registersNumber;
_values.resize(registersNumber);
}
void setValues(const std::vector<ModbusCell> &values) { _values = values; }
};
} // namespace MB