ESPHome 2026.3.0-dev
Loading...
Searching...
No Matches
sonoff_d1.cpp
Go to the documentation of this file.
1/*
2 sonoff_d1.cpp - Sonoff D1 Dimmer support for ESPHome
3
4 Copyright © 2021 Anatoly Savchenkov
5 Copyright © 2020 Jeff Rescignano
6
7 Permission is hereby granted, free of charge, to any person obtaining a copy of this software
8 and associated documentation files (the “Software”), to deal in the Software without
9 restriction, including without limitation the rights to use, copy, modify, merge, publish,
10 distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom
11 the Software is furnished to do so, subject to the following conditions:
12
13 The above copyright notice and this permission notice shall be included in all copies or
14 substantial portions of the Software.
15
16 THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
17 BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
19 DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20 FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21
22 -----
23
24 If modifying this file, in addition to the license above, please ensure to include links back to the original code:
25 https://jeffresc.dev/blog/2020-10-10
26 https://github.com/JeffResc/Sonoff-D1-Dimmer
27 https://github.com/arendst/Tasmota/blob/2d4a6a29ebc7153dbe2717e3615574ac1c84ba1d/tasmota/xdrv_37_sonoff_d1.ino#L119-L131
28
29 -----
30*/
31
32/*********************************************************************************************\
33 * Sonoff D1 dimmer 433
34 * Mandatory/Optional
35 * ^ 0 1 2 3 4 5 6 7 8 9 A B C D E F 10
36 * M AA 55 - Header
37 * M 01 04 - Version?
38 * M 00 0A - Following data length (10 bytes)
39 * O 01 - Power state (00 = off, 01 = on, FF = ignore)
40 * O 64 - Dimmer percentage (01 to 64 = 1 to 100%, 0 - ignore)
41 * O FF FF FF FF FF FF FF FF - Not used
42 * M 6C - CRC over bytes 2 to F (Addition)
43\*********************************************************************************************/
44#include "sonoff_d1.h"
46
47namespace esphome {
48namespace sonoff_d1 {
49
50static const char *const TAG = "sonoff_d1";
51
52// Protocol constants
53static constexpr size_t SONOFF_D1_ACK_SIZE = 7;
54static constexpr size_t SONOFF_D1_MAX_CMD_SIZE = 17;
55
56uint8_t SonoffD1Output::calc_checksum_(const uint8_t *cmd, const size_t len) {
57 uint8_t crc = 0;
58 for (size_t i = 2; i < len - 1; i++) {
59 crc += cmd[i];
60 }
61 return crc;
62}
63
64void SonoffD1Output::populate_checksum_(uint8_t *cmd, const size_t len) {
65 // Update the checksum
66 cmd[len - 1] = this->calc_checksum_(cmd, len);
67}
68
70 size_t garbage = 0;
71 // Read out everything from the UART FIFO
72 while (this->available()) {
73 uint8_t value = this->read();
74 ESP_LOGW(TAG, "[%04d] Skip %02d: 0x%02x from the dimmer", this->write_count_, garbage, value);
75 garbage++;
76 }
77
78 // Warn about unexpected bytes in the protocol with UART dimmer
79 if (garbage) {
80 ESP_LOGW(TAG, "[%04d] Skip %d bytes from the dimmer", this->write_count_, garbage);
81 }
82}
83
84// This assumes some data is already available
85bool SonoffD1Output::read_command_(uint8_t *cmd, size_t &len) {
86 // Do consistency check
87 if (cmd == nullptr || len < 7) {
88 ESP_LOGW(TAG, "[%04d] Too short command buffer (actual len is %d bytes, minimal is 7)", this->write_count_, len);
89 return false;
90 }
91
92 // Read a minimal packet
93 if (this->read_array(cmd, 6)) {
94#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
95 char hex_buf[format_hex_pretty_size(6)];
96 ESP_LOGV(TAG, "[%04d] Reading from dimmer: %s", this->write_count_, format_hex_pretty_to(hex_buf, cmd, 6));
97#endif
98
99 if (cmd[0] != 0xAA || cmd[1] != 0x55) {
100 ESP_LOGW(TAG, "[%04d] RX: wrong header (%x%x, must be AA55)", this->write_count_, cmd[0], cmd[1]);
101 this->skip_command_();
102 return false;
103 }
104 if ((cmd[5] + 7 /*mandatory header + crc suffix length*/) > len) {
105 ESP_LOGW(TAG, "[%04d] RX: Payload length is unexpected (%d, max expected %d)", this->write_count_, cmd[5],
106 len - 7);
107 this->skip_command_();
108 return false;
109 }
110 if (this->read_array(&cmd[6], cmd[5] + 1 /*checksum suffix*/)) {
111#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
112 char hex_buf2[format_hex_pretty_size(SONOFF_D1_MAX_CMD_SIZE)];
113 ESP_LOGV(TAG, "[%04d] %s", this->write_count_, format_hex_pretty_to(hex_buf2, &cmd[6], cmd[5] + 1));
114#endif
115
116 // Check the checksum
117 uint8_t valid_checksum = this->calc_checksum_(cmd, cmd[5] + 7);
118 if (valid_checksum != cmd[cmd[5] + 7 - 1]) {
119 ESP_LOGW(TAG, "[%04d] RX: checksum mismatch (%d, expected %d)", this->write_count_, cmd[cmd[5] + 7 - 1],
120 valid_checksum);
121 this->skip_command_();
122 return false;
123 }
124 len = cmd[5] + 7 /*mandatory header + suffix length*/;
125
126 // Read remaining gardbled data (just in case, I don't see where this can appear now)
127 this->skip_command_();
128 return true;
129 }
130 } else {
131 ESP_LOGW(TAG, "[%04d] RX: feedback timeout", this->write_count_);
132 this->skip_command_();
133 }
134 return false;
135}
136
137bool SonoffD1Output::read_ack_(const uint8_t *cmd, const size_t len) {
138 // Expected acknowledgement from rf chip
139 uint8_t ref_buffer[7] = {0xAA, 0x55, cmd[2], cmd[3], 0x00, 0x00, 0x00};
140 uint8_t buffer[sizeof(ref_buffer)] = {0};
141 uint32_t pos = 0;
142 size_t buf_len = sizeof(ref_buffer);
143
144 // Update the reference checksum
145 this->populate_checksum_(ref_buffer, sizeof(ref_buffer));
146
147 // Read ack code, this either reads 7 bytes or exits with a timeout
148 this->read_command_(buffer, buf_len);
149
150 // Compare response with expected response
151 while (pos < sizeof(ref_buffer) && ref_buffer[pos] == buffer[pos]) {
152 pos++;
153 }
154 if (pos == sizeof(ref_buffer)) {
155 ESP_LOGD(TAG, "[%04d] Acknowledge received", this->write_count_);
156 return true;
157 } else {
158 char hex_buf[format_hex_pretty_size(SONOFF_D1_ACK_SIZE)];
159 ESP_LOGW(TAG, "[%04d] Unexpected acknowledge received (possible clash of RF/HA commands), expected ack was:",
160 this->write_count_);
161 ESP_LOGW(TAG, "[%04d] %s", this->write_count_, format_hex_pretty_to(hex_buf, ref_buffer, sizeof(ref_buffer)));
162 }
163 return false;
164}
165
166bool SonoffD1Output::write_command_(uint8_t *cmd, const size_t len, bool needs_ack) {
167 // Do some consistency checks
168 if (len < 7) {
169 ESP_LOGW(TAG, "[%04d] Too short command (actual len is %d bytes, minimal is 7)", this->write_count_, len);
170 return false;
171 }
172 if (cmd[0] != 0xAA || cmd[1] != 0x55) {
173 ESP_LOGW(TAG, "[%04d] Wrong header (%x%x, must be AA55)", this->write_count_, cmd[0], cmd[1]);
174 return false;
175 }
176 if ((cmd[5] + 7 /*mandatory header + suffix length*/) != len) {
177 ESP_LOGW(TAG, "[%04d] Payload length field does not match packet length (%d, expected %d)", this->write_count_,
178 cmd[5], len - 7);
179 return false;
180 }
181 this->populate_checksum_(cmd, len);
182
183 // Need retries here to handle the following cases:
184 // 1. On power up companion MCU starts to respond with a delay, so few first commands are ignored
185 // 2. UART command initiated by this component can clash with a command initiated by RF
186 uint32_t retries = 10;
187 do {
188#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
189 char hex_buf[format_hex_pretty_size(SONOFF_D1_MAX_CMD_SIZE)];
190 ESP_LOGV(TAG, "[%04d] Writing to the dimmer: %s", this->write_count_, format_hex_pretty_to(hex_buf, cmd, len));
191#endif
192 this->write_array(cmd, len);
193 this->write_count_++;
194 if (!needs_ack)
195 return true;
196 retries--;
197 } while (!this->read_ack_(cmd, len) && retries > 0);
198
199 if (retries) {
200 return true;
201 } else {
202 ESP_LOGE(TAG, "[%04d] Unable to write to the dimmer", this->write_count_);
203 }
204 return false;
205}
206
207bool SonoffD1Output::control_dimmer_(const bool binary, const uint8_t brightness) {
208 // Include our basic code from the Tasmota project, thank you again!
209 // 0 1 2 3 4 5 6 7 8
210 uint8_t cmd[17] = {0xAA, 0x55, 0x01, 0x04, 0x00, 0x0A, 0x00, 0x00, 0xFF,
211 // 9 10 11 12 13 14 15 16
212 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00};
213
214 cmd[6] = binary;
215 cmd[7] = remap<uint8_t, uint8_t>(brightness, 0, 100, this->min_value_, this->max_value_);
216 ESP_LOGI(TAG, "[%04d] Setting dimmer state to %s, raw brightness=%d", this->write_count_, ONOFF(binary), cmd[7]);
217 return this->write_command_(cmd, sizeof(cmd));
218}
219
220void SonoffD1Output::process_command_(const uint8_t *cmd, const size_t len) {
221 if (cmd[2] == 0x01 && cmd[3] == 0x04 && cmd[4] == 0x00 && cmd[5] == 0x0A) {
222 uint8_t ack_buffer[7] = {0xAA, 0x55, cmd[2], cmd[3], 0x00, 0x00, 0x00};
223 // Ack a command from RF to ESP to prevent repeating commands
224 this->write_command_(ack_buffer, sizeof(ack_buffer), false);
225 ESP_LOGI(TAG, "[%04d] RF sets dimmer state to %s, raw brightness=%d", this->write_count_, ONOFF(cmd[6]), cmd[7]);
226 const uint8_t new_brightness = remap<uint8_t, uint8_t>(cmd[7], this->min_value_, this->max_value_, 0, 100);
227 const bool new_state = cmd[6];
228
229 // Got light change state command. In all cases we revert the command immediately
230 // since we want to rely on ESP controlled transitions
231 if (new_state != this->last_binary_ || new_brightness != this->last_brightness_) {
233 }
234
235 if (!this->use_rm433_remote_) {
236 // If RF remote is not used, this is a known ghost RF command
237 ESP_LOGI(TAG, "[%04d] Ghost command from RF detected, reverted", this->write_count_);
238 } else {
239 // If remote is used, initiate transition to the new state
240 this->publish_state_(new_state, new_brightness);
241 }
242 } else {
243 ESP_LOGW(TAG, "[%04d] Unexpected command received", this->write_count_);
244 }
245}
246
247void SonoffD1Output::publish_state_(const bool is_on, const uint8_t brightness) {
248 if (light_state_) {
249 ESP_LOGV(TAG, "Publishing new state: %s, brightness=%d", ONOFF(is_on), brightness);
250 auto call = light_state_->make_call();
251 call.set_state(is_on);
252 if (brightness != 0) {
253 // Brightness equal to 0 has a special meaning.
254 // D1 uses 0 as "previously set brightness".
255 // Usually zero brightness comes inside light ON command triggered by RF remote.
256 // Since we unconditionally override commands coming from RF remote in process_command_(),
257 // here we mimic the original behavior but with LightCall functionality
258 call.set_brightness((float) brightness / 100.0f);
259 }
260 call.perform();
261 }
262}
263
264// Set the device's traits
266 auto traits = light::LightTraits();
267 traits.set_supported_color_modes({light::ColorMode::BRIGHTNESS});
268 return traits;
269}
270
272 bool binary;
273 float brightness;
274
275 // Fill our variables with the device's current state
276 state->current_values_as_binary(&binary);
277 state->current_values_as_brightness(&brightness);
278
279 // Convert ESPHome's brightness (0-1) to the device's internal brightness (0-100)
280 const uint8_t calculated_brightness = (uint8_t) roundf(brightness * 100);
281
282 if (calculated_brightness == 0) {
283 // if(binary) ESP_LOGD(TAG, "current_values_as_binary() returns true for zero brightness");
284 binary = false;
285 }
286
287 // If a new value, write to the dimmer
288 if (binary != this->last_binary_ || calculated_brightness != this->last_brightness_) {
289 if (this->control_dimmer_(binary, calculated_brightness)) {
290 this->last_brightness_ = calculated_brightness;
291 this->last_binary_ = binary;
292 } else {
293 // Return to original value if failed to write to the dimmer
294 // TODO: Test me, can be tested if high-voltage part is not connected
295 ESP_LOGW(TAG, "Failed to update the dimmer, publishing the previous state");
296 this->publish_state_(this->last_binary_, this->last_brightness_);
297 }
298 }
299}
300
302 ESP_LOGCONFIG(TAG,
303 "Sonoff D1 Dimmer: '%s'\n"
304 " Use RM433 Remote: %s\n"
305 " Minimal brightness: %d\n"
306 " Maximal brightness: %d",
307 this->light_state_ ? this->light_state_->get_name().c_str() : "", ONOFF(this->use_rm433_remote_),
308 this->min_value_, this->max_value_);
309}
310
312 // Read commands from the dimmer
313 // RF chip notifies ESP about remotely changed state with the same commands as we send
314 if (this->available()) {
315 ESP_LOGV(TAG, "Have some UART data in loop()");
316 uint8_t buffer[17] = {0};
317 size_t len = sizeof(buffer);
318 if (this->read_command_(buffer, len)) {
319 this->process_command_(buffer, len);
320 }
321 }
322}
323
324} // namespace sonoff_d1
325} // namespace esphome
const StringRef & get_name() const
constexpr const char * c_str() const
Definition string_ref.h:73
LightCall & set_state(optional< bool > state)
Set the binary ON/OFF state of the light.
This class represents the communication layer between the front-end MQTT layer and the hardware outpu...
Definition light_state.h:93
This class is used to represent the capabilities of a light.
bool read_command_(uint8_t *cmd, size_t &len)
Definition sonoff_d1.cpp:85
void process_command_(const uint8_t *cmd, size_t len)
uint8_t calc_checksum_(const uint8_t *cmd, size_t len)
Definition sonoff_d1.cpp:56
bool control_dimmer_(bool binary, uint8_t brightness)
void populate_checksum_(uint8_t *cmd, size_t len)
Definition sonoff_d1.cpp:64
light::LightState * light_state_
Definition sonoff_d1.h:70
void write_state(light::LightState *state) override
void publish_state_(bool is_on, uint8_t brightness)
bool read_ack_(const uint8_t *cmd, size_t len)
light::LightTraits get_traits() override
bool write_command_(uint8_t *cmd, size_t len, bool needs_ack=true)
optional< std::array< uint8_t, N > > read_array()
Definition uart.h:38
void write_array(const uint8_t *data, size_t len)
Definition uart.h:26
bool state
Definition fan.h:2
@ BRIGHTNESS
Dimmable light.
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:817
char * format_hex_pretty_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length, char separator)
Format byte array as uppercase hex to buffer (base implementation).
Definition helpers.cpp:353
size_t size_t pos
Definition helpers.h:854
constexpr size_t format_hex_pretty_size(size_t byte_count)
Calculate buffer size needed for format_hex_pretty_to with separator: "XX:XX:...:XX\0".
Definition helpers.h:1103
T remap(U value, U min, U max, T min_out, T max_out)
Remap value from the range (min, max) to (min_out, max_out).
Definition helpers.h:530