FastLED 3.9.13
Loading...
Searching...
No Matches
led_strip_rmt_dev.c
1
2
3#ifdef ESP32
4
5#include "enabled.h"
6
7#if FASTLED_RMT5
8
9
10/*
11 * SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD
12 *
13 * SPDX-License-Identifier: Apache-2.0
14 */
15#include <stdlib.h>
16#include <string.h>
17#include <sys/cdefs.h>
18#include "esp_log.h"
19#include "esp_check.h"
20#include "driver/rmt_tx.h"
21#include "led_strip.h"
22#include "led_strip_interface.h"
23#include "led_strip_rmt_encoder.h"
24
25#define LED_STRIP_RMT_DEFAULT_RESOLUTION 10000000 // 10MHz resolution
26#define LED_STRIP_RMT_DEFAULT_TRANS_QUEUE_SIZE 4
27
28#ifndef LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS_MULTIPLIER
29#if CONFIG_IDF_TARGET_ESP32C3
30// Fixes https://github.com/FastLED/FastLED/issues/1846
31#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS_MULTIPLIER 2
32#else
33#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS_MULTIPLIER 1
34#endif
35#endif
36
37// the memory size of each RMT channel, in words (4 bytes)
38#ifndef LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS
39#if CONFIG_IDF_TARGET_ESP32 || CONFIG_IDF_TARGET_ESP32S2
40#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS (LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS_MULTIPLIER * 64)
41#else
42#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS (LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS_MULTIPLIER * 48)
43#endif
44#endif
45
46
47static const char *TAG = "led_strip_rmt";
48
49typedef struct {
50 led_strip_t base;
51 rmt_channel_handle_t rmt_chan;
52 rmt_encoder_handle_t strip_encoder;
53 uint32_t strip_len;
54 uint8_t bytes_per_pixel;
55 led_color_component_format_t component_fmt;
56 uint8_t pixel_buf[];
57} led_strip_rmt_obj;
58
59static esp_err_t led_strip_rmt_set_pixel(led_strip_t *strip, uint32_t index, uint32_t red, uint32_t green, uint32_t blue)
60{
61 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
62 ESP_RETURN_ON_FALSE(index < rmt_strip->strip_len, ESP_ERR_INVALID_ARG, TAG, "index out of maximum number of LEDs");
63
64 led_color_component_format_t component_fmt = rmt_strip->component_fmt;
65 uint32_t start = index * rmt_strip->bytes_per_pixel;
66 uint8_t *pixel_buf = rmt_strip->pixel_buf;
67
68 pixel_buf[start + component_fmt.format.r_pos] = red & 0xFF;
69 pixel_buf[start + component_fmt.format.g_pos] = green & 0xFF;
70 pixel_buf[start + component_fmt.format.b_pos] = blue & 0xFF;
71 if (component_fmt.format.num_components > 3) {
72 pixel_buf[start + component_fmt.format.w_pos] = 0;
73 }
74
75 return ESP_OK;
76}
77
78static esp_err_t led_strip_rmt_set_pixel_rgbw(led_strip_t *strip, uint32_t index, uint32_t red, uint32_t green, uint32_t blue, uint32_t white)
79{
80 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
81 led_color_component_format_t component_fmt = rmt_strip->component_fmt;
82 ESP_RETURN_ON_FALSE(index < rmt_strip->strip_len, ESP_ERR_INVALID_ARG, TAG, "index out of maximum number of LEDs");
83 ESP_RETURN_ON_FALSE(component_fmt.format.num_components == 4, ESP_ERR_INVALID_ARG, TAG, "led doesn't have 4 components");
84
85 uint32_t start = index * rmt_strip->bytes_per_pixel;
86 uint8_t *pixel_buf = rmt_strip->pixel_buf;
87
88 pixel_buf[start + component_fmt.format.r_pos] = red & 0xFF;
89 pixel_buf[start + component_fmt.format.g_pos] = green & 0xFF;
90 pixel_buf[start + component_fmt.format.b_pos] = blue & 0xFF;
91 pixel_buf[start + component_fmt.format.w_pos] = white & 0xFF;
92
93 return ESP_OK;
94}
95
96static esp_err_t led_strip_rmt_refresh_async(led_strip_t *strip)
97{
98 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
99 rmt_transmit_config_t tx_conf = {
100 .loop_count = 0,
101 };
102
103 ESP_RETURN_ON_ERROR(rmt_enable(rmt_strip->rmt_chan), TAG, "enable RMT channel failed");
104 ESP_RETURN_ON_ERROR(rmt_transmit(rmt_strip->rmt_chan, rmt_strip->strip_encoder, rmt_strip->pixel_buf,
105 rmt_strip->strip_len * rmt_strip->bytes_per_pixel, &tx_conf), TAG, "transmit pixels by RMT failed");
106 return ESP_OK;
107}
108
109static esp_err_t led_strip_rmt_wait_for_done(led_strip_t *strip)
110{
111 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
112 ESP_RETURN_ON_ERROR(rmt_tx_wait_all_done(rmt_strip->rmt_chan, -1), TAG, "wait for RMT done failed");
113 ESP_RETURN_ON_ERROR(rmt_disable(rmt_strip->rmt_chan), TAG, "disable RMT channel failed");
114 return ESP_OK;
115}
116
117static esp_err_t led_strip_rmt_refresh(led_strip_t *strip)
118{
119 ESP_RETURN_ON_ERROR(led_strip_rmt_refresh_async(strip), TAG, "refresh async failed");
120 ESP_RETURN_ON_ERROR(led_strip_rmt_wait_for_done(strip), TAG, "wait for done failed");
121 return ESP_OK;
122}
123
124static esp_err_t led_strip_rmt_clear(led_strip_t *strip)
125{
126 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
127 // Write zero to turn off all leds
128 memset(rmt_strip->pixel_buf, 0, rmt_strip->strip_len * rmt_strip->bytes_per_pixel);
129 return led_strip_rmt_refresh(strip);
130}
131
132static esp_err_t led_strip_rmt_del(led_strip_t *strip)
133{
134 led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base);
135 ESP_RETURN_ON_ERROR(rmt_del_channel(rmt_strip->rmt_chan), TAG, "delete RMT channel failed");
136 ESP_RETURN_ON_ERROR(rmt_del_encoder(rmt_strip->strip_encoder), TAG, "delete strip encoder failed");
137 free(rmt_strip);
138 return ESP_OK;
139}
140
141esp_err_t led_strip_new_rmt_device(const led_strip_config_t *led_config, const led_strip_rmt_config_t *rmt_config, led_strip_handle_t *ret_strip)
142{
143 led_strip_rmt_obj *rmt_strip = NULL;
144 esp_err_t ret = ESP_OK;
145 ESP_GOTO_ON_FALSE(led_config && rmt_config && ret_strip, ESP_ERR_INVALID_ARG, err, TAG, "invalid argument");
146 led_color_component_format_t component_fmt = led_config->color_component_format;
147 // If R/G/B order is not specified, set default GRB order as fallback
148 if (component_fmt.format_id == 0) {
149 component_fmt = LED_STRIP_COLOR_COMPONENT_FMT_GRB;
150 }
151 // check the validation of the color component format
152 uint8_t mask = 0;
153 if (component_fmt.format.num_components == 3) {
154 mask = BIT(component_fmt.format.r_pos) | BIT(component_fmt.format.g_pos) | BIT(component_fmt.format.b_pos);
155 // Check for invalid values
156 ESP_RETURN_ON_FALSE(mask == 0x07, ESP_ERR_INVALID_ARG, TAG, "invalid order argument");
157 } else if (component_fmt.format.num_components == 4) {
158 mask = BIT(component_fmt.format.r_pos) | BIT(component_fmt.format.g_pos) | BIT(component_fmt.format.b_pos) | BIT(component_fmt.format.w_pos);
159 // Check for invalid values
160 ESP_RETURN_ON_FALSE(mask == 0x0F, ESP_ERR_INVALID_ARG, TAG, "invalid order argument");
161 } else {
162 ESP_RETURN_ON_FALSE(false, ESP_ERR_INVALID_ARG, TAG, "invalid number of color components: %d", component_fmt.format.num_components);
163 }
164 // TODO: we assume each color component is 8 bits, may need to support other configurations in the future, e.g. 10bits per color component?
165 uint8_t bytes_per_pixel = component_fmt.format.num_components;
166 rmt_strip = calloc(1, sizeof(led_strip_rmt_obj) + led_config->max_leds * bytes_per_pixel);
167 ESP_GOTO_ON_FALSE(rmt_strip, ESP_ERR_NO_MEM, err, TAG, "no mem for rmt strip");
168 uint32_t resolution = rmt_config->resolution_hz ? rmt_config->resolution_hz : LED_STRIP_RMT_DEFAULT_RESOLUTION;
169
170 // for backward compatibility, if the user does not set the clk_src, use the default value
171 rmt_clock_source_t clk_src = RMT_CLK_SRC_DEFAULT;
172 if (rmt_config->clk_src) {
173 clk_src = rmt_config->clk_src;
174 }
175 size_t mem_block_symbols = LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS;
176 // override the default value if the user sets it
177 if (rmt_config->mem_block_symbols) {
178 mem_block_symbols = rmt_config->mem_block_symbols;
179 }
180 rmt_tx_channel_config_t rmt_chan_config = {
181 .clk_src = clk_src,
182 .gpio_num = led_config->strip_gpio_num,
183 .mem_block_symbols = mem_block_symbols,
184 .resolution_hz = resolution,
185 .trans_queue_depth = LED_STRIP_RMT_DEFAULT_TRANS_QUEUE_SIZE,
186 .flags.with_dma = rmt_config->flags.with_dma,
187 .flags.invert_out = led_config->flags.invert_out,
188 };
189 ESP_GOTO_ON_ERROR(rmt_new_tx_channel(&rmt_chan_config, &rmt_strip->rmt_chan), err, TAG, "create RMT TX channel failed");
190
191 led_strip_encoder_config_t strip_encoder_conf = {
192 .resolution = resolution,
193 .led_model = led_config->led_model,
194 .timings = led_config->timings,
195 };
196 ESP_GOTO_ON_ERROR(rmt_new_led_strip_encoder(&strip_encoder_conf, &rmt_strip->strip_encoder), err, TAG, "create LED strip encoder failed");
197
198 rmt_strip->component_fmt = component_fmt;
199 rmt_strip->bytes_per_pixel = bytes_per_pixel;
200 rmt_strip->strip_len = led_config->max_leds;
201 rmt_strip->base.set_pixel = led_strip_rmt_set_pixel;
202 rmt_strip->base.set_pixel_rgbw = led_strip_rmt_set_pixel_rgbw;
203 rmt_strip->base.refresh = led_strip_rmt_refresh;
204 rmt_strip->base.refresh_async = led_strip_rmt_refresh_async;
205 rmt_strip->base.refresh_wait_done = led_strip_rmt_wait_for_done;
206 rmt_strip->base.clear = led_strip_rmt_clear;
207 rmt_strip->base.del = led_strip_rmt_del;
208
209 *ret_strip = &rmt_strip->base;
210 return ESP_OK;
211err:
212 if (rmt_strip) {
213 if (rmt_strip->rmt_chan) {
214 rmt_del_channel(rmt_strip->rmt_chan);
215 }
216 if (rmt_strip->strip_encoder) {
217 rmt_del_encoder(rmt_strip->strip_encoder);
218 }
219 free(rmt_strip);
220 }
221 return ret;
222}
223
224#endif // FASTLED_RMT5
225
226#endif // ESP32
LED Strip common configurations The common configurations are not specific to any backend peripheral.
led_color_component_format_t color_component_format
struct led_strip_config_t::led_strip_extra_flags flags
led_strip_encoder_timings_t timings
Type of led strip encoder configuration.
LED Strip RMT specific configuration.
struct led_strip_rmt_config_t::led_strip_rmt_extra_config flags
rmt_clock_source_t clk_src
LED strip interface definition.
LED color component format.
struct led_color_component_format_t::format_layout format