ESPHome 2026.3.0-dev
Loading...
Searching...
No Matches
rd03d.cpp
Go to the documentation of this file.
1#include "rd03d.h"
3#include "esphome/core/log.h"
4#include <cmath>
5
6namespace esphome::rd03d {
7
8static const char *const TAG = "rd03d";
9
10// Delay before sending configuration commands to allow radar to initialize
11static constexpr uint32_t SETUP_TIMEOUT_MS = 100;
12
13// Data frame format (radar -> host)
14static constexpr uint8_t FRAME_HEADER[] = {0xAA, 0xFF, 0x03, 0x00};
15static constexpr uint8_t FRAME_FOOTER[] = {0x55, 0xCC};
16
17// Command frame format (host -> radar)
18static constexpr uint8_t CMD_FRAME_HEADER[] = {0xFD, 0xFC, 0xFB, 0xFA};
19static constexpr uint8_t CMD_FRAME_FOOTER[] = {0x04, 0x03, 0x02, 0x01};
20
21// RD-03D tracking mode commands
22static constexpr uint16_t CMD_SINGLE_TARGET = 0x0080;
23static constexpr uint16_t CMD_MULTI_TARGET = 0x0090;
24
25// Speed sentinel values (cm/s) - radar outputs these when no valid Doppler measurement
26// FMCW radars detect motion via Doppler shift; targets with these speeds are likely noise
27static constexpr int16_t SPEED_SENTINEL_248 = 248;
28static constexpr int16_t SPEED_SENTINEL_256 = 256;
29
30// Decode coordinate/speed value from RD-03D format
31// Per datasheet: MSB=1 means positive, MSB=0 means negative
32static constexpr int16_t decode_value(uint8_t low_byte, uint8_t high_byte) {
33 int16_t value = ((high_byte & 0x7F) << 8) | low_byte;
34 if ((high_byte & 0x80) == 0) {
35 value = -value;
36 }
37 return value;
38}
39
40// Check if speed value indicates a valid Doppler measurement
41// Zero, ±248, or ±256 cm/s are sentinel values from the radar firmware
42static constexpr bool is_speed_valid(int16_t speed) {
43 int16_t abs_speed = speed < 0 ? -speed : speed;
44 return speed != 0 && abs_speed != SPEED_SENTINEL_248 && abs_speed != SPEED_SENTINEL_256;
45}
46
48 ESP_LOGCONFIG(TAG, "Setting up RD-03D...");
49 this->set_timeout(SETUP_TIMEOUT_MS, [this]() { this->apply_config_(); });
50}
51
53 ESP_LOGCONFIG(TAG, "RD-03D:");
54 if (this->tracking_mode_.has_value()) {
55 ESP_LOGCONFIG(TAG, " Tracking Mode: %s",
56 *this->tracking_mode_ == TrackingMode::SINGLE_TARGET ? "single" : "multi");
57 }
58 if (this->throttle_ > 0) {
59 ESP_LOGCONFIG(TAG, " Throttle: %ums", this->throttle_);
60 }
61#ifdef USE_SENSOR
62 LOG_SENSOR(" ", "Target Count", this->target_count_sensor_);
63#endif
64#ifdef USE_BINARY_SENSOR
65 LOG_BINARY_SENSOR(" ", "Target", this->target_binary_sensor_);
66#endif
67 for (uint8_t i = 0; i < MAX_TARGETS; i++) {
68 ESP_LOGCONFIG(TAG, " Target %d:", i + 1);
69#ifdef USE_SENSOR
70 LOG_SENSOR(" ", "X", this->targets_[i].x);
71 LOG_SENSOR(" ", "Y", this->targets_[i].y);
72 LOG_SENSOR(" ", "Speed", this->targets_[i].speed);
73 LOG_SENSOR(" ", "Distance", this->targets_[i].distance);
74 LOG_SENSOR(" ", "Resolution", this->targets_[i].resolution);
75 LOG_SENSOR(" ", "Angle", this->targets_[i].angle);
76#endif
77#ifdef USE_BINARY_SENSOR
78 LOG_BINARY_SENSOR(" ", "Presence", this->target_presence_[i]);
79#endif
80 }
81}
82
84 // Read all available bytes in batches to reduce UART call overhead.
85 size_t avail = this->available();
86 uint8_t buf[64];
87 while (avail > 0) {
88 size_t to_read = std::min(avail, sizeof(buf));
89 if (!this->read_array(buf, to_read)) {
90 break;
91 }
92 avail -= to_read;
93 for (size_t i = 0; i < to_read; i++) {
94 uint8_t byte = buf[i];
95 ESP_LOGVV(TAG, "Received byte: 0x%02X, buffer_pos: %d", byte, this->buffer_pos_);
96
97 // Check if we're looking for frame header
98 if (this->buffer_pos_ < FRAME_HEADER_SIZE) {
99 if (byte == FRAME_HEADER[this->buffer_pos_]) {
100 this->buffer_[this->buffer_pos_++] = byte;
101 } else if (byte == FRAME_HEADER[0]) {
102 // Start over if we see a potential new header
103 this->buffer_[0] = byte;
104 this->buffer_pos_ = 1;
105 } else {
106 this->buffer_pos_ = 0;
107 }
108 continue;
109 }
110
111 // Accumulate data bytes
112 this->buffer_[this->buffer_pos_++] = byte;
113
114 // Check if we have a complete frame
115 if (this->buffer_pos_ == FRAME_SIZE) {
116 // Validate footer
117 if (this->buffer_[FRAME_SIZE - 2] == FRAME_FOOTER[0] && this->buffer_[FRAME_SIZE - 1] == FRAME_FOOTER[1]) {
118 this->process_frame_();
119 } else {
120 ESP_LOGW(TAG, "Invalid frame footer: 0x%02X 0x%02X (expected 0x55 0xCC)", this->buffer_[FRAME_SIZE - 2],
121 this->buffer_[FRAME_SIZE - 1]);
122 }
123 this->buffer_pos_ = 0;
124 }
125 }
126 }
127}
128
130 // Apply throttle if configured
131 if (this->throttle_ > 0) {
132 uint32_t now = millis();
133 if (now - this->last_publish_time_ < this->throttle_) {
134 return;
135 }
136 this->last_publish_time_ = now;
137 }
138
139 uint8_t target_count = 0;
140
141 for (uint8_t i = 0; i < MAX_TARGETS; i++) {
142 // Calculate offset for this target's data
143 // Header is 4 bytes, each target is 8 bytes
144 uint8_t offset = FRAME_HEADER_SIZE + (i * TARGET_DATA_SIZE);
145
146 // Extract raw bytes for this target (per datasheet Table 5-2: X, Y, Speed, Resolution)
147 uint8_t x_low = this->buffer_[offset + 0];
148 uint8_t x_high = this->buffer_[offset + 1];
149 uint8_t y_low = this->buffer_[offset + 2];
150 uint8_t y_high = this->buffer_[offset + 3];
151 uint8_t speed_low = this->buffer_[offset + 4];
152 uint8_t speed_high = this->buffer_[offset + 5];
153 uint8_t res_low = this->buffer_[offset + 6];
154 uint8_t res_high = this->buffer_[offset + 7];
155
156 // Decode values per RD-03D format
157 int16_t x = decode_value(x_low, x_high);
158 int16_t y = decode_value(y_low, y_high);
159 int16_t speed = decode_value(speed_low, speed_high);
160 uint16_t resolution = (res_high << 8) | res_low;
161
162 // Check if target is present
163 // Requires non-zero coordinates AND valid speed (not a sentinel value)
164 // FMCW radars detect motion via Doppler; sentinel speed indicates no real target
165 bool has_position = (x != 0 || y != 0);
166 bool has_valid_speed = is_speed_valid(speed);
167 bool target_present = has_position && has_valid_speed;
168 if (target_present) {
169 target_count++;
170 }
171
172#ifdef USE_SENSOR
173 this->publish_target_(i, x, y, speed, resolution);
174#endif
175
176#ifdef USE_BINARY_SENSOR
177 if (this->target_presence_[i] != nullptr) {
178 this->target_presence_[i]->publish_state(target_present);
179 }
180#endif
181 }
182
183#ifdef USE_SENSOR
184 if (this->target_count_sensor_ != nullptr) {
185 this->target_count_sensor_->publish_state(target_count);
186 }
187#endif
188
189#ifdef USE_BINARY_SENSOR
190 if (this->target_binary_sensor_ != nullptr) {
191 this->target_binary_sensor_->publish_state(target_count > 0);
192 }
193#endif
194}
195
196#ifdef USE_SENSOR
197void RD03DComponent::publish_target_(uint8_t target_num, int16_t x, int16_t y, int16_t speed, uint16_t resolution) {
198 TargetSensor &target = this->targets_[target_num];
199 bool valid = is_speed_valid(speed);
200
201 // Publish X coordinate (mm) - NaN if target invalid
202 if (target.x != nullptr) {
203 target.x->publish_state(valid ? static_cast<float>(x) : NAN);
204 }
205
206 // Publish Y coordinate (mm) - NaN if target invalid
207 if (target.y != nullptr) {
208 target.y->publish_state(valid ? static_cast<float>(y) : NAN);
209 }
210
211 // Publish speed (convert from cm/s to mm/s) - NaN if target invalid
212 if (target.speed != nullptr) {
213 target.speed->publish_state(valid ? static_cast<float>(speed) * 10.0f : NAN);
214 }
215
216 // Publish resolution (mm)
217 if (target.resolution != nullptr) {
219 }
220
221 // Calculate and publish distance (mm) - NaN if target invalid
222 if (target.distance != nullptr) {
223 if (valid) {
224 target.distance->publish_state(std::hypot(static_cast<float>(x), static_cast<float>(y)));
225 } else {
226 target.distance->publish_state(NAN);
227 }
228 }
229
230 // Calculate and publish angle (degrees) - NaN if target invalid
231 // Angle is measured from the Y axis (radar forward direction)
232 if (target.angle != nullptr) {
233 if (valid) {
234 float angle = std::atan2(static_cast<float>(x), static_cast<float>(y)) * 180.0f / M_PI;
235 target.angle->publish_state(angle);
236 } else {
237 target.angle->publish_state(NAN);
238 }
239 }
240}
241#endif
242
243void RD03DComponent::send_command_(uint16_t command, const uint8_t *data, uint8_t data_len) {
244 // Send header
245 this->write_array(CMD_FRAME_HEADER, sizeof(CMD_FRAME_HEADER));
246
247 // Send length (command word + data)
248 uint16_t len = 2 + data_len;
249 this->write_byte(len & 0xFF);
250 this->write_byte((len >> 8) & 0xFF);
251
252 // Send command word (little-endian)
253 this->write_byte(command & 0xFF);
254 this->write_byte((command >> 8) & 0xFF);
255
256 // Send data if any
257 if (data != nullptr && data_len > 0) {
258 this->write_array(data, data_len);
259 }
260
261 // Send footer
262 this->write_array(CMD_FRAME_FOOTER, sizeof(CMD_FRAME_FOOTER));
263
264 ESP_LOGD(TAG, "Sent command 0x%04X with %d bytes of data", command, data_len);
265}
266
268 if (this->tracking_mode_.has_value()) {
269 uint16_t mode_cmd = (*this->tracking_mode_ == TrackingMode::SINGLE_TARGET) ? CMD_SINGLE_TARGET : CMD_MULTI_TARGET;
270 this->send_command_(mode_cmd);
271 }
272}
273
274} // namespace esphome::rd03d
ESPDEPRECATED("Use const char* or uint32_t overload instead. Removed in 2026.7.0", "2026.1.0") void set_timeout(const std voi set_timeout)(const char *name, uint32_t timeout, std::function< void()> &&f)
Set a timeout function with a unique name.
Definition component.h:429
void publish_state(bool new_state)
Publish a new state to the front-end.
bool has_value() const
Definition optional.h:92
optional< TrackingMode > tracking_mode_
Definition rd03d.h:84
std::array< TargetSensor, MAX_TARGETS > targets_
Definition rd03d.h:75
void dump_config() override
Definition rd03d.cpp:52
void send_command_(uint16_t command, const uint8_t *data=nullptr, uint8_t data_len=0)
Definition rd03d.cpp:243
binary_sensor::BinarySensor * target_binary_sensor_
Definition rd03d.h:80
std::array< uint8_t, FRAME_SIZE > buffer_
Definition rd03d.h:88
void publish_target_(uint8_t target_num, int16_t x, int16_t y, int16_t speed, uint16_t resolution)
Definition rd03d.cpp:197
std::array< binary_sensor::BinarySensor *, MAX_TARGETS > target_presence_
Definition rd03d.h:79
sensor::Sensor * target_count_sensor_
Definition rd03d.h:76
void publish_state(float state)
Publish a new state to the front-end.
Definition sensor.cpp:65
optional< std::array< uint8_t, N > > read_array()
Definition uart.h:38
void write_byte(uint8_t data)
Definition uart.h:18
void write_array(const uint8_t *data, size_t len)
Definition uart.h:26
int speed
Definition fan.h:1
Resolution resolution
Definition msa3xx.h:1
std::string size_t len
Definition helpers.h:692
uint32_t IRAM_ATTR HOT millis()
Definition core.cpp:25
sensor::Sensor * resolution
Definition rd03d.h:35
sensor::Sensor * distance
Definition rd03d.h:34
sensor::Sensor * x
Definition rd03d.h:31
sensor::Sensor * speed
Definition rd03d.h:33
sensor::Sensor * y
Definition rd03d.h:32
sensor::Sensor * angle
Definition rd03d.h:36
uint16_t x
Definition tt21100.cpp:5
uint16_t y
Definition tt21100.cpp:6