mirror of
https://github.com/ruvnet/RuView.git
synced 2026-04-28 05:59:32 +00:00
571 lines
18 KiB
C
571 lines
18 KiB
C
/**
|
|
* @file mmwave_sensor.c
|
|
* @brief ADR-063: mmWave sensor UART driver with auto-detection.
|
|
*
|
|
* Supports Seeed MR60BHA2 (60 GHz) and HLK-LD2410 (24 GHz).
|
|
* Under QEMU (CONFIG_CSI_MOCK_ENABLED), uses a mock generator
|
|
* that produces synthetic vital signs for pipeline testing.
|
|
*
|
|
* MR60BHA2 frame format (Seeed mmWave protocol):
|
|
* [0] SOF = 0x01
|
|
* [1-2] Frame ID (uint16, big-endian)
|
|
* [3-4] Data Length (uint16, big-endian)
|
|
* [5-6] Frame Type (uint16, big-endian)
|
|
* [7] Header Checksum = ~XOR(bytes 0..6)
|
|
* [8..N] Payload (N = data_length)
|
|
* [N+1] Data Checksum = ~XOR(payload bytes)
|
|
*
|
|
* Frame types: 0x0A14=breathing, 0x0A15=heart rate,
|
|
* 0x0A16=distance, 0x0F09=presence
|
|
*
|
|
* LD2410 frame format (HLK binary, 256000 baud):
|
|
* Header: 0xF4 0xF3 0xF2 0xF1
|
|
* Length: uint16 LE
|
|
* Data: [type 0xAA] [target_state] [moving_dist LE] [energy] ...
|
|
* Footer: 0xF8 0xF7 0xF6 0xF5
|
|
*/
|
|
|
|
#include "mmwave_sensor.h"
|
|
|
|
#include <string.h>
|
|
#include <math.h>
|
|
#include "freertos/FreeRTOS.h"
|
|
#include "freertos/task.h"
|
|
#include "esp_log.h"
|
|
#include "esp_timer.h"
|
|
#include "sdkconfig.h"
|
|
|
|
#ifndef CONFIG_CSI_MOCK_ENABLED
|
|
#include "driver/uart.h"
|
|
#endif
|
|
|
|
static const char *TAG = "mmwave";
|
|
|
|
/* ---- Configuration ---- */
|
|
#define MMWAVE_UART_NUM UART_NUM_1
|
|
#define MMWAVE_MR60_BAUD 115200
|
|
#define MMWAVE_LD2410_BAUD 256000
|
|
#define MMWAVE_BUF_SIZE 256
|
|
#define MMWAVE_TASK_STACK 4096
|
|
#define MMWAVE_TASK_PRIORITY 3
|
|
#define MMWAVE_PROBE_TIMEOUT_MS 2000
|
|
#define MMWAVE_MR60_MAX_PAYLOAD 30 /* Sanity limit from Arduino lib */
|
|
|
|
/* ---- MR60BHA2 protocol constants (Seeed mmWave) ---- */
|
|
#define MR60_SOF 0x01
|
|
|
|
/* Frame types (big-endian uint16 at offset 5-6) */
|
|
#define MR60_TYPE_BREATHING 0x0A14
|
|
#define MR60_TYPE_HEARTRATE 0x0A15
|
|
#define MR60_TYPE_DISTANCE 0x0A16
|
|
#define MR60_TYPE_PRESENCE 0x0F09
|
|
#define MR60_TYPE_PHASE 0x0A13
|
|
#define MR60_TYPE_POINTCLOUD 0x0A04
|
|
|
|
/* ---- LD2410 protocol constants ---- */
|
|
#define LD2410_REPORT_HEAD 0xAA
|
|
#define LD2410_REPORT_TAIL 0x55
|
|
|
|
/* ---- Shared state ---- */
|
|
static mmwave_state_t s_state;
|
|
static volatile bool s_running;
|
|
|
|
/* ======================================================================
|
|
* MR60BHA2 Parser (corrected protocol from Seeed Arduino library)
|
|
* ====================================================================== */
|
|
|
|
static uint8_t mr60_calc_checksum(const uint8_t *data, uint16_t len)
|
|
{
|
|
uint8_t cksum = 0;
|
|
for (uint16_t i = 0; i < len; i++) {
|
|
cksum ^= data[i];
|
|
}
|
|
return ~cksum;
|
|
}
|
|
|
|
typedef enum {
|
|
MR60_WAIT_SOF,
|
|
MR60_READ_HEADER, /* Accumulate bytes 1..7 (frame_id, len, type, hdr_cksum) */
|
|
MR60_READ_DATA,
|
|
MR60_READ_DATA_CKSUM,
|
|
} mr60_parse_state_t;
|
|
|
|
typedef struct {
|
|
mr60_parse_state_t state;
|
|
uint8_t header[8]; /* Full header: SOF + frame_id(2) + len(2) + type(2) + hdr_cksum */
|
|
uint8_t hdr_idx;
|
|
uint16_t data_len;
|
|
uint16_t frame_type;
|
|
uint16_t data_idx;
|
|
uint8_t data[MMWAVE_BUF_SIZE];
|
|
} mr60_parser_t;
|
|
|
|
static mr60_parser_t s_mr60;
|
|
|
|
static void mr60_process_frame(uint16_t type, const uint8_t *data, uint16_t len)
|
|
{
|
|
s_state.frame_count++;
|
|
s_state.last_update_us = esp_timer_get_time();
|
|
|
|
switch (type) {
|
|
case MR60_TYPE_BREATHING:
|
|
if (len >= sizeof(float)) {
|
|
/* Breathing rate as float32 (little-endian in payload). */
|
|
float br;
|
|
memcpy(&br, data, sizeof(float));
|
|
if (br >= 0.0f && br <= 60.0f) {
|
|
s_state.breathing_rate = br;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case MR60_TYPE_HEARTRATE:
|
|
if (len >= sizeof(float)) {
|
|
float hr;
|
|
memcpy(&hr, data, sizeof(float));
|
|
if (hr >= 0.0f && hr <= 250.0f) {
|
|
s_state.heart_rate_bpm = hr;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case MR60_TYPE_DISTANCE:
|
|
if (len >= sizeof(uint32_t) + sizeof(float)) {
|
|
/* Bytes 0-3: range flag (uint32 LE). 0 = no valid distance. */
|
|
uint32_t range_flag;
|
|
memcpy(&range_flag, data, sizeof(uint32_t));
|
|
if (range_flag != 0) {
|
|
float dist;
|
|
memcpy(&dist, &data[sizeof(uint32_t)], sizeof(float));
|
|
s_state.distance_cm = dist;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case MR60_TYPE_PRESENCE:
|
|
if (len >= 1) {
|
|
s_state.person_present = (data[0] != 0);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void mr60_feed_byte(uint8_t b)
|
|
{
|
|
switch (s_mr60.state) {
|
|
case MR60_WAIT_SOF:
|
|
if (b == MR60_SOF) {
|
|
s_mr60.header[0] = b;
|
|
s_mr60.hdr_idx = 1;
|
|
s_mr60.state = MR60_READ_HEADER;
|
|
}
|
|
break;
|
|
|
|
case MR60_READ_HEADER:
|
|
s_mr60.header[s_mr60.hdr_idx++] = b;
|
|
if (s_mr60.hdr_idx >= 8) {
|
|
/* Validate header checksum: ~XOR(bytes 0..6) == byte 7 */
|
|
uint8_t expected = mr60_calc_checksum(s_mr60.header, 7);
|
|
if (expected != s_mr60.header[7]) {
|
|
s_state.error_count++;
|
|
s_mr60.state = MR60_WAIT_SOF;
|
|
break;
|
|
}
|
|
/* Parse header fields (big-endian) */
|
|
s_mr60.data_len = ((uint16_t)s_mr60.header[3] << 8) | s_mr60.header[4];
|
|
s_mr60.frame_type = ((uint16_t)s_mr60.header[5] << 8) | s_mr60.header[6];
|
|
s_mr60.data_idx = 0;
|
|
|
|
if (s_mr60.data_len > MMWAVE_MR60_MAX_PAYLOAD) {
|
|
s_state.error_count++;
|
|
s_mr60.state = MR60_WAIT_SOF;
|
|
} else if (s_mr60.data_len == 0) {
|
|
s_mr60.state = MR60_READ_DATA_CKSUM;
|
|
} else {
|
|
s_mr60.state = MR60_READ_DATA;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case MR60_READ_DATA:
|
|
s_mr60.data[s_mr60.data_idx++] = b;
|
|
if (s_mr60.data_idx >= s_mr60.data_len) {
|
|
s_mr60.state = MR60_READ_DATA_CKSUM;
|
|
}
|
|
break;
|
|
|
|
case MR60_READ_DATA_CKSUM:
|
|
/* Validate data checksum */
|
|
if (s_mr60.data_len > 0) {
|
|
uint8_t expected = mr60_calc_checksum(s_mr60.data, s_mr60.data_len);
|
|
if (expected == b) {
|
|
mr60_process_frame(s_mr60.frame_type, s_mr60.data, s_mr60.data_len);
|
|
} else {
|
|
s_state.error_count++;
|
|
}
|
|
} else {
|
|
/* Zero-length payload — checksum byte is for empty data */
|
|
mr60_process_frame(s_mr60.frame_type, s_mr60.data, 0);
|
|
}
|
|
s_mr60.state = MR60_WAIT_SOF;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* ======================================================================
|
|
* LD2410 Parser (HLK binary protocol, 256000 baud)
|
|
* ====================================================================== */
|
|
|
|
typedef enum {
|
|
LD_WAIT_F4, LD_WAIT_F3, LD_WAIT_F2, LD_WAIT_F1,
|
|
LD_READ_LEN_L, LD_READ_LEN_H,
|
|
LD_READ_DATA,
|
|
LD_WAIT_F8, LD_WAIT_F7, LD_WAIT_F6, LD_WAIT_F5,
|
|
} ld2410_parse_state_t;
|
|
|
|
typedef struct {
|
|
ld2410_parse_state_t state;
|
|
uint16_t data_len;
|
|
uint16_t data_idx;
|
|
uint8_t data[MMWAVE_BUF_SIZE];
|
|
} ld2410_parser_t;
|
|
|
|
static ld2410_parser_t s_ld;
|
|
|
|
static void ld2410_process_frame(const uint8_t *data, uint16_t len)
|
|
{
|
|
s_state.frame_count++;
|
|
s_state.last_update_us = esp_timer_get_time();
|
|
|
|
if (len < 12) return;
|
|
|
|
uint8_t data_type = data[0]; /* 0x02 = normal, 0x01 = engineering */
|
|
uint8_t head_marker = data[1]; /* Must be 0xAA */
|
|
|
|
if (head_marker != LD2410_REPORT_HEAD) return;
|
|
|
|
/* Normal mode target report (data_type 0x02 or 0x01) */
|
|
uint8_t target_state = data[2];
|
|
uint16_t moving_dist = data[3] | ((uint16_t)data[4] << 8);
|
|
uint8_t moving_energy = data[5];
|
|
uint16_t static_dist = data[6] | ((uint16_t)data[7] << 8);
|
|
uint8_t static_energy = data[8];
|
|
uint16_t detect_dist = data[9] | ((uint16_t)data[10] << 8);
|
|
|
|
(void)moving_energy;
|
|
(void)static_energy;
|
|
(void)detect_dist;
|
|
|
|
s_state.person_present = (target_state != 0);
|
|
s_state.target_count = (target_state != 0) ? 1 : 0;
|
|
|
|
if (target_state == 1 || target_state == 3) {
|
|
s_state.distance_cm = (float)moving_dist;
|
|
} else if (target_state == 2) {
|
|
s_state.distance_cm = (float)static_dist;
|
|
} else {
|
|
s_state.distance_cm = 0.0f;
|
|
}
|
|
}
|
|
|
|
static void ld2410_feed_byte(uint8_t b)
|
|
{
|
|
switch (s_ld.state) {
|
|
case LD_WAIT_F4: s_ld.state = (b == 0xF4) ? LD_WAIT_F3 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F3: s_ld.state = (b == 0xF3) ? LD_WAIT_F2 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F2: s_ld.state = (b == 0xF2) ? LD_WAIT_F1 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F1: s_ld.state = (b == 0xF1) ? LD_READ_LEN_L : LD_WAIT_F4; break;
|
|
case LD_READ_LEN_L:
|
|
s_ld.data_len = b;
|
|
s_ld.state = LD_READ_LEN_H;
|
|
break;
|
|
case LD_READ_LEN_H:
|
|
s_ld.data_len |= ((uint16_t)b << 8);
|
|
s_ld.data_idx = 0;
|
|
if (s_ld.data_len == 0 || s_ld.data_len > MMWAVE_BUF_SIZE) {
|
|
s_ld.state = LD_WAIT_F4;
|
|
} else {
|
|
s_ld.state = LD_READ_DATA;
|
|
}
|
|
break;
|
|
case LD_READ_DATA:
|
|
s_ld.data[s_ld.data_idx++] = b;
|
|
if (s_ld.data_idx >= s_ld.data_len) s_ld.state = LD_WAIT_F8;
|
|
break;
|
|
case LD_WAIT_F8: s_ld.state = (b == 0xF8) ? LD_WAIT_F7 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F7: s_ld.state = (b == 0xF7) ? LD_WAIT_F6 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F6: s_ld.state = (b == 0xF6) ? LD_WAIT_F5 : LD_WAIT_F4; break;
|
|
case LD_WAIT_F5:
|
|
if (b == 0xF5) {
|
|
ld2410_process_frame(s_ld.data, s_ld.data_len);
|
|
}
|
|
s_ld.state = LD_WAIT_F4;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* ======================================================================
|
|
* Mock mmWave Generator (for QEMU testing)
|
|
* ====================================================================== */
|
|
|
|
#ifdef CONFIG_CSI_MOCK_ENABLED
|
|
|
|
static void mock_mmwave_task(void *arg)
|
|
{
|
|
(void)arg;
|
|
ESP_LOGI(TAG, "Mock mmWave generator started (simulating MR60BHA2)");
|
|
|
|
s_state.type = MMWAVE_TYPE_MOCK;
|
|
s_state.detected = true;
|
|
s_state.capabilities = MMWAVE_CAP_HEART_RATE | MMWAVE_CAP_BREATHING
|
|
| MMWAVE_CAP_PRESENCE | MMWAVE_CAP_DISTANCE;
|
|
|
|
float hr_base = 72.0f;
|
|
float br_base = 16.0f;
|
|
uint32_t tick = 0;
|
|
|
|
while (s_running) {
|
|
tick++;
|
|
|
|
/* Simulate realistic vital sign variation. */
|
|
float hr_noise = 2.0f * sinf((float)tick * 0.1f) + 0.5f * sinf((float)tick * 0.37f);
|
|
float br_noise = 1.0f * sinf((float)tick * 0.07f) + 0.3f * sinf((float)tick * 0.23f);
|
|
|
|
s_state.heart_rate_bpm = hr_base + hr_noise;
|
|
s_state.breathing_rate = br_base + br_noise;
|
|
s_state.person_present = true;
|
|
s_state.distance_cm = 150.0f + 20.0f * sinf((float)tick * 0.05f);
|
|
s_state.target_count = 1;
|
|
s_state.frame_count++;
|
|
s_state.last_update_us = esp_timer_get_time();
|
|
|
|
/* Simulate person leaving at tick 200-250 (for scenario testing). */
|
|
if (tick >= 200 && tick <= 250) {
|
|
s_state.person_present = false;
|
|
s_state.heart_rate_bpm = 0.0f;
|
|
s_state.breathing_rate = 0.0f;
|
|
s_state.distance_cm = 0.0f;
|
|
s_state.target_count = 0;
|
|
}
|
|
|
|
/* ~1 Hz update rate (matches real MR60BHA2). */
|
|
vTaskDelay(pdMS_TO_TICKS(1000));
|
|
}
|
|
|
|
vTaskDelete(NULL);
|
|
}
|
|
|
|
#endif /* CONFIG_CSI_MOCK_ENABLED */
|
|
|
|
/* ======================================================================
|
|
* UART Auto-Detection and Task
|
|
* ====================================================================== */
|
|
|
|
#ifndef CONFIG_CSI_MOCK_ENABLED
|
|
|
|
/**
|
|
* Try to detect a sensor at the given baud rate.
|
|
* Returns the sensor type if detected, MMWAVE_TYPE_NONE otherwise.
|
|
*/
|
|
static mmwave_type_t probe_at_baud(uint32_t baud)
|
|
{
|
|
/* Reconfigure baud rate. */
|
|
uart_set_baudrate(MMWAVE_UART_NUM, baud);
|
|
uart_flush_input(MMWAVE_UART_NUM);
|
|
|
|
uint8_t buf[128];
|
|
int mr60_sof_seen = 0;
|
|
int ld2410_header_seen = 0;
|
|
|
|
int64_t deadline = esp_timer_get_time() + (int64_t)(MMWAVE_PROBE_TIMEOUT_MS / 2) * 1000;
|
|
|
|
while (esp_timer_get_time() < deadline) {
|
|
int len = uart_read_bytes(MMWAVE_UART_NUM, buf, sizeof(buf), pdMS_TO_TICKS(100));
|
|
if (len <= 0) continue;
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
/* MR60BHA2: SOF = 0x01, followed by valid-looking frame_id bytes */
|
|
if (buf[i] == MR60_SOF && baud == MMWAVE_MR60_BAUD) {
|
|
mr60_sof_seen++;
|
|
}
|
|
/* LD2410: 4-byte header 0xF4F3F2F1 */
|
|
if (i + 3 < len && buf[i] == 0xF4 && buf[i+1] == 0xF3
|
|
&& buf[i+2] == 0xF2 && buf[i+3] == 0xF1
|
|
&& baud == MMWAVE_LD2410_BAUD) {
|
|
ld2410_header_seen++;
|
|
}
|
|
}
|
|
|
|
if (mr60_sof_seen >= 3) return MMWAVE_TYPE_MR60BHA2;
|
|
if (ld2410_header_seen >= 2) return MMWAVE_TYPE_LD2410;
|
|
}
|
|
|
|
if (mr60_sof_seen > 0) return MMWAVE_TYPE_MR60BHA2;
|
|
if (ld2410_header_seen > 0) return MMWAVE_TYPE_LD2410;
|
|
|
|
return MMWAVE_TYPE_NONE;
|
|
}
|
|
|
|
/**
|
|
* Auto-detect sensor by probing at both baud rates.
|
|
* MR60BHA2 uses 115200, LD2410 uses 256000.
|
|
*/
|
|
static mmwave_type_t probe_sensor(void)
|
|
{
|
|
ESP_LOGI(TAG, "Probing at %d baud (MR60BHA2)...", MMWAVE_MR60_BAUD);
|
|
mmwave_type_t result = probe_at_baud(MMWAVE_MR60_BAUD);
|
|
if (result != MMWAVE_TYPE_NONE) return result;
|
|
|
|
ESP_LOGI(TAG, "Probing at %d baud (LD2410)...", MMWAVE_LD2410_BAUD);
|
|
result = probe_at_baud(MMWAVE_LD2410_BAUD);
|
|
return result;
|
|
}
|
|
|
|
static void mmwave_uart_task(void *arg)
|
|
{
|
|
(void)arg;
|
|
ESP_LOGI(TAG, "mmWave UART task started (type=%s)",
|
|
mmwave_type_name(s_state.type));
|
|
|
|
uint8_t buf[128];
|
|
|
|
while (s_running) {
|
|
int len = uart_read_bytes(MMWAVE_UART_NUM, buf, sizeof(buf), pdMS_TO_TICKS(100));
|
|
if (len <= 0) {
|
|
vTaskDelay(1);
|
|
continue;
|
|
}
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
if (s_state.type == MMWAVE_TYPE_MR60BHA2) {
|
|
mr60_feed_byte(buf[i]);
|
|
} else if (s_state.type == MMWAVE_TYPE_LD2410) {
|
|
ld2410_feed_byte(buf[i]);
|
|
}
|
|
}
|
|
|
|
vTaskDelay(1);
|
|
}
|
|
|
|
vTaskDelete(NULL);
|
|
}
|
|
|
|
#endif /* !CONFIG_CSI_MOCK_ENABLED */
|
|
|
|
/* ======================================================================
|
|
* Public API
|
|
* ====================================================================== */
|
|
|
|
const char *mmwave_type_name(mmwave_type_t type)
|
|
{
|
|
switch (type) {
|
|
case MMWAVE_TYPE_MR60BHA2: return "MR60BHA2";
|
|
case MMWAVE_TYPE_LD2410: return "LD2410";
|
|
case MMWAVE_TYPE_MOCK: return "Mock";
|
|
case MMWAVE_TYPE_NONE:
|
|
default: return "None";
|
|
}
|
|
}
|
|
|
|
esp_err_t mmwave_sensor_init(int uart_tx_pin, int uart_rx_pin)
|
|
{
|
|
memset(&s_state, 0, sizeof(s_state));
|
|
memset(&s_mr60, 0, sizeof(s_mr60));
|
|
memset(&s_ld, 0, sizeof(s_ld));
|
|
s_running = true;
|
|
|
|
#ifdef CONFIG_CSI_MOCK_ENABLED
|
|
ESP_LOGI(TAG, "Mock mode: starting synthetic mmWave generator");
|
|
|
|
BaseType_t ret = xTaskCreatePinnedToCore(
|
|
mock_mmwave_task, "mmwave_mock", MMWAVE_TASK_STACK,
|
|
NULL, MMWAVE_TASK_PRIORITY, NULL, 0);
|
|
|
|
if (ret != pdPASS) {
|
|
ESP_LOGE(TAG, "Failed to create mock mmWave task");
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
return ESP_OK;
|
|
|
|
#else
|
|
if (uart_tx_pin < 0) uart_tx_pin = 17;
|
|
if (uart_rx_pin < 0) uart_rx_pin = 18;
|
|
|
|
/* Install UART driver at MR60 baud (will be changed during probe). */
|
|
uart_config_t uart_config = {
|
|
.baud_rate = MMWAVE_MR60_BAUD,
|
|
.data_bits = UART_DATA_8_BITS,
|
|
.parity = UART_PARITY_DISABLE,
|
|
.stop_bits = UART_STOP_BITS_1,
|
|
.flow_ctrl = UART_HW_FLOWCTRL_DISABLE,
|
|
.source_clk = UART_SCLK_DEFAULT,
|
|
};
|
|
|
|
esp_err_t err = uart_driver_install(MMWAVE_UART_NUM, MMWAVE_BUF_SIZE * 2, 0, 0, NULL, 0);
|
|
if (err != ESP_OK) {
|
|
ESP_LOGE(TAG, "UART driver install failed: %s", esp_err_to_name(err));
|
|
return err;
|
|
}
|
|
|
|
uart_param_config(MMWAVE_UART_NUM, &uart_config);
|
|
uart_set_pin(MMWAVE_UART_NUM, uart_tx_pin, uart_rx_pin,
|
|
UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
|
|
|
|
ESP_LOGI(TAG, "Probing UART%d (TX=%d, RX=%d) for mmWave sensor...",
|
|
MMWAVE_UART_NUM, uart_tx_pin, uart_rx_pin);
|
|
|
|
mmwave_type_t detected = probe_sensor();
|
|
|
|
if (detected == MMWAVE_TYPE_NONE) {
|
|
ESP_LOGI(TAG, "No mmWave sensor detected on UART%d", MMWAVE_UART_NUM);
|
|
uart_driver_delete(MMWAVE_UART_NUM);
|
|
return ESP_ERR_NOT_FOUND;
|
|
}
|
|
|
|
/* Set final baud rate for the detected sensor. */
|
|
uint32_t final_baud = (detected == MMWAVE_TYPE_LD2410)
|
|
? MMWAVE_LD2410_BAUD : MMWAVE_MR60_BAUD;
|
|
uart_set_baudrate(MMWAVE_UART_NUM, final_baud);
|
|
|
|
s_state.type = detected;
|
|
s_state.detected = true;
|
|
|
|
switch (detected) {
|
|
case MMWAVE_TYPE_MR60BHA2:
|
|
s_state.capabilities = MMWAVE_CAP_HEART_RATE | MMWAVE_CAP_BREATHING
|
|
| MMWAVE_CAP_PRESENCE | MMWAVE_CAP_DISTANCE;
|
|
break;
|
|
case MMWAVE_TYPE_LD2410:
|
|
s_state.capabilities = MMWAVE_CAP_PRESENCE | MMWAVE_CAP_DISTANCE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ESP_LOGI(TAG, "Detected %s at %lu baud (caps=0x%04x)",
|
|
mmwave_type_name(detected), (unsigned long)final_baud,
|
|
s_state.capabilities);
|
|
|
|
BaseType_t ret = xTaskCreatePinnedToCore(
|
|
mmwave_uart_task, "mmwave_uart", MMWAVE_TASK_STACK,
|
|
NULL, MMWAVE_TASK_PRIORITY, NULL, 0);
|
|
|
|
if (ret != pdPASS) {
|
|
ESP_LOGE(TAG, "Failed to create mmWave UART task");
|
|
return ESP_ERR_NO_MEM;
|
|
}
|
|
|
|
return ESP_OK;
|
|
#endif
|
|
}
|
|
|
|
bool mmwave_sensor_get_state(mmwave_state_t *state)
|
|
{
|
|
if (!s_state.detected || state == NULL) return false;
|
|
memcpy(state, &s_state, sizeof(mmwave_state_t));
|
|
return true;
|
|
}
|