ESPHome 2026.5.1
Loading...
Searching...
No Matches
modbus.h
Go to the documentation of this file.
1#pragma once
2
5
7
8#include <cstring>
9#include <memory>
10#include <vector>
11#include <queue>
12
13namespace esphome::modbus {
14
15static constexpr uint16_t MODBUS_TX_BUFFER_SIZE = 15;
16
21
22class ModbusDevice;
23
25 // Frame with exact-size allocation to avoid std::vector overhead
26 std::unique_ptr<uint8_t[]> data;
27 uint16_t size; // Modbus RTU max is 256 bytes
28
29 ModbusDeviceCommand(const uint8_t *src, uint16_t len) : data(std::make_unique<uint8_t[]>(len + 2)), size(len + 2) {
30 std::memcpy(this->data.get(), src, len);
31 auto crc = crc16(data.get(), len);
32 data[len + 0] = crc >> 0;
33 data[len + 1] = crc >> 8;
34 }
35};
36
37class Modbus : public uart::UARTDevice, public Component {
38 public:
39 Modbus() = default;
40
41 void setup() override;
42
43 void loop() override;
44
45 void dump_config() override;
46
47 void register_device(ModbusDevice *device) { this->devices_.push_back(device); }
48
49 float get_setup_priority() const override;
50 bool tx_buffer_empty();
51 bool tx_blocked();
52
53 void send(uint8_t address, uint8_t function_code, uint16_t start_address, uint16_t number_of_entities,
54 uint8_t payload_len = 0, const uint8_t *payload = nullptr);
55 void send_raw(const std::vector<uint8_t> &payload);
56 void set_role(ModbusRole role) { this->role = role; }
57 void set_flow_control_pin(GPIOPin *flow_control_pin) { this->flow_control_pin_ = flow_control_pin; }
58 void set_send_wait_time(uint16_t time_in_ms) { this->send_wait_time_ = time_in_ms; }
59 void set_turnaround_time(uint16_t time_in_ms) { this->turnaround_delay_ms_ = time_in_ms; }
60 void set_disable_crc(bool disable_crc) { this->disable_crc_ = disable_crc; }
61
63
64 protected:
65 bool parse_modbus_byte_(uint8_t byte);
67 void clear_rx_buffer_(const LogString *reason, bool warn = false);
68 void send_next_frame_();
69 void queue_raw_(const uint8_t *data, uint16_t len);
70
74 uint16_t frame_delay_ms_{5};
76 uint16_t send_wait_time_{250};
77 uint16_t turnaround_delay_ms_{100};
79 bool disable_crc_{false};
80
82
83 std::vector<uint8_t> rx_buffer_;
84 std::vector<ModbusDevice *> devices_;
85 // std::deque is appropriate here since we need a FIFO buffer, and we can't know ahead of time how many
86 // requests will be queued. Each modbus component may queue multiple requests, and the sequence of scheduling
87 // may change at run time.
88 std::deque<ModbusDeviceCommand> tx_buffer_;
89};
90
92 public:
93 void set_parent(Modbus *parent) { parent_ = parent; }
94 void set_address(uint8_t address) { address_ = address; }
95 virtual void on_modbus_data(const std::vector<uint8_t> &data) = 0;
96 virtual void on_modbus_error(uint8_t function_code, uint8_t exception_code) {}
97 virtual void on_modbus_read_registers(uint8_t function_code, uint16_t start_address, uint16_t number_of_registers){};
98 virtual void on_modbus_write_registers(uint8_t function_code, const std::vector<uint8_t> &data){};
99 void send(uint8_t function, uint16_t start_address, uint16_t number_of_entities, uint8_t payload_len = 0,
100 const uint8_t *payload = nullptr) {
101 this->parent_->send(this->address_, function, start_address, number_of_entities, payload_len, payload);
102 }
103 void send_raw(const std::vector<uint8_t> &payload) { this->parent_->send_raw(payload); }
104 void send_error(uint8_t function_code, ModbusExceptionCode exception_code) {
105 std::vector<uint8_t> error_response;
106 error_response.reserve(3);
107 error_response.push_back(this->address_);
108 error_response.push_back(function_code | FUNCTION_CODE_EXCEPTION_MASK);
109 error_response.push_back(static_cast<uint8_t>(exception_code));
110 this->send_raw(error_response);
111 }
112 // If more than one device is connected block sending a new command before a response is received
113 ESPDEPRECATED("Use ready_for_immediate_send() instead. Removed in 2026.9.0", "2026.3.0")
114 bool waiting_for_response() { return !ready_for_immediate_send(); }
116
117 protected:
118 friend Modbus;
119
121 uint8_t address_;
122};
123
124} // namespace esphome::modbus
uint8_t address
Definition bl0906.h:4
void send_raw(const std::vector< uint8_t > &payload)
Definition modbus.h:103
void set_parent(Modbus *parent)
Definition modbus.h:93
virtual void on_modbus_read_registers(uint8_t function_code, uint16_t start_address, uint16_t number_of_registers)
Definition modbus.h:97
void send(uint8_t function, uint16_t start_address, uint16_t number_of_entities, uint8_t payload_len=0, const uint8_t *payload=nullptr)
Definition modbus.h:99
virtual void on_modbus_data(const std::vector< uint8_t > &data)=0
ESPDEPRECATED("Use ready_for_immediate_send() instead. Removed in 2026.9.0", "2026.3.0") bool waiting_for_response()
Definition modbus.h:113
void send_error(uint8_t function_code, ModbusExceptionCode exception_code)
Definition modbus.h:104
virtual void on_modbus_error(uint8_t function_code, uint8_t exception_code)
Definition modbus.h:96
void set_address(uint8_t address)
Definition modbus.h:94
virtual void on_modbus_write_registers(uint8_t function_code, const std::vector< uint8_t > &data)
Definition modbus.h:98
void set_flow_control_pin(GPIOPin *flow_control_pin)
Definition modbus.h:57
std::deque< ModbusDeviceCommand > tx_buffer_
Definition modbus.h:88
bool parse_modbus_byte_(uint8_t byte)
Definition modbus.cpp:119
void send_raw(const std::vector< uint8_t > &payload)
Definition modbus.cpp:383
void setup() override
Definition modbus.cpp:18
uint16_t frame_delay_ms_
Definition modbus.h:74
uint16_t send_wait_time_
Definition modbus.h:76
void set_turnaround_time(uint16_t time_in_ms)
Definition modbus.h:59
void set_role(ModbusRole role)
Definition modbus.h:56
uint8_t waiting_for_response_
Definition modbus.h:78
uint32_t last_modbus_byte_
Definition modbus.h:71
GPIOPin * flow_control_pin_
Definition modbus.h:81
void set_send_wait_time(uint16_t time_in_ms)
Definition modbus.h:58
std::vector< ModbusDevice * > devices_
Definition modbus.h:84
uint32_t last_send_tx_offset_
Definition modbus.h:73
void loop() override
Definition modbus.cpp:39
void queue_raw_(const uint8_t *data, uint16_t len)
Definition modbus.cpp:402
void set_disable_crc(bool disable_crc)
Definition modbus.h:60
float get_setup_priority() const override
Definition modbus.cpp:331
uint16_t long_rx_buffer_delay_ms_
Definition modbus.h:75
void dump_config() override
Definition modbus.cpp:319
void receive_and_parse_modbus_bytes_()
Definition modbus.cpp:91
void register_device(ModbusDevice *device)
Definition modbus.h:47
std::vector< uint8_t > rx_buffer_
Definition modbus.h:83
void clear_rx_buffer_(const LogString *reason, bool warn=false)
Definition modbus.cpp:413
void send(uint8_t address, uint8_t function_code, uint16_t start_address, uint16_t number_of_entities, uint8_t payload_len=0, const uint8_t *payload=nullptr)
Definition modbus.cpp:336
uint16_t turnaround_delay_ms_
Definition modbus.h:77
const uint8_t FUNCTION_CODE_EXCEPTION_MASK
uint16_t crc16(const uint8_t *data, uint16_t len, uint16_t crc, uint16_t reverse_poly, bool refin, bool refout)
Calculate a CRC-16 checksum of data with size len.
Definition helpers.cpp:86
std::string size_t len
static void uint32_t
std::unique_ptr< uint8_t[]> data
Definition modbus.h:26
ModbusDeviceCommand(const uint8_t *src, uint16_t len)
Definition modbus.h:29