18static const char *
const TAG =
"ld2450";
92constexpr uint32_t
BAUD_RATES[] = {9600, 19200, 38400, 57600, 115200, 230400, 256000, 460800};
95template<
size_t N> uint8_t
find_uint8(
const StringToUint8 (&arr)[N],
const std::string &str) {
96 for (
const auto &entry : arr) {
103template<
size_t N>
const char *
find_str(
const Uint8ToString (&arr)[N], uint8_t value) {
104 for (
const auto &entry : arr) {
105 if (value == entry.value)
112static constexpr uint8_t CMD_ENABLE_CONF = 0xFF;
113static constexpr uint8_t CMD_DISABLE_CONF = 0xFE;
114static constexpr uint8_t CMD_QUERY_VERSION = 0xA0;
115static constexpr uint8_t CMD_QUERY_MAC_ADDRESS = 0xA5;
116static constexpr uint8_t CMD_RESET = 0xA2;
117static constexpr uint8_t CMD_RESTART = 0xA3;
118static constexpr uint8_t CMD_BLUETOOTH = 0xA4;
119static constexpr uint8_t CMD_SINGLE_TARGET_MODE = 0x80;
120static constexpr uint8_t CMD_MULTI_TARGET_MODE = 0x90;
121static constexpr uint8_t CMD_QUERY_TARGET_MODE = 0x91;
122static constexpr uint8_t CMD_SET_BAUD_RATE = 0xA1;
123static constexpr uint8_t CMD_QUERY_ZONE = 0xC1;
124static constexpr uint8_t CMD_SET_ZONE = 0xC2;
126static constexpr uint8_t HEADER_FOOTER_SIZE = 4;
128static constexpr uint8_t CMD_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xFD, 0xFC, 0xFB, 0xFA};
129static constexpr uint8_t CMD_FRAME_FOOTER[HEADER_FOOTER_SIZE] = {0x04, 0x03, 0x02, 0x01};
131static constexpr uint8_t DATA_FRAME_HEADER[HEADER_FOOTER_SIZE] = {0xAA, 0xFF, 0x03, 0x00};
132static constexpr uint8_t DATA_FRAME_FOOTER[2] = {0x55, 0xCC};
134static constexpr uint8_t NO_MAC[] = {0x08, 0x05, 0x04, 0x03, 0x02, 0x01};
136static inline uint16_t convert_seconds_to_ms(uint16_t value) {
return value * 1000; };
138static inline void convert_int_values_to_hex(
const int *values, uint8_t *bytes) {
139 for (uint8_t i = 0; i < 4; i++) {
140 uint16_t
val = values[i] & 0xFFFF;
142 bytes[i * 2 + 1] = (
val >> 8) & 0xFF;
146static inline int16_t decode_coordinate(uint8_t low_byte, uint8_t high_byte) {
147 int16_t coordinate = (high_byte & 0x7F) << 8 | low_byte;
148 if ((high_byte & 0x80) == 0) {
149 coordinate = -coordinate;
154static inline int16_t decode_speed(uint8_t low_byte, uint8_t high_byte) {
155 int16_t
speed = (high_byte & 0x7F) << 8 | low_byte;
156 if ((high_byte & 0x80) == 0) {
162static inline int16_t hex_to_signed_int(
const uint8_t *buffer, uint8_t offset) {
163 uint16_t hex_val = (buffer[offset + 1] << 8) | buffer[offset];
164 int16_t dec_val =
static_cast<int16_t
>(hex_val);
165 if (dec_val & 0x8000) {
171static inline bool validate_header_footer(
const uint8_t *header_footer,
const uint8_t *buffer) {
172 return std::memcmp(header_footer, buffer, HEADER_FOOTER_SIZE) == 0;
177 if (this->presence_timeout_number_ !=
nullptr) {
178 this->
pref_ = this->presence_timeout_number_->make_entity_preference<
float>();
179 this->set_presence_timeout();
182 this->restart_and_read_all_info();
185void LD2450Component::dump_config() {
192 " Firmware version: %s\n"
195#ifdef USE_BINARY_SENSOR
196 ESP_LOGCONFIG(TAG,
"Binary Sensors:");
197 LOG_BINARY_SENSOR(
" ",
"MovingTarget", this->moving_target_binary_sensor_);
198 LOG_BINARY_SENSOR(
" ",
"StillTarget", this->still_target_binary_sensor_);
199 LOG_BINARY_SENSOR(
" ",
"Target", this->target_binary_sensor_);
202 ESP_LOGCONFIG(TAG,
"Sensors:");
203 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"MovingTargetCount", this->moving_target_count_sensor_);
204 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"StillTargetCount", this->still_target_count_sensor_);
205 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetCount", this->target_count_sensor_);
207 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetX", s);
210 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetY", s);
213 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetAngle", s);
216 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetDistance", s);
219 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetResolution", s);
222 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"TargetSpeed", s);
225 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"ZoneTargetCount", s);
228 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"ZoneMovingTargetCount", s);
231 LOG_SENSOR_WITH_DEDUP_SAFE(
" ",
"ZoneStillTargetCount", s);
234#ifdef USE_TEXT_SENSOR
235 ESP_LOGCONFIG(TAG,
"Text Sensors:");
236 LOG_TEXT_SENSOR(
" ",
"Version", this->version_text_sensor_);
237 LOG_TEXT_SENSOR(
" ",
"MAC address", this->mac_text_sensor_);
239 LOG_TEXT_SENSOR(
" ",
"Direction", s);
243 ESP_LOGCONFIG(TAG,
"Numbers:");
244 LOG_NUMBER(
" ",
"PresenceTimeout", this->presence_timeout_number_);
246 LOG_NUMBER(
" ",
"ZoneX1", n.x1);
247 LOG_NUMBER(
" ",
"ZoneY1", n.y1);
248 LOG_NUMBER(
" ",
"ZoneX2", n.x2);
249 LOG_NUMBER(
" ",
"ZoneY2", n.y2);
253 ESP_LOGCONFIG(TAG,
"Selects:");
254 LOG_SELECT(
" ",
"BaudRate", this->baud_rate_select_);
255 LOG_SELECT(
" ",
"ZoneType", this->zone_type_select_);
258 ESP_LOGCONFIG(TAG,
"Switches:");
259 LOG_SWITCH(
" ",
"Bluetooth", this->bluetooth_switch_);
260 LOG_SWITCH(
" ",
"MultiTarget", this->multi_target_switch_);
263 ESP_LOGCONFIG(TAG,
"Buttons:");
264 LOG_BUTTON(
" ",
"FactoryReset", this->factory_reset_button_);
265 LOG_BUTTON(
" ",
"Restart", this->restart_button_);
269void LD2450Component::loop() {
272 uint8_t buf[MAX_LINE_LENGTH];
274 size_t to_read = std::min(avail,
sizeof(buf));
280 for (
size_t i = 0; i < to_read; i++) {
291 if (target.x > zone.
x1 && target.x < zone.
x2 && target.y > zone.
y1 && target.y < zone.
y2) {
292 if (target.is_moving) {
302void LD2450Component::reset_radar_zone() {
313void LD2450Component::set_radar_zone(int32_t zone_type, int32_t zone1_x1, int32_t zone1_y1, int32_t zone1_x2,
314 int32_t zone1_y2, int32_t zone2_x1, int32_t zone2_y1, int32_t zone2_x2,
315 int32_t zone2_y2, int32_t zone3_x1, int32_t zone3_y1, int32_t zone3_x2,
318 int zone_parameters[12] = {zone1_x1, zone1_y1, zone1_x2, zone1_y2, zone2_x1, zone2_y1,
319 zone2_x2, zone2_y2, zone3_x1, zone3_y1, zone3_x2, zone3_y2};
320 for (uint8_t i = 0; i < MAX_ZONES; i++) {
331 uint8_t cmd_value[26] = {};
332 uint8_t zone_type_bytes[2] = {
static_cast<uint8_t
>(this->
zone_type_), 0x00};
333 uint8_t area_config[24] = {};
334 for (uint8_t i = 0; i < MAX_ZONES; i++) {
337 ld2450::convert_int_values_to_hex(values, area_config + (i * 8));
339 std::memcpy(cmd_value, zone_type_bytes,
sizeof(zone_type_bytes));
340 std::memcpy(cmd_value + 2, area_config,
sizeof(area_config));
342 this->
send_command_(CMD_SET_ZONE, cmd_value,
sizeof(cmd_value));
348 if (check_millis == 0) {
352 this->
timeout_ = ld2450::convert_seconds_to_ms(DEFAULT_PRESENCE_TIMEOUT);
359 uint8_t index, start;
360 for (index = 0; index < MAX_ZONES; index++) {
361 start = 12 + index * 8;
379void LD2450Component::read_all_info() {
387 if (this->baud_rate_select_ !=
nullptr) {
389 this->baud_rate_select_->publish_state(*index);
392 this->publish_zone_type();
397void LD2450Component::query_zone_info() {
404void LD2450Component::restart_and_read_all_info() {
407 this->
set_timeout(1500, [
this]() { this->read_all_info(); });
410void LD2450Component::add_on_data_callback(std::function<
void()> &&callback) {
416 ESP_LOGV(TAG,
"Sending COMMAND %02X", command);
418 this->
write_array(CMD_FRAME_HEADER,
sizeof(CMD_FRAME_HEADER));
421 if (command_value !=
nullptr) {
422 len += command_value_len;
425 uint8_t len_cmd[] = {
len, 0x00, command, 0x00};
428 if (command_value !=
nullptr) {
429 this->
write_array(command_value, command_value_len);
432 this->
write_array(CMD_FRAME_FOOTER,
sizeof(CMD_FRAME_FOOTER));
434 if (command != CMD_ENABLE_CONF && command != CMD_DISABLE_CONF) {
444 ESP_LOGE(TAG,
"Invalid length");
447 if (!ld2450::validate_header_footer(DATA_FRAME_HEADER, this->
buffer_data_) ||
450 ESP_LOGE(TAG,
"Invalid header/footer");
454 int16_t target_count = 0;
455 int16_t still_target_count = 0;
456 int16_t moving_target_count = 0;
466 bool is_moving =
false;
468#if defined(USE_BINARY_SENSOR) || defined(USE_SENSOR) || defined(USE_TEXT_SENSOR)
470 for (index = 0; index < MAX_TARGETS; index++) {
489 moving_target_count++;
493 int32_t x_squared = (int32_t) tx * tx;
494 int32_t y_squared = (int32_t) ty * ty;
495 td = (uint16_t) sqrtf(x_squared + y_squared);
501 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_x_sensors_[index]);
502 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_y_sensors_[index]);
503 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_resolution_sensors_[index]);
504 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_speed_sensors_[index]);
505 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_distance_sensors_[index]);
506 SAFE_PUBLISH_SENSOR_UNKNOWN(this->move_angle_sensors_[index]);
508 SAFE_PUBLISH_SENSOR(this->move_x_sensors_[index], tx);
509 SAFE_PUBLISH_SENSOR(this->move_y_sensors_[index], ty);
510 SAFE_PUBLISH_SENSOR(this->move_resolution_sensors_[index], res);
511 SAFE_PUBLISH_SENSOR(this->move_speed_sensors_[index], ts);
512 SAFE_PUBLISH_SENSOR(this->move_distance_sensors_[index], td);
514 angle = atan2f(
static_cast<float>(-tx),
static_cast<float>(ty)) * (180.0f / std::numbers::pi_v<float>);
515 SAFE_PUBLISH_SENSOR(this->move_angle_sensors_[index], angle);
518#ifdef USE_TEXT_SENSOR
531 if (tsd !=
nullptr) {
544 still_target_count = target_count - moving_target_count;
549 uint8_t zone_still_targets = 0;
550 uint8_t zone_moving_targets = 0;
551 uint8_t zone_all_targets = 0;
552 for (index = 0; index < MAX_ZONES; index++) {
554 zone_all_targets = zone_still_targets + zone_moving_targets;
557 SAFE_PUBLISH_SENSOR(this->zone_still_target_count_sensors_[index], zone_still_targets);
559 SAFE_PUBLISH_SENSOR(this->zone_moving_target_count_sensors_[index], zone_moving_targets);
561 SAFE_PUBLISH_SENSOR(this->zone_target_count_sensors_[index], zone_all_targets);
565 SAFE_PUBLISH_SENSOR(this->target_count_sensor_, target_count);
567 SAFE_PUBLISH_SENSOR(this->still_target_count_sensor_, still_target_count);
569 SAFE_PUBLISH_SENSOR(this->moving_target_count_sensor_, moving_target_count);
572#ifdef USE_BINARY_SENSOR
574 if (this->target_binary_sensor_ !=
nullptr) {
575 if (target_count > 0) {
576 this->target_binary_sensor_->publish_state(
true);
579 this->target_binary_sensor_->publish_state(
false);
581 ESP_LOGV(TAG,
"Clear presence waiting timeout: %d", this->
timeout_);
586 if (this->moving_target_binary_sensor_ !=
nullptr) {
587 if (moving_target_count > 0) {
588 this->moving_target_binary_sensor_->publish_state(
true);
591 this->moving_target_binary_sensor_->publish_state(
false);
596 if (this->still_target_binary_sensor_ !=
nullptr) {
597 if (still_target_count > 0) {
598 this->still_target_binary_sensor_->publish_state(
true);
601 this->still_target_binary_sensor_->publish_state(
false);
608 if (target_count > 0) {
611 if (moving_target_count > 0) {
614 if (still_target_count > 0) {
625 ESP_LOGE(TAG,
"Invalid length");
628 if (!ld2450::validate_header_footer(CMD_FRAME_HEADER, this->
buffer_data_)) {
634 ESP_LOGE(TAG,
"Invalid status");
643 case CMD_ENABLE_CONF:
644 ESP_LOGV(TAG,
"Enable conf");
647 case CMD_DISABLE_CONF:
648 ESP_LOGV(TAG,
"Disabled conf");
651 case CMD_SET_BAUD_RATE:
652 ESP_LOGV(TAG,
"Baud rate change");
654 if (this->baud_rate_select_ !=
nullptr) {
655 auto baud = this->baud_rate_select_->current_option();
656 ESP_LOGE(TAG,
"Change baud rate to %.*s and reinstall", (
int) baud.size(), baud.c_str());
661 case CMD_QUERY_VERSION: {
665 ESP_LOGV(TAG,
"Firmware version: %s", version_s);
666#ifdef USE_TEXT_SENSOR
667 if (this->version_text_sensor_ !=
nullptr) {
668 this->version_text_sensor_->publish_state(version_s);
674 case CMD_QUERY_MAC_ADDRESS: {
686 ESP_LOGV(TAG,
"MAC address: %s", mac_str);
687#ifdef USE_TEXT_SENSOR
688 if (this->mac_text_sensor_ !=
nullptr) {
689 this->mac_text_sensor_->publish_state(mac_str);
693 if (this->bluetooth_switch_ !=
nullptr) {
701 ESP_LOGV(TAG,
"Bluetooth");
704 case CMD_SINGLE_TARGET_MODE:
705 ESP_LOGV(TAG,
"Single target conf");
707 if (this->multi_target_switch_ !=
nullptr) {
708 this->multi_target_switch_->publish_state(
false);
713 case CMD_MULTI_TARGET_MODE:
714 ESP_LOGV(TAG,
"Multi target conf");
716 if (this->multi_target_switch_ !=
nullptr) {
717 this->multi_target_switch_->publish_state(
true);
722 case CMD_QUERY_TARGET_MODE:
723 ESP_LOGV(TAG,
"Query target tracking mode");
725 if (this->multi_target_switch_ !=
nullptr) {
726 this->multi_target_switch_->publish_state(this->
buffer_data_[10] == 0x02);
732 ESP_LOGV(TAG,
"Query zone conf");
734 this->publish_zone_type();
736 if (this->zone_type_select_ !=
nullptr) {
737 auto zone = this->zone_type_select_->current_option();
738 ESP_LOGV(TAG,
"Change zone type to: %.*s", (
int) zone.size(), zone.c_str());
742 ESP_LOGV(TAG,
"Zone: Disabled");
745 ESP_LOGV(TAG,
"Zone: Area detection");
748 ESP_LOGV(TAG,
"Zone: Area filter");
754 ESP_LOGV(TAG,
"Set zone conf");
755 this->query_zone_info();
775 ESP_LOGW(TAG,
"Max command length exceeded; ignoring");
784#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
791#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
798 ESP_LOGV(TAG,
"Ack Data incomplete");
805 const uint8_t cmd = enable ? CMD_ENABLE_CONF : CMD_DISABLE_CONF;
806 const uint8_t cmd_value[2] = {0x01, 0x00};
807 this->
send_command_(cmd, enable ? cmd_value :
nullptr,
sizeof(cmd_value));
811void LD2450Component::set_bluetooth(
bool enable) {
813 const uint8_t cmd_value[2] = {enable ? (uint8_t) 0x01 : (uint8_t) 0x00, 0x00};
814 this->
send_command_(CMD_BLUETOOTH, cmd_value,
sizeof(cmd_value));
815 this->
set_timeout(200, [
this]() { this->restart_and_read_all_info(); });
819void LD2450Component::set_baud_rate(
const char *
state) {
822 this->
send_command_(CMD_SET_BAUD_RATE, cmd_value,
sizeof(cmd_value));
827void LD2450Component::set_zone_type(
const char *
state) {
828 ESP_LOGV(TAG,
"Set zone type: %s",
state);
835void LD2450Component::publish_zone_type() {
837 if (this->zone_type_select_ !=
nullptr) {
844void LD2450Component::set_multi_target(
bool enable) {
846 uint8_t cmd = enable ? CMD_MULTI_TARGET_MODE : CMD_SINGLE_TARGET_MODE;
852void LD2450Component::factory_reset() {
855 this->
set_timeout(200, [
this]() { this->restart_and_read_all_info(); });
866 uint8_t cmd_value[2] = {0x01, 0x00};
878void LD2450Component::set_move_x_sensor(uint8_t target,
sensor::Sensor *s) {
881void LD2450Component::set_move_y_sensor(uint8_t target,
sensor::Sensor *s) {
882 this->move_y_sensors_[target] =
new SensorWithDedup<int16_t>(s);
884void LD2450Component::set_move_speed_sensor(uint8_t target, sensor::Sensor *s) {
885 this->move_speed_sensors_[target] =
new SensorWithDedup<int16_t>(s);
887void LD2450Component::set_move_angle_sensor(uint8_t target, sensor::Sensor *s) {
888 this->move_angle_sensors_[target] =
new SensorWithDedup<float>(s);
890void LD2450Component::set_move_distance_sensor(uint8_t target, sensor::Sensor *s) {
891 this->move_distance_sensors_[target] =
new SensorWithDedup<uint16_t>(s);
893void LD2450Component::set_move_resolution_sensor(uint8_t target, sensor::Sensor *s) {
894 this->move_resolution_sensors_[target] =
new SensorWithDedup<uint16_t>(s);
896void LD2450Component::set_zone_target_count_sensor(uint8_t zone, sensor::Sensor *s) {
897 this->zone_target_count_sensors_[zone] =
new SensorWithDedup<uint8_t>(s);
899void LD2450Component::set_zone_still_target_count_sensor(uint8_t zone, sensor::Sensor *s) {
900 this->zone_still_target_count_sensors_[zone] =
new SensorWithDedup<uint8_t>(s);
902void LD2450Component::set_zone_moving_target_count_sensor(uint8_t zone, sensor::Sensor *s) {
903 this->zone_moving_target_count_sensors_[zone] =
new SensorWithDedup<uint8_t>(s);
906#ifdef USE_TEXT_SENSOR
907void LD2450Component::set_direction_text_sensor(uint8_t target, text_sensor::TextSensor *s) {
908 this->direction_text_sensors_[target] = s;
914void LD2450Component::set_zone_coordinate(uint8_t zone) {
919 if (!x1sens->has_state() || !y1sens->has_state() || !x2sens->has_state() || !y2sens->has_state()) {
929void LD2450Component::set_zone_numbers(uint8_t zone, number::Number *x1, number::Number *y1, number::Number *x2,
930 number::Number *y2) {
931 if (zone < MAX_ZONES) {
942void LD2450Component::set_presence_timeout() {
943 if (this->presence_timeout_number_ !=
nullptr) {
944 if (this->presence_timeout_number_->state == 0) {
946 this->presence_timeout_number_->publish_state(timeout);
947 this->
timeout_ = ld2450::convert_seconds_to_ms(timeout);
949 if (this->presence_timeout_number_->has_state()) {
951 this->
timeout_ = ld2450::convert_seconds_to_ms(this->presence_timeout_number_->state);
963 value = DEFAULT_PRESENCE_TIMEOUT;
uint32_t IRAM_ATTR HOT get_loop_component_start_time() const
Get the cached time in milliseconds from when the current component started its loop execution.
virtual void setup()
Where the component's initialization should happen.
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.
void save_to_flash_(float value)
std::array< SensorWithDedup< int16_t > *, MAX_TARGETS > move_speed_sensors_
std::array< SensorWithDedup< int16_t > *, MAX_TARGETS > move_x_sensors_
std::array< Deduplicator< uint8_t >, MAX_TARGETS > direction_dedup_
uint32_t still_presence_millis_
void send_command_(uint8_t command_str, const uint8_t *command_value, uint8_t command_value_len)
void query_target_tracking_mode_()
void send_set_zone_command_()
std::array< SensorWithDedup< float > *, MAX_TARGETS > move_angle_sensors_
void set_config_mode_(bool enable)
std::array< SensorWithDedup< uint8_t > *, MAX_ZONES > zone_still_target_count_sensors_
std::array< text_sensor::TextSensor *, MAX_TARGETS > direction_text_sensors_
void count_targets_in_zone_(const Zone &zone, uint8_t &still, uint8_t &moving)
uint32_t moving_presence_millis_
std::array< SensorWithDedup< uint16_t > *, MAX_TARGETS > move_resolution_sensors_
std::array< SensorWithDedup< uint8_t > *, MAX_ZONES > zone_target_count_sensors_
void readline_(int readch)
uint8_t buffer_data_[MAX_LINE_LENGTH]
std::array< SensorWithDedup< uint8_t > *, MAX_ZONES > zone_moving_target_count_sensors_
Target target_info_[MAX_TARGETS]
uint32_t presence_millis_
Zone zone_config_[MAX_ZONES]
std::array< SensorWithDedup< uint16_t > *, MAX_TARGETS > move_distance_sensors_
ESPPreferenceObject pref_
LazyCallbackManager< void()> data_callback_
std::array< SensorWithDedup< int16_t > *, MAX_TARGETS > move_y_sensors_
float restore_from_flash_()
ZoneOfNumbers zone_numbers_[MAX_ZONES]
void handle_periodic_data_()
bool get_timeout_status_(uint32_t check_millis)
void publish_state(float state)
Base-class for all sensors.
void publish_state(const std::string &state)
uint32_t get_baud_rate() const
optional< std::array< uint8_t, N > > read_array()
void write_array(const uint8_t *data, size_t len)
constexpr uint32_t BAUD_RATES[]
constexpr Uint8ToString DIRECTION_BY_UINT[]
constexpr StringToUint8 ZONE_TYPE_BY_STR[]
constexpr StringToUint8 BAUD_RATES_BY_STR[]
constexpr Uint8ToString ZONE_TYPE_BY_UINT[]
uint8_t find_uint8(const StringToUint8(&arr)[N], const std::string &str)
const char * find_str(const Uint8ToString(&arr)[N], uint8_t value)
void format_version_str(const uint8_t *version, std::span< char, 20 > buffer)
const char * format_mac_str(const uint8_t *mac_address, std::span< char, 18 > buffer)
optional< size_t > find_index(const uint32_t(&arr)[N], uint32_t value)
std::vector< uint8_t > bytes
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).
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".
void HOT delay(uint32_t ms)
Application App
Global storage of Application pointer - only one Application can exist.