powerpods/main/led_ring.c
simon eb67a46158 Add LED ring control per client and broadcast over REST and WebSocket.
Solid color mode fills all ring LEDs; master routes UART commands to slaves
via ESPNOW_LED_RING. goTool exposes POST /api/led-ring, WebSocket set_led_ring,
and a dashboard LED panel with master/slave/all targets.

Co-authored-by: Cursor <cursoragent@cursor.com>
2026-05-29 19:24:55 +02:00

227 lines
7.2 KiB
C

#include "led_ring.h"
#include "driver/i2c_master.h"
#include "driver/i2c_types.h"
#include "esp_err.h"
#include "esp_log.h"
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "led_strip.h"
#include <stdint.h>
typedef struct {
const uint8_t *leds;
uint8_t count;
} digit_definition_t;
static const char *TAG = "[LED_RING]";
static led_strip_handle_t led_ring;
#define RING_LEDS 95
#define LED_RING_PIN 7
#define LED_RING_BLINK_ON_MS 350
#define LED_RING_BLINK_OFF_MS 150
#define LED_RING_FIND_ME_ON_MS 300
#define LED_RING_FIND_ME_OFF_MS 150
#define LED_RING_FIND_ME_BLINKS_PER_COLOR 3
static QueueHandle_t led_queue;
// Led Matrix Maps
const uint8_t d0[] = {46, 47, 60, 61, 62, 75, 78, 79,
80, 81, 82, 86, 87, 88, 89, 90};
const uint8_t d1[] = {23, 46, 47, 48, 61, 62, 74, 75, 76, 84, 86, 93, 95};
const uint8_t d2[] = {21, 22, 23, 24, 25, 26, 46, 47, 48, 49, 59,
64, 71, 72, 73, 74, 75, 83, 89, 92, 95};
const uint8_t d3[] = {1, 2, 21, 22, 23, 24, 25, 26, 27, 41, 42,
43, 44, 45, 48, 59, 77, 83, 92, 93, 95};
const uint8_t d4[] = {21, 26, 47, 59, 64, 77, 82, 86, 94, 95};
const uint8_t d5[] = {19, 20, 21, 22, 23, 24, 25, 26, 63, 76, 77,
78, 79, 80, 81, 82, 83, 84, 85, 90, 91};
const uint8_t d6[] = {19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 65, 76, 77, 78,
79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91};
const uint8_t d7[] = {1, 47, 58, 59, 60, 61, 62, 63,
64, 65, 77, 80, 82, 92, 94};
const uint8_t d8[] = {1, 2, 3, 4, 5, 19, 20, 21, 22, 23, 24,
25, 26, 27, 41, 42, 43, 44, 45, 49, 58, 64,
73, 78, 82, 86, 90, 92, 93, 94, 95};
const uint8_t d9[] = {19, 20, 21, 22, 23, 24, 25, 26, 27, 46, 47, 58,
64, 71, 72, 73, 74, 75, 77, 82, 86, 94, 95};
const uint8_t d10[] = {46, 50, 57, 61, 65, 72, 76, 78, 80,
82, 84, 86, 88, 90, 92, 93, 94, 95};
const digit_definition_t digit_lookup[] = {
{d0, sizeof(d0)}, {d1, sizeof(d1)}, {d2, sizeof(d2)}, {d3, sizeof(d3)},
{d4, sizeof(d4)}, {d5, sizeof(d5)}, {d6, sizeof(d6)}, {d7, sizeof(d7)},
{d8, sizeof(d8)}, {d9, sizeof(d9)}, {d10, sizeof(d10)}};
void led_ring_scale_rgb(uint8_t *r, uint8_t *g, uint8_t *b, uint8_t intensity) {
if (intensity == 0) {
intensity = LED_RING_DEFAULT_INTENSITY;
}
*r = (uint16_t)(*r) * intensity / 255;
*g = (uint16_t)(*g) * intensity / 255;
*b = (uint16_t)(*b) * intensity / 255;
}
static void ring_fill_color(uint8_t r, uint8_t g, uint8_t b) {
for (uint32_t i = 0; i < RING_LEDS; i++) {
led_strip_set_pixel(led_ring, i, r, g, b);
}
}
static void ring_blink_scaled(uint8_t r, uint8_t g, uint8_t b, uint8_t intensity,
uint8_t count, uint16_t on_ms, uint16_t off_ms) {
led_ring_scale_rgb(&r, &g, &b, intensity);
for (uint8_t n = 0; n < count; n++) {
ring_fill_color(r, g, b);
led_strip_refresh(led_ring);
vTaskDelay(pdMS_TO_TICKS(on_ms));
led_strip_clear(led_ring);
led_strip_refresh(led_ring);
if (n + 1 < count) {
vTaskDelay(pdMS_TO_TICKS(off_ms));
}
}
}
void vTaskLedRing(void *pvParameters) {
led_strip_config_t ring_config = {
.strip_gpio_num = LED_RING_PIN,
.max_leds = RING_LEDS,
};
led_strip_rmt_config_t rmt_ring_config = {
.resolution_hz = 10 * 1000 * 1000,
};
esp_err_t err =
led_strip_new_rmt_device(&ring_config, &rmt_ring_config, &led_ring);
if (err == ESP_OK) {
ESP_LOGI(TAG, "GPIO_RING_TASK started");
}
led_command_t cmd;
while (1) {
if (xQueueReceive(led_queue, &cmd, portMAX_DELAY)) {
uint8_t r = cmd.r;
uint8_t g = cmd.g;
uint8_t b = cmd.b;
led_ring_scale_rgb(&r, &g, &b, cmd.intensity);
led_strip_clear(led_ring);
if (cmd.mode == LED_CMD_CLEAR) {
/* ring already cleared */
} else if (cmd.mode == LED_CMD_SET_DIGIT && cmd.value <= 10) {
digit_definition_t digit = digit_lookup[cmd.value];
for (int i = 0; i < digit.count; i++) {
led_strip_set_pixel(led_ring, RING_LEDS - digit.leds[i], r, g, b);
}
} else if (cmd.mode == LED_CMD_SET_COLOR) {
ring_fill_color(r, g, b);
} else if (cmd.mode == LED_CMD_PROGRESS) {
uint32_t lit = ((uint32_t)cmd.progress * RING_LEDS + 50) / 100;
if (lit > RING_LEDS) {
lit = RING_LEDS;
}
for (uint32_t i = 0; i < lit; i++) {
led_strip_set_pixel(led_ring, i, r, g, b);
}
} else if (cmd.mode == LED_CMD_BLINK) {
uint16_t on_ms = cmd.blink_ms > 0 ? cmd.blink_ms : LED_RING_BLINK_ON_MS;
uint8_t count = cmd.blink_count > 0 ? cmd.blink_count : 1;
ring_blink_scaled(cmd.r, cmd.g, cmd.b, cmd.intensity, count, on_ms,
LED_RING_BLINK_OFF_MS);
continue;
} else if (cmd.mode == LED_CMD_FIND_ME) {
static const struct {
uint8_t r, g, b;
} colors[] = {{255, 0, 0}, {0, 255, 0}, {0, 0, 255}};
for (size_t c = 0; c < sizeof(colors) / sizeof(colors[0]); c++) {
ring_blink_scaled(colors[c].r, colors[c].g, colors[c].b,
LED_RING_FULL_INTENSITY, LED_RING_FIND_ME_BLINKS_PER_COLOR,
LED_RING_FIND_ME_ON_MS, LED_RING_FIND_ME_OFF_MS);
if (c + 1 < sizeof(colors) / sizeof(colors[0])) {
vTaskDelay(pdMS_TO_TICKS(LED_RING_FIND_ME_OFF_MS));
}
}
continue;
}
led_strip_refresh(led_ring);
}
}
}
void led_ring_init(void) {
led_queue = xQueueCreate(10, sizeof(led_command_t));
xTaskCreate(vTaskLedRing, "led_task", 4096, NULL, 5, NULL);
}
void led_ring_send_command(led_command_t *cmd) {
if (led_queue != NULL) {
xQueueSend(led_queue, cmd, portMAX_DELAY);
}
}
void led_ring_show_ota_clear(void) {
led_command_t cmd = {.mode = LED_CMD_CLEAR};
led_ring_send_command(&cmd);
}
void led_ring_show_ota_progress(uint32_t bytes_done, uint32_t total_bytes,
uint8_t r, uint8_t g, uint8_t b) {
static struct {
uint8_t pct;
uint8_t r, g, b;
} last = {255, 0, 0, 0};
if (total_bytes == 0) {
return;
}
uint32_t pct32 = (bytes_done * 100u + total_bytes / 2) / total_bytes;
if (pct32 > 100) {
pct32 = 100;
}
uint8_t pct = (uint8_t)pct32;
if (pct == last.pct && r == last.r && g == last.g && b == last.b) {
return;
}
last.pct = pct;
last.r = r;
last.g = g;
last.b = b;
led_command_t cmd = {
.mode = LED_CMD_PROGRESS,
.progress = pct,
.r = r,
.g = g,
.b = b,
.intensity = LED_RING_DEFAULT_INTENSITY,
};
led_ring_send_command(&cmd);
}
void led_ring_blink_once(uint8_t r, uint8_t g, uint8_t b) {
led_command_t cmd = {
.mode = LED_CMD_BLINK,
.r = r,
.g = g,
.b = b,
.intensity = LED_RING_DEFAULT_INTENSITY,
.blink_ms = LED_RING_BLINK_ON_MS,
.blink_count = 1,
};
led_ring_send_command(&cmd);
}
void led_ring_ota_success(void) { led_ring_blink_once(0, 255, 0); }
void led_ring_ota_failed(void) { led_ring_blink_once(255, 0, 0); }
void led_ring_find_me(void) {
led_command_t cmd = {.mode = LED_CMD_FIND_ME};
led_ring_send_command(&cmd);
}