8 namespace seeed_mr60bha2 {
10 static const char *
const TAG =
"seeed_mr60bha2";
15 ESP_LOGCONFIG(TAG,
"MR60BHA2:");
16 #ifdef USE_BINARY_SENSOR 17 LOG_BINARY_SENSOR(
" ",
"People Exist Binary Sensor", this->has_target_binary_sensor_);
20 LOG_SENSOR(
" ",
"Breath Rate Sensor", this->breath_rate_sensor_);
21 LOG_SENSOR(
" ",
"Heart Rate Sensor", this->heart_rate_sensor_);
22 LOG_SENSOR(
" ",
"Distance Sensor", this->distance_sensor_);
23 LOG_SENSOR(
" ",
"Target Number Sensor", this->num_targets_sensor_);
51 static uint8_t calculate_checksum(
const uint8_t *data,
size_t len) {
53 for (
size_t i = 0; i <
len; i++) {
71 static bool validate_checksum(
const uint8_t *data,
size_t len, uint8_t expected_checksum) {
72 return calculate_checksum(data, len) == expected_checksum;
78 uint8_t new_byte = data[at];
81 return new_byte == FRAME_HEADER_BUFFER;
101 if (frame_type != BREATH_RATE_TYPE_BUFFER && frame_type != HEART_RATE_TYPE_BUFFER &&
102 frame_type != DISTANCE_TYPE_BUFFER && frame_type != PEOPLE_EXIST_TYPE_BUFFER &&
103 frame_type != PRINT_CLOUD_BUFFER) {
107 uint8_t header_checksum = new_byte;
110 if (!validate_checksum(data, 7, header_checksum)) {
111 ESP_LOGE(TAG,
"HEAD_CKSUM_FRAME ERROR: 0x%02x", header_checksum);
119 if (at - 8 < length) {
123 uint8_t data_checksum = new_byte;
124 if (at == 8 + length) {
125 if (!validate_checksum(data + 8, length, data_checksum)) {
126 ESP_LOGE(TAG,
"DATA_CKSUM_FRAME ERROR: 0x%02x", data_checksum);
132 const uint8_t *frame_data = data + 8;
133 ESP_LOGV(TAG,
"Received Frame: ID: 0x%04x, Type: 0x%04x, Data: [%s] Raw Data: [%s]", frame_id, frame_type,
142 switch (frame_type) {
143 case BREATH_RATE_TYPE_BUFFER:
144 if (this->breath_rate_sensor_ !=
nullptr && length >= 4) {
145 uint32_t current_breath_rate_int =
encode_uint32(data[3], data[2], data[1], data[0]);
146 if (current_breath_rate_int != 0) {
147 float breath_rate_float;
148 memcpy(&breath_rate_float, ¤t_breath_rate_int,
sizeof(
float));
149 this->breath_rate_sensor_->publish_state(breath_rate_float);
153 case PEOPLE_EXIST_TYPE_BUFFER:
154 if (this->has_target_binary_sensor_ !=
nullptr && length >= 2) {
156 this->has_target_binary_sensor_->publish_state(has_target_int);
157 if (has_target_int == 0) {
158 this->breath_rate_sensor_->publish_state(0.0);
159 this->heart_rate_sensor_->publish_state(0.0);
160 this->distance_sensor_->publish_state(0.0);
161 this->num_targets_sensor_->publish_state(0);
165 case HEART_RATE_TYPE_BUFFER:
166 if (this->heart_rate_sensor_ !=
nullptr && length >= 4) {
167 uint32_t current_heart_rate_int =
encode_uint32(data[3], data[2], data[1], data[0]);
168 if (current_heart_rate_int != 0) {
169 float heart_rate_float;
170 memcpy(&heart_rate_float, ¤t_heart_rate_int,
sizeof(
float));
171 this->heart_rate_sensor_->publish_state(heart_rate_float);
175 case DISTANCE_TYPE_BUFFER:
177 if (this->distance_sensor_ !=
nullptr && length >= 8) {
178 uint32_t current_distance_int =
encode_uint32(data[7], data[6], data[5], data[4]);
179 float distance_float;
180 memcpy(&distance_float, ¤t_distance_int,
sizeof(
float));
181 this->distance_sensor_->publish_state(distance_float);
185 case PRINT_CLOUD_BUFFER:
186 if (this->num_targets_sensor_ !=
nullptr && length >= 4) {
187 uint32_t current_num_targets_int =
encode_uint32(data[3], data[2], data[1], data[0]);
188 this->num_targets_sensor_->publish_state(current_num_targets_int);
std::string format_hex_pretty(const uint8_t *data, size_t length)
Format the byte array data of length len in pretty-printed, human-readable hex.
void process_frame_(uint16_t frame_id, uint16_t frame_type, const uint8_t *data, size_t length)
std::vector< uint8_t > rx_message_
constexpr uint32_t encode_uint32(uint8_t byte1, uint8_t byte2, uint8_t byte3, uint8_t byte4)
Encode a 32-bit value given four bytes in most to least significant byte order.
void dump_config() override
bool read_byte(uint8_t *data)
constexpr uint16_t encode_uint16(uint8_t msb, uint8_t lsb)
Encode a 16-bit value given the most and least significant byte.
Implementation of SPI Controller mode.