diff --git a/led_strip/CHANGELOG.md b/led_strip/CHANGELOG.md index bfbb0c7315..bd1b6e902f 100644 --- a/led_strip/CHANGELOG.md +++ b/led_strip/CHANGELOG.md @@ -1,7 +1,8 @@ -## 2.6.0 +## 3.0.0 + +- Discontinued support for ESP-IDF v4.x +- Added configuration for user-defined color component order -- Add pixel order configuration to support user-defined pixel order. - ## 2.5.5 - Simplified the led_strip component dependency, the time of full build with ESP-IDF v5.3 can now be shorter. diff --git a/led_strip/CMakeLists.txt b/led_strip/CMakeLists.txt index 13c35ec3d3..4fb17a839b 100644 --- a/led_strip/CMakeLists.txt +++ b/led_strip/CMakeLists.txt @@ -3,13 +3,8 @@ include($ENV{IDF_PATH}/tools/cmake/version.cmake) set(srcs "src/led_strip_api.c" "src/led_strip_common.c") set(public_requires) -# Starting from esp-idf v5.x, the RMT driver is rewritten -if("${IDF_VERSION_MAJOR}.${IDF_VERSION_MINOR}" VERSION_GREATER_EQUAL "5.0") - if(CONFIG_SOC_RMT_SUPPORTED) - list(APPEND srcs "src/led_strip_rmt_dev.c" "src/led_strip_rmt_encoder.c") - endif() -else() - list(APPEND srcs "src/led_strip_rmt_dev_idf4.c") +if(CONFIG_SOC_RMT_SUPPORTED) + list(APPEND srcs "src/led_strip_rmt_dev.c" "src/led_strip_rmt_encoder.c") endif() # the SPI backend driver relies on some feature that was available in IDF 5.1 diff --git a/led_strip/README.md b/led_strip/README.md index 6634b4f7cf..5761bf1252 100644 --- a/led_strip/README.md +++ b/led_strip/README.md @@ -15,28 +15,30 @@ This is the most economical way to drive the LEDs because it only consumes one R ```c #define BLINK_GPIO 0 -led_strip_handle_t led_strip; - -/* LED strip initialization with the GPIO and pixels number*/ +/// LED strip common configuration led_strip_config_t strip_config = { - .strip_gpio_num = BLINK_GPIO, // The GPIO that connected to the LED strip's data line - .max_leds = 1, // The number of LEDs in the strip, - .bytes_per_pixel = 3, // 3 bytes per pixel of the LED strip + .strip_gpio_num = BLINK_GPIO, // The GPIO that connected to the LED strip's data line + .max_leds = 1, // The number of LEDs in the strip, .led_model = LED_MODEL_WS2812, // LED strip model - .flags.invert_out = false, // whether to invert the output signal (useful when your hardware has a level inverter) - .pixel_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), /* The order of the pixel color. Not set or set to 0 if the default order is used. - Here set to the default GRB order to demonstrate usage */ + .num_color_components = 3, // Each pixel has 3 color components: R,G,B + .color_component_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), // Component order: G,R,B + .flags = { + .invert_out = false, // don't invert the output signal + } }; +/// RMT backend specific configuration led_strip_rmt_config_t rmt_config = { -#if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 0, 0) - .rmt_channel = 0, -#else - .clk_src = RMT_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption - .resolution_hz = 10 * 1000 * 1000, // 10MHz - .flags.with_dma = false, // whether to enable the DMA feature -#endif + .clk_src = RMT_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption + .resolution_hz = 10 * 1000 * 1000, // RMT counter clock frequency: 10MHz + .mem_block_symbols = 64, // the memory size of each RMT channel, in words (4 bytes) + .flags = { + .with_dma = false, // DMA feature is available on chips like ESP32-S3/P4 + } }; + +/// Create the LED strip object +led_strip_handle_t led_strip; ESP_ERROR_CHECK(led_strip_new_rmt_device(&strip_config, &rmt_config, &led_strip)); ``` @@ -53,24 +55,29 @@ Please note, the SPI backend has a dependency of **ESP-IDF >= 5.1** ```c #define BLINK_GPIO 0 -led_strip_handle_t led_strip; - -/* LED strip initialization with the GPIO and pixels number*/ +/// LED strip common configuration led_strip_config_t strip_config = { - .strip_gpio_num = BLINK_GPIO, // The GPIO that connected to the LED strip's data line - .max_leds = 1, // The number of LEDs in the strip, - .bytes_per_pixel = 3, // 3 bytes per pixel of the LED strip + .strip_gpio_num = BLINK_GPIO, // The GPIO that connected to the LED strip's data line + .max_leds = 1, // The number of LEDs in the strip, .led_model = LED_MODEL_WS2812, // LED strip model - .flags.invert_out = false, // whether to invert the output signal (useful when your hardware has a level inverter) - .pixel_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), /* The order of the pixel color. Not set or set to 0 if the default order is used. - Here set to the default GRB order to demonstrate usage */ + .num_color_components = 3, // Each pixel has 3 color components: R,G,B + .color_component_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), // Component order: G,R,B + .flags = { + .invert_out = false, // don't invert the output signal + } }; +/// SPI backend specific configuration led_strip_spi_config_t spi_config = { .clk_src = SPI_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption - .flags.with_dma = true, // Using DMA can improve performance and help drive more LEDs - .spi_bus = SPI2_HOST, // SPI bus ID + .spi_bus = SPI2_HOST, // SPI bus ID + .flags = { + .with_dma = true, // Using DMA can improve performance and help drive more LEDs + } }; + +/// Create the LED strip object +led_strip_handle_t led_strip; ESP_ERROR_CHECK(led_strip_new_spi_device(&strip_config, &spi_config, &led_strip)); ``` diff --git a/led_strip/api.md b/led_strip/api.md index 1f589c75bf..1ff31ef299 100644 --- a/led_strip/api.md +++ b/led_strip/api.md @@ -21,13 +21,6 @@ | esp\_err\_t | [**led\_strip\_set\_pixel\_hsv**](#function-led_strip_set_pixel_hsv) ([**led\_strip\_handle\_t**](#typedef-led_strip_handle_t) strip, uint32\_t index, uint16\_t hue, uint8\_t saturation, uint8\_t value)
_Set HSV for a specific pixel._ | | esp\_err\_t | [**led\_strip\_set\_pixel\_rgbw**](#function-led_strip_set_pixel_rgbw) ([**led\_strip\_handle\_t**](#typedef-led_strip_handle_t) strip, uint32\_t index, uint32\_t red, uint32\_t green, uint32\_t blue, uint32\_t white)
_Set RGBW for a specific pixel._ | -## Macros - -| Type | Name | -| ---: | :--- | -| define | [**LED\_STRIP\_SET\_RGBW\_ORDER**](#define-led_strip_set_rgbw_order) (R, G, B, W) (R << 0 \| G << 2 \| B << 4 \| W << 6)
_Help macro to set pixel RGBW color order The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ | -| define | [**LED\_STRIP\_SET\_RGB\_ORDER**](#define-led_strip_set_rgb_order) (R, G, B) (R << 0 \| G << 2 \| B << 4)
_Help macro to set pixel RGB color order The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ | - ## Functions Documentation ### function `led_strip_clear` @@ -185,45 +178,6 @@ Also see `led_strip_set_pixel` if you only want to specify the RGB part of the c - ESP\_ERR\_INVALID\_ARG: Set RGBW color for a specific pixel failed because of an invalid argument - ESP\_FAIL: Set RGBW color for a specific pixel failed because other error occurred -## Macros Documentation - -### define `LED_STRIP_SET_RGBW_ORDER` - -_Help macro to set pixel RGBW color order The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ - -```c -#define LED_STRIP_SET_RGBW_ORDER (R, G, B, W) (R << 0 | G << 2 | B << 4 | W << 6) -``` - -**Parameters:** - -- `R` The position of the red channel in the color order. -- `G` The position of the green channel in the color order. -- `B` The position of the blue channel in the color order. -- `W` The position of the white channel in the color order. - -**Note:** - -The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. - -### define `LED_STRIP_SET_RGB_ORDER` - -_Help macro to set pixel RGB color order The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ - -```c -#define LED_STRIP_SET_RGB_ORDER (R, G, B) (R << 0 | G << 2 | B << 4) -``` - -**Parameters:** - -- `R` The position of the red channel in the color order. -- `G` The position of the green channel in the color order. -- `B` The position of the blue channel in the color order. - -**Note:** - -The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. - ## File include/led_strip_rmt.h ## Structures and Types @@ -231,6 +185,7 @@ The order starts from 0. And the user needs to make sure that all the numbers ap | Type | Name | | ---: | :--- | | struct | [**led\_strip\_rmt\_config\_t**](#struct-led_strip_rmt_config_t)
_LED Strip RMT specific configuration._ | +| struct | [**led\_strip\_rmt\_extra\_config**](#struct-led_strip_rmt_config_tled_strip_rmt_extra_config)
| ## Functions @@ -248,12 +203,16 @@ Variables: - rmt\_clock\_source\_t clk_src
RMT clock source -- struct led\_strip\_rmt\_config\_t::@0 flags
Extra driver flags +- struct [**led\_strip\_rmt\_config\_t::led\_strip\_rmt\_extra\_config**](#struct-led_strip_rmt_config_tled_strip_rmt_extra_config) flags
Extra driver flags -- size\_t mem_block_symbols
How many RMT symbols can one RMT channel hold at one time. Set to 0 will fallback to use the default size. +- size\_t mem_block_symbols
How many RMT symbols can one RMT channel hold at one time. Set to 0 will fallback to use the default size. Extra RMT specific driver flags - uint32\_t resolution_hz
RMT tick resolution, if set to zero, a default resolution (10MHz) will be applied +### struct `led_strip_rmt_config_t::led_strip_rmt_extra_config` + +Variables: + - uint32\_t with_dma
Use DMA to transmit data ## Functions Documentation @@ -307,7 +266,7 @@ Variables: - spi\_clock\_source\_t clk_src
SPI clock source -- struct led\_strip\_spi\_config\_t::@1 flags
Extra driver flags +- struct [**led\_strip\_spi\_config\_t**](#struct-led_strip_spi_config_t) flags
Extra driver flags - spi\_host\_device\_t spi_bus
SPI bus ID. Which buses are available depends on the specific chip @@ -352,8 +311,16 @@ Although only the MOSI line is used for generating the signal, the whole SPI bus | Type | Name | | ---: | :--- | | enum | [**led\_model\_t**](#enum-led_model_t)
_LED strip model._ | -| struct | [**led\_strip\_config\_t**](#struct-led_strip_config_t)
_LED Strip Configuration._ | -| typedef struct [**led\_strip\_t**](#struct-led_strip_t) \* | [**led\_strip\_handle\_t**](#typedef-led_strip_handle_t)
_LED strip handle._ | +| struct | [**led\_strip\_config\_t**](#struct-led_strip_config_t)
_LED Strip common configurations The common configurations are not specific to any backend peripheral._ | +| struct | [**led\_strip\_extra\_flags**](#struct-led_strip_config_tled_strip_extra_flags)
| +| typedef struct [**led\_strip\_t**](#struct-led_strip_t) \* | [**led\_strip\_handle\_t**](#typedef-led_strip_handle_t)
_Type of LED strip handle._ | + +## Macros + +| Type | Name | +| ---: | :--- | +| define | [**LED\_STRIP\_SET\_RGBW\_ORDER**](#define-led_strip_set_rgbw_order) (R, G, B, W) (R << 0 | G << 2 | B << 4 | W << 6)
_Help macro to set pixel RGBW color order The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ | +| define | [**LED\_STRIP\_SET\_RGB\_ORDER**](#define-led_strip_set_rgb_order) (R, G, B) (R << 0 | G << 2 | B << 4)
_Help macro to set pixel RGB color order The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ | ## Structures and Types Documentation @@ -375,32 +342,84 @@ Different led model may have different timing parameters, so we need to distingu ### struct `led_strip_config_t` -_LED Strip Configuration._ +_LED Strip common configurations The common configurations are not specific to any backend peripheral._ Variables: -- uint8\_t bytes_per_pixel
bytes per LED pixel. Should be 3 or 4 +- uint8\_t color_component_order
Specifies the order of color components in each pixel. Use helper macros LED\_STRIP\_SET\_RGB\_ORDER or LED\_STRIP\_SET\_RGBW\_ORDER to set the order. Set to 0 to use the default order. LED strip extra driver flags -- struct led\_strip\_config\_t::@2 flags
The order of the pixel color. Use help macro LED\_STRIP\_SET\_RGB\_ORDER or LED\_STRIP\_SET\_RGBW\_ORDER to set. Not set or set to 0 if the default order is used. Extra driver flags +- struct [**led\_strip\_config\_t::led\_strip\_extra\_flags**](#struct-led_strip_config_tled_strip_extra_flags) flags
Extra driver flags -- uint32\_t invert_out
Invert output signal - -- [**led\_model\_t**](#enum-led_model_t) led_model
LED model +- [**led\_model\_t**](#enum-led_model_t) led_model
Specifies the LED strip model (e.g., WS2812, SK6812) -- uint32\_t max_leds
Maximum LEDs in a single strip +- uint32\_t max_leds
Maximum number of LEDs that can be controlled in a single strip -- uint8\_t pixel_order +- uint8\_t num_color_components
Number of color components per LED pixel. Use 3 for RGB (Red, Green, Blue) or 4 for RGBW (Red, Green, Blue, White). If set to 0, the driver will default to 3 (RGB). - int strip_gpio_num
GPIO number that used by LED strip +### struct `led_strip_config_t::led_strip_extra_flags` + +Variables: + +- uint32\_t invert_out
Invert output signal + ### typedef `led_strip_handle_t` -_LED strip handle._ +_Type of LED strip handle._ ```c typedef struct led_strip_t* led_strip_handle_t; ``` +## Macros Documentation + +### define `LED_STRIP_SET_RGBW_ORDER` + +_Help macro to set pixel RGBW color order The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ + +```c +#define LED_STRIP_SET_RGBW_ORDER ( + R, + G, + B, + W +) (R << 0 | G << 2 | B << 4 | W << 6) +``` + +**Parameters:** + +- `R` The position of the red channel in the color order. +- `G` The position of the green channel in the color order. +- `B` The position of the blue channel in the color order. +- `W` The position of the white channel in the color order. + +**Note:** + +The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. + +### define `LED_STRIP_SET_RGB_ORDER` + +_Help macro to set pixel RGB color order The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set_ `pixel_order`_in_[_**led\_strip\_config\_t**_](#struct-led_strip_config_t)_. The positions are counted from the least significant bit (LSB)._ + +```c +#define LED_STRIP_SET_RGB_ORDER ( + R, + G, + B +) (R << 0 | G << 2 | B << 4) +``` + +**Parameters:** + +- `R` The position of the red channel in the color order. +- `G` The position of the green channel in the color order. +- `B` The position of the blue channel in the color order. + +**Note:** + +The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. + ## File interface/led_strip_interface.h ## Structures and Types @@ -408,7 +427,7 @@ typedef struct led_strip_t* led_strip_handle_t; | Type | Name | | ---: | :--- | | struct | [**led\_strip\_t**](#struct-led_strip_t)
_LED strip interface definition._ | -| typedef struct [**led\_strip\_t**](#struct-led_strip_t) | [**led\_strip\_t**](#typedef-led_strip_t)
| +| typedef struct led\_strip\_t | [**led\_strip\_t**](#typedef-led_strip_t)
| ## Structures and Types Documentation diff --git a/led_strip/examples/led_strip_rmt_ws2812/CMakeLists.txt b/led_strip/examples/led_strip_rmt_ws2812/CMakeLists.txt index fc2dfc7a0e..ff10c822c7 100644 --- a/led_strip/examples/led_strip_rmt_ws2812/CMakeLists.txt +++ b/led_strip/examples/led_strip_rmt_ws2812/CMakeLists.txt @@ -1,8 +1,6 @@ -# For more information about build system see -# https://docs.espressif.com/projects/esp-idf/en/latest/api-guides/build-system.html -# The following five lines of boilerplate have to be in your project's -# CMakeLists in this exact order for cmake to work correctly cmake_minimum_required(VERSION 3.16) +set(COMPONENTS main) + include($ENV{IDF_PATH}/tools/cmake/project.cmake) project(led_strip_rmt_ws2812) diff --git a/led_strip/examples/led_strip_rmt_ws2812/main/idf_component.yml b/led_strip/examples/led_strip_rmt_ws2812/main/idf_component.yml index 916c366c79..67a577507b 100644 --- a/led_strip/examples/led_strip_rmt_ws2812/main/idf_component.yml +++ b/led_strip/examples/led_strip_rmt_ws2812/main/idf_component.yml @@ -1,5 +1,5 @@ ## IDF Component Manager Manifest File dependencies: espressif/led_strip: - version: '^2' + version: '^3' override_path: '../../../' diff --git a/led_strip/examples/led_strip_rmt_ws2812/main/led_strip_rmt_ws2812_main.c b/led_strip/examples/led_strip_rmt_ws2812/main/led_strip_rmt_ws2812_main.c index 863e7d63b0..87e07fbf5c 100644 --- a/led_strip/examples/led_strip_rmt_ws2812/main/led_strip_rmt_ws2812_main.c +++ b/led_strip/examples/led_strip_rmt_ws2812/main/led_strip_rmt_ws2812_main.c @@ -23,24 +23,24 @@ led_strip_handle_t configure_led(void) { // LED strip general initialization, according to your led board design led_strip_config_t strip_config = { - .strip_gpio_num = LED_STRIP_GPIO_PIN, // The GPIO that connected to the LED strip's data line - .max_leds = LED_STRIP_LED_COUNT, // The number of LEDs in the strip, - .bytes_per_pixel = 3, // 3 bytes per pixel of the LED strip - .led_model = LED_MODEL_WS2812, // LED strip model - .flags.invert_out = false, // whether to invert the output signal - .pixel_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), /* The order of the pixel color. Not set or set to 0 if the default order is used. - Here set to the default GRB order to demonstrate usage */ + .strip_gpio_num = LED_STRIP_GPIO_PIN, // The GPIO that connected to the LED strip's data line + .max_leds = LED_STRIP_LED_COUNT, // The number of LEDs in the strip, + .led_model = LED_MODEL_WS2812, // LED strip model + .num_color_components = 3, // Each pixel has 3 color components: R,G,B + .color_component_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), // Component order: G,R,B + .flags = { + .invert_out = false, // don't invert the output signal + } }; // LED strip backend configuration: RMT led_strip_rmt_config_t rmt_config = { -#if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 0, 0) - .rmt_channel = 0, -#else .clk_src = RMT_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption .resolution_hz = LED_STRIP_RMT_RES_HZ, // RMT counter clock frequency - .flags.with_dma = false, // DMA feature is available on ESP target like ESP32-S3 -#endif + .mem_block_symbols = 64, // the memory size of each RMT channel, in words (4 bytes) + .flags = { + .with_dma = false, // DMA feature is available on chips like ESP32-S3/P4 + } }; // LED Strip object handle diff --git a/led_strip/examples/led_strip_spi_ws2812/CMakeLists.txt b/led_strip/examples/led_strip_spi_ws2812/CMakeLists.txt index 7d3af2dfae..822cf2bbad 100644 --- a/led_strip/examples/led_strip_spi_ws2812/CMakeLists.txt +++ b/led_strip/examples/led_strip_spi_ws2812/CMakeLists.txt @@ -1,8 +1,6 @@ -# For more information about build system see -# https://docs.espressif.com/projects/esp-idf/en/latest/api-guides/build-system.html -# The following five lines of boilerplate have to be in your project's -# CMakeLists in this exact order for cmake to work correctly cmake_minimum_required(VERSION 3.16) +set(COMPONENTS main) + include($ENV{IDF_PATH}/tools/cmake/project.cmake) project(led_strip_spi_ws2812) diff --git a/led_strip/examples/led_strip_spi_ws2812/main/idf_component.yml b/led_strip/examples/led_strip_spi_ws2812/main/idf_component.yml index 050d8d269f..ed00d7eb3c 100644 --- a/led_strip/examples/led_strip_spi_ws2812/main/idf_component.yml +++ b/led_strip/examples/led_strip_spi_ws2812/main/idf_component.yml @@ -1,6 +1,6 @@ ## IDF Component Manager Manifest File dependencies: espressif/led_strip: - version: '^2.4' + version: '^3' override_path: '../../../' idf: ">=5.1" diff --git a/led_strip/examples/led_strip_spi_ws2812/main/led_strip_spi_ws2812_main.c b/led_strip/examples/led_strip_spi_ws2812/main/led_strip_spi_ws2812_main.c index 54a1b54c74..d78eb4f291 100644 --- a/led_strip/examples/led_strip_spi_ws2812/main/led_strip_spi_ws2812_main.c +++ b/led_strip/examples/led_strip_spi_ws2812/main/led_strip_spi_ws2812_main.c @@ -21,20 +21,23 @@ led_strip_handle_t configure_led(void) { // LED strip general initialization, according to your led board design led_strip_config_t strip_config = { - .strip_gpio_num = LED_STRIP_GPIO_PIN, // The GPIO that connected to the LED strip's data line - .max_leds = LED_STRIP_LED_COUNT, // The number of LEDs in the strip, - .bytes_per_pixel = 3, // 3 bytes per pixel of the LED strip - .led_model = LED_MODEL_WS2812, // LED strip model - .flags.invert_out = false, // whether to invert the output signal - .pixel_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), /* The order of the pixel color. Not set or set to 0 if the default order is used. - Here set to the default GRB order to demonstrate usage */ + .strip_gpio_num = LED_STRIP_GPIO_PIN, // The GPIO that connected to the LED strip's data line + .max_leds = LED_STRIP_LED_COUNT, // The number of LEDs in the strip, + .led_model = LED_MODEL_WS2812, // LED strip model + .num_color_components = 3, // Each pixel has 3 color components: R,G,B + .color_component_order = LED_STRIP_SET_RGB_ORDER(1, 0, 2), // Component order: G,R,B + .flags = { + .invert_out = false, // don't invert the output signal + } }; // LED strip backend configuration: SPI led_strip_spi_config_t spi_config = { .clk_src = SPI_CLK_SRC_DEFAULT, // different clock source can lead to different power consumption - .flags.with_dma = true, // Using DMA can improve performance and help drive more LEDs .spi_bus = SPI2_HOST, // SPI bus ID + .flags = { + .with_dma = true, // Using DMA can improve performance and help drive more LEDs + } }; // LED Strip object handle diff --git a/led_strip/idf_component.yml b/led_strip/idf_component.yml index ae2e6031bc..eb75f1ed1a 100644 --- a/led_strip/idf_component.yml +++ b/led_strip/idf_component.yml @@ -1,5 +1,7 @@ -version: "2.5.6" +version: "3.0.0" description: Driver for Addressable LED Strip (WS2812, etc) url: https://github.com/espressif/idf-extra-components/tree/master/led_strip +issues: https://github.com/espressif/idf-extra-components/issues +repository: https://github.com/espressif/idf-extra-components.git dependencies: - idf: ">=4.4" + idf: ">=5.0" diff --git a/led_strip/include/led_strip.h b/led_strip/include/led_strip.h index 3c4758009c..36fe5fefff 100644 --- a/led_strip/include/led_strip.h +++ b/led_strip/include/led_strip.h @@ -8,39 +8,12 @@ #include #include "esp_err.h" #include "led_strip_rmt.h" -#include "esp_idf_version.h" - -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 1, 0) #include "led_strip_spi.h" -#endif #ifdef __cplusplus extern "C" { #endif -/** - * @brief Help macro to set pixel RGB color order - * The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set `pixel_order` in led_strip_config_t. - * The positions are counted from the least significant bit (LSB). - * @param R The position of the red channel in the color order. - * @param G The position of the green channel in the color order. - * @param B The position of the blue channel in the color order. - * @note The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. - */ -#define LED_STRIP_SET_RGB_ORDER(R, G, B) (R << 0 | G << 2 | B << 4) - -/** - * @brief Help macro to set pixel RGBW color order - * The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set `pixel_order` in led_strip_config_t. - * The positions are counted from the least significant bit (LSB). - * @param R The position of the red channel in the color order. - * @param G The position of the green channel in the color order. - * @param B The position of the blue channel in the color order. - * @param W The position of the white channel in the color order. - * @note The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. - */ -#define LED_STRIP_SET_RGBW_ORDER(R, G, B, W) (R << 0 | G << 2 | B << 4 | W << 6) - /** * @brief Set RGB for a specific pixel * diff --git a/led_strip/include/led_strip_rmt.h b/led_strip/include/led_strip_rmt.h index b575aeaba1..3b58276b14 100644 --- a/led_strip/include/led_strip_rmt.h +++ b/led_strip/include/led_strip_rmt.h @@ -1,5 +1,5 @@ /* - * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD + * SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 */ @@ -9,10 +9,7 @@ #include "esp_err.h" #include "led_strip_types.h" #include "esp_idf_version.h" - -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0) #include "driver/rmt_types.h" -#endif #ifdef __cplusplus extern "C" { @@ -22,14 +19,11 @@ extern "C" { * @brief LED Strip RMT specific configuration */ typedef struct { -#if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 0, 0) - uint8_t rmt_channel; /*!< Specify the channel number, the legacy RMT driver doesn't support channel allocator */ -#else // new driver supports specify the clock source and clock resolution rmt_clock_source_t clk_src; /*!< RMT clock source */ uint32_t resolution_hz; /*!< RMT tick resolution, if set to zero, a default resolution (10MHz) will be applied */ -#endif size_t mem_block_symbols; /*!< How many RMT symbols can one RMT channel hold at one time. Set to 0 will fallback to use the default size. */ - struct { + /*!< Extra RMT specific driver flags */ + struct led_strip_rmt_extra_config { uint32_t with_dma: 1; /*!< Use DMA to transmit data */ } flags; /*!< Extra driver flags */ } led_strip_rmt_config_t; diff --git a/led_strip/include/led_strip_spi.h b/led_strip/include/led_strip_spi.h index eb35249360..cd66e7a8f1 100644 --- a/led_strip/include/led_strip_spi.h +++ b/led_strip/include/led_strip_spi.h @@ -1,5 +1,5 @@ /* - * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD + * SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 */ @@ -27,6 +27,7 @@ typedef struct { /** * @brief Create LED strip based on SPI MOSI channel + * * @note Although only the MOSI line is used for generating the signal, the whole SPI bus can't be used for other purposes. * * @param led_config LED strip configuration diff --git a/led_strip/include/led_strip_types.h b/led_strip/include/led_strip_types.h index 47aea74d8a..9667871516 100644 --- a/led_strip/include/led_strip_types.h +++ b/led_strip/include/led_strip_types.h @@ -11,6 +11,11 @@ extern "C" { #endif +/** + * @brief Type of LED strip handle + */ +typedef struct led_strip_t *led_strip_handle_t; + /** * @brief LED strip model * @note Different led model may have different timing parameters, so we need to distinguish them. @@ -22,24 +27,46 @@ typedef enum { } led_model_t; /** - * @brief LED strip handle + * @brief Help macro to set pixel RGB color order + * The default order of the three-color LED strips is GRB. If you have a different order, you can use the macro to set `pixel_order` in led_strip_config_t. + * The positions are counted from the least significant bit (LSB). + * @param R The position of the red channel in the color order. + * @param G The position of the green channel in the color order. + * @param B The position of the blue channel in the color order. + * @note The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. */ -typedef struct led_strip_t *led_strip_handle_t; +#define LED_STRIP_SET_RGB_ORDER(R, G, B) (R << 0 | G << 2 | B << 4) + +/** + * @brief Help macro to set pixel RGBW color order + * The default order of the four-color LED strips is GRBW. If you have a different order, you can use the macro to set `pixel_order` in led_strip_config_t. + * The positions are counted from the least significant bit (LSB). + * @param R The position of the red channel in the color order. + * @param G The position of the green channel in the color order. + * @param B The position of the blue channel in the color order. + * @param W The position of the white channel in the color order. + * @note The order starts from 0. And the user needs to make sure that all the numbers appear exactly once and are all less than the number of colors per pixel. + */ +#define LED_STRIP_SET_RGBW_ORDER(R, G, B, W) (R << 0 | G << 2 | B << 4 | W << 6) /** - * @brief LED Strip Configuration + * @brief LED Strip common configurations + * The common configurations are not specific to any backend peripheral. */ typedef struct { - int strip_gpio_num; /*!< GPIO number that used by LED strip */ - uint32_t max_leds; /*!< Maximum LEDs in a single strip */ - uint8_t bytes_per_pixel; /*!< bytes per LED pixel. Should be 3 or 4 */ - led_model_t led_model; /*!< LED model */ - uint8_t pixel_order; /*! The order of the pixel color. - Use help macro LED_STRIP_SET_RGB_ORDER or LED_STRIP_SET_RGBW_ORDER to set. - Not set or set to 0 if the default order is used. */ - struct { + int strip_gpio_num; /*!< GPIO number that used by LED strip */ + uint32_t max_leds; /*!< Maximum number of LEDs that can be controlled in a single strip */ + led_model_t led_model; /*!< Specifies the LED strip model (e.g., WS2812, SK6812) */ + uint8_t num_color_components; /*!< Number of color components per LED pixel. + Use 3 for RGB (Red, Green, Blue) or 4 for RGBW (Red, Green, Blue, White). + If set to 0, the driver will default to 3 (RGB). */ + uint8_t color_component_order; /*!< Specifies the order of color components in each pixel. + Use helper macros LED_STRIP_SET_RGB_ORDER or LED_STRIP_SET_RGBW_ORDER to set the order. + Set to 0 to use the default order. */ + /*!< LED strip extra driver flags */ + struct led_strip_extra_flags { uint32_t invert_out: 1; /*!< Invert output signal */ - } flags; /*!< Extra driver flags */ + } flags; /*!< Extra driver flags */ } led_strip_config_t; #ifdef __cplusplus diff --git a/led_strip/src/led_strip_common.c b/led_strip/src/led_strip_common.c index 6338e9ffed..5028961809 100644 --- a/led_strip/src/led_strip_common.c +++ b/led_strip/src/led_strip_common.c @@ -11,17 +11,23 @@ static const char *TAG = "led_strip_common"; -esp_err_t led_strip_set_color_order(uint8_t *led_pixel_offset, const uint8_t pixel_order, const uint8_t bytes_per_pixel) +esp_err_t led_strip_set_color_order(uint8_t *led_pixel_offset, uint8_t pixel_order, uint8_t num_color_components) { if (pixel_order) { uint8_t r_order = (pixel_order >> 0) & 0x03; uint8_t g_order = (pixel_order >> 2) & 0x03; uint8_t b_order = (pixel_order >> 4) & 0x03; uint8_t w_order = (pixel_order >> 6) & 0x03; - uint8_t mask = bytes_per_pixel == 3 ? BIT(r_order) | BIT(g_order) | BIT(b_order) : BIT(r_order) | BIT(g_order) | BIT(b_order) | BIT(w_order); - // Check for invalid values - ESP_RETURN_ON_FALSE(__builtin_popcount(mask) == bytes_per_pixel && r_order < bytes_per_pixel && g_order < bytes_per_pixel && b_order < bytes_per_pixel && w_order < bytes_per_pixel, - ESP_ERR_INVALID_ARG, TAG, "invalid order argument"); + uint8_t mask = 0; + if (num_color_components == 3) { + mask = BIT(r_order) | BIT(g_order) | BIT(b_order); + // Check for invalid values + ESP_RETURN_ON_FALSE(mask == 0x07, ESP_ERR_INVALID_ARG, TAG, "invalid order argument"); + } else { + mask = BIT(r_order) | BIT(g_order) | BIT(b_order) | BIT(w_order); + // Check for invalid values + ESP_RETURN_ON_FALSE(mask == 0x0F, ESP_ERR_INVALID_ARG, TAG, "invalid order argument"); + } led_pixel_offset[LED_PIXEL_INDEX_RED] = r_order; led_pixel_offset[LED_PIXEL_INDEX_GREEN] = g_order; led_pixel_offset[LED_PIXEL_INDEX_BLUE] = b_order; diff --git a/led_strip/src/led_strip_common.h b/led_strip/src/led_strip_common.h index e4ca2811f3..ff2522786b 100644 --- a/led_strip/src/led_strip_common.h +++ b/led_strip/src/led_strip_common.h @@ -29,12 +29,12 @@ typedef enum { * * @param led_pixel_offset Each pixel's offset * @param pixel_order `pixel_order` parameter in LED strip configuration - * @param bytes_per_pixel bytes per pixel + * @param num_color_components Number of color components per LED pixel * @return * - ESP_OK: Set LED color order successfully * - ESP_ERR_INVALID_ARG: Set LED color order failed because of invalid argument */ -esp_err_t led_strip_set_color_order(uint8_t *led_pixel_offset, const uint8_t pixel_order, const uint8_t bytes_per_pixel); +esp_err_t led_strip_set_color_order(uint8_t *led_pixel_offset, uint8_t pixel_order, uint8_t num_color_components); #ifdef __cplusplus } diff --git a/led_strip/src/led_strip_rmt_dev.c b/led_strip/src/led_strip_rmt_dev.c index 15a2ef4a48..970dd6e826 100644 --- a/led_strip/src/led_strip_rmt_dev.c +++ b/led_strip/src/led_strip_rmt_dev.c @@ -106,8 +106,14 @@ esp_err_t led_strip_new_rmt_device(const led_strip_config_t *led_config, const l led_strip_rmt_obj *rmt_strip = NULL; esp_err_t ret = ESP_OK; ESP_GOTO_ON_FALSE(led_config && rmt_config && ret_strip, ESP_ERR_INVALID_ARG, err, TAG, "invalid argument"); - ESP_GOTO_ON_FALSE(led_config->bytes_per_pixel == 3 || led_config->bytes_per_pixel == 4, ESP_ERR_INVALID_ARG, err, TAG, "invalid led_pixel bytes"); - uint8_t bytes_per_pixel = led_config->bytes_per_pixel; + // by default, each pixel should have at least 3 color components: R, G, B + uint8_t num_color_components = 3; + // but sometimes, there can be different number of components, e.g. RGBW + if (led_config->num_color_components) { + num_color_components = led_config->num_color_components; + } + // TODO: we assume each color component is 8 bits, may need to support other configurations in the future, e.g. 10bits per color component? + uint8_t bytes_per_pixel = num_color_components; rmt_strip = calloc(1, sizeof(led_strip_rmt_obj) + led_config->max_leds * bytes_per_pixel); ESP_GOTO_ON_FALSE(rmt_strip, ESP_ERR_NO_MEM, err, TAG, "no mem for rmt strip"); uint32_t resolution = rmt_config->resolution_hz ? rmt_config->resolution_hz : LED_STRIP_RMT_DEFAULT_RESOLUTION; @@ -138,7 +144,8 @@ esp_err_t led_strip_new_rmt_device(const led_strip_config_t *led_config, const l .led_model = led_config->led_model }; ESP_GOTO_ON_ERROR(rmt_new_led_strip_encoder(&strip_encoder_conf, &rmt_strip->strip_encoder), err, TAG, "create LED strip encoder failed"); - ESP_GOTO_ON_ERROR(led_strip_set_color_order(rmt_strip->led_pixel_offset, led_config->pixel_order, bytes_per_pixel), err, TAG, "adjust color order failed"); + ESP_GOTO_ON_ERROR(led_strip_set_color_order(rmt_strip->led_pixel_offset, led_config->color_component_order, num_color_components), + err, TAG, "set color component order failed"); rmt_strip->bytes_per_pixel = bytes_per_pixel; rmt_strip->strip_len = led_config->max_leds; diff --git a/led_strip/src/led_strip_rmt_dev_idf4.c b/led_strip/src/led_strip_rmt_dev_idf4.c deleted file mode 100644 index 091ccf8106..0000000000 --- a/led_strip/src/led_strip_rmt_dev_idf4.c +++ /dev/null @@ -1,191 +0,0 @@ -/* - * SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD - * - * SPDX-License-Identifier: Apache-2.0 - */ -#include -#include -#include -#include "esp_log.h" -#include "esp_check.h" -#include "driver/rmt.h" -#include "led_strip.h" -#include "led_strip_interface.h" -#include "led_strip_common.h" - -static const char *TAG = "led_strip_rmt"; - -#define WS2812_T0H_NS (300) -#define WS2812_T0L_NS (900) -#define WS2812_T1H_NS (900) -#define WS2812_T1L_NS (300) - -#define SK6812_T0H_NS (300) -#define SK6812_T0L_NS (900) -#define SK6812_T1H_NS (600) -#define SK6812_T1L_NS (600) - -#define LED_STRIP_RESET_MS (10) - -// the memory size of each RMT channel, in words (4 bytes) -#if CONFIG_IDF_TARGET_ESP32 || CONFIG_IDF_TARGET_ESP32S2 -#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS 64 -#else -#define LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS 48 -#endif - -static uint32_t led_t0h_ticks = 0; -static uint32_t led_t1h_ticks = 0; -static uint32_t led_t0l_ticks = 0; -static uint32_t led_t1l_ticks = 0; - -typedef struct { - led_strip_t base; - rmt_channel_t rmt_channel; - uint32_t strip_len; - uint8_t bytes_per_pixel; - uint8_t led_pixel_offset[LED_PIXEL_INDEX_MAX]; - uint8_t buffer[0]; -} led_strip_rmt_obj; - -static void IRAM_ATTR ws2812_rmt_adapter(const void *src, rmt_item32_t *dest, size_t src_size, - size_t wanted_num, size_t *translated_size, size_t *item_num) -{ - if (src == NULL || dest == NULL) { - *translated_size = 0; - *item_num = 0; - return; - } - const rmt_item32_t bit0 = {{{ led_t0h_ticks, 1, led_t0l_ticks, 0 }}}; //Logical 0 - const rmt_item32_t bit1 = {{{ led_t1h_ticks, 1, led_t1l_ticks, 0 }}}; //Logical 1 - size_t size = 0; - size_t num = 0; - uint8_t *psrc = (uint8_t *)src; - rmt_item32_t *pdest = dest; - while (size < src_size && num < wanted_num) { - for (int i = 0; i < 8; i++) { - // MSB first - if (*psrc & (1 << (7 - i))) { - pdest->val = bit1.val; - } else { - pdest->val = bit0.val; - } - num++; - pdest++; - } - size++; - psrc++; - } - *translated_size = size; - *item_num = num; -} - -static esp_err_t led_strip_rmt_set_pixel(led_strip_t *strip, uint32_t index, uint32_t red, uint32_t green, uint32_t blue) -{ - led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base); - ESP_RETURN_ON_FALSE(index < rmt_strip->strip_len, ESP_ERR_INVALID_ARG, TAG, "index out of the maximum number of leds"); - uint32_t start = index * rmt_strip->bytes_per_pixel; - uint8_t *pixel_buf = rmt_strip->buffer; - uint8_t *offset = rmt_strip->led_pixel_offset; - // Support all kinds of pixel order - pixel_buf[start + offset[LED_PIXEL_INDEX_RED]] = red & 0xFF; - pixel_buf[start + offset[LED_PIXEL_INDEX_GREEN]] = green & 0xFF; - pixel_buf[start + offset[LED_PIXEL_INDEX_BLUE]] = blue & 0xFF; - if (rmt_strip->bytes_per_pixel > 3) { - rmt_strip->buffer[start + offset[LED_PIXEL_INDEX_WHITE]] = 0; - } - return ESP_OK; -} - -static esp_err_t led_strip_rmt_refresh(led_strip_t *strip) -{ - led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base); - ESP_RETURN_ON_ERROR(rmt_write_sample(rmt_strip->rmt_channel, rmt_strip->buffer, rmt_strip->strip_len * rmt_strip->bytes_per_pixel, true), TAG, - "transmit RMT samples failed"); - vTaskDelay(pdMS_TO_TICKS(LED_STRIP_RESET_MS)); - return ESP_OK; -} - -static esp_err_t led_strip_rmt_clear(led_strip_t *strip) -{ - led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base); - // Write zero to turn off all LEDs - memset(rmt_strip->buffer, 0, rmt_strip->strip_len * rmt_strip->bytes_per_pixel); - return led_strip_rmt_refresh(strip); -} - -static esp_err_t led_strip_rmt_del(led_strip_t *strip) -{ - led_strip_rmt_obj *rmt_strip = __containerof(strip, led_strip_rmt_obj, base); - ESP_RETURN_ON_ERROR(rmt_driver_uninstall(rmt_strip->rmt_channel), TAG, "uninstall RMT driver failed"); - free(rmt_strip); - return ESP_OK; -} - -esp_err_t led_strip_new_rmt_device(const led_strip_config_t *led_config, const led_strip_rmt_config_t *dev_config, led_strip_handle_t *ret_strip) -{ - led_strip_rmt_obj *rmt_strip = NULL; - esp_err_t ret = ESP_OK; - ESP_RETURN_ON_FALSE(led_config && dev_config && ret_strip, ESP_ERR_INVALID_ARG, TAG, "invalid argument"); - ESP_RETURN_ON_FALSE(dev_config->flags.with_dma == 0, ESP_ERR_NOT_SUPPORTED, TAG, "DMA is not supported"); - ESP_RETURN_ON_FALSE(led_config->bytes_per_pixel == 3 || led_config->bytes_per_pixel == 4, ESP_ERR_INVALID_ARG, TAG, "invalid led_pixel bytes"); - uint8_t bytes_per_pixel = led_config->bytes_per_pixel; - - // allocate memory for led_strip object - rmt_strip = calloc(1, sizeof(led_strip_rmt_obj) + led_config->max_leds * bytes_per_pixel); - ESP_RETURN_ON_FALSE(rmt_strip, ESP_ERR_NO_MEM, TAG, "request memory for les_strip failed"); - - // install RMT channel driver - rmt_config_t config = RMT_DEFAULT_CONFIG_TX(led_config->strip_gpio_num, dev_config->rmt_channel); - // set the minimal clock division because the LED strip needs a high clock resolution - config.clk_div = 2; - - uint8_t mem_block_num = 2; - // override the default value if the user specify the mem block size - if (dev_config->mem_block_symbols) { - mem_block_num = (dev_config->mem_block_symbols + LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS / 2) / LED_STRIP_RMT_DEFAULT_MEM_BLOCK_SYMBOLS; - } - config.mem_block_num = mem_block_num; - - ESP_GOTO_ON_ERROR(rmt_config(&config), err, TAG, "RMT config failed"); - ESP_GOTO_ON_ERROR(rmt_driver_install(config.channel, 0, 0), err, TAG, "RMT install failed"); - - uint32_t counter_clk_hz = 0; - rmt_get_counter_clock((rmt_channel_t)dev_config->rmt_channel, &counter_clk_hz); - // ns -> ticks - float ratio = (float)counter_clk_hz / 1e9; - if (led_config->led_model == LED_MODEL_WS2812) { - led_t0h_ticks = (uint32_t)(ratio * WS2812_T0H_NS); - led_t0l_ticks = (uint32_t)(ratio * WS2812_T0L_NS); - led_t1h_ticks = (uint32_t)(ratio * WS2812_T1H_NS); - led_t1l_ticks = (uint32_t)(ratio * WS2812_T1L_NS); - } else if (led_config->led_model == LED_MODEL_SK6812) { - led_t0h_ticks = (uint32_t)(ratio * SK6812_T0H_NS); - led_t0l_ticks = (uint32_t)(ratio * SK6812_T0L_NS); - led_t1h_ticks = (uint32_t)(ratio * SK6812_T1H_NS); - led_t1l_ticks = (uint32_t)(ratio * SK6812_T1L_NS); - } else { - assert(false); - } - - // adapter to translates the LES strip date frame into RMT symbols - rmt_translator_init((rmt_channel_t)dev_config->rmt_channel, ws2812_rmt_adapter); - ESP_GOTO_ON_ERROR(led_strip_set_color_order(rmt_strip->led_pixel_offset, led_config->pixel_order, bytes_per_pixel), err, TAG, "adjust color order failed"); - - rmt_strip->bytes_per_pixel = bytes_per_pixel; - rmt_strip->rmt_channel = (rmt_channel_t)dev_config->rmt_channel; - rmt_strip->strip_len = led_config->max_leds; - rmt_strip->base.set_pixel = led_strip_rmt_set_pixel; - rmt_strip->base.refresh = led_strip_rmt_refresh; - rmt_strip->base.clear = led_strip_rmt_clear; - rmt_strip->base.del = led_strip_rmt_del; - - *ret_strip = &rmt_strip->base; - return ESP_OK; - -err: - if (rmt_strip) { - free(rmt_strip); - } - return ret; -} diff --git a/led_strip/src/led_strip_spi_dev.c b/led_strip/src/led_strip_spi_dev.c index 248c474cf5..410abe79fb 100644 --- a/led_strip/src/led_strip_spi_dev.c +++ b/led_strip/src/led_strip_spi_dev.c @@ -10,7 +10,6 @@ #include "esp_check.h" #include "esp_rom_gpio.h" #include "soc/spi_periph.h" -#include "hal/spi_hal.h" #include "led_strip.h" #include "led_strip_interface.h" #include "led_strip_common.h" @@ -130,8 +129,14 @@ esp_err_t led_strip_new_spi_device(const led_strip_config_t *led_config, const l led_strip_spi_obj *spi_strip = NULL; esp_err_t ret = ESP_OK; ESP_GOTO_ON_FALSE(led_config && spi_config && ret_strip, ESP_ERR_INVALID_ARG, err, TAG, "invalid argument"); - ESP_GOTO_ON_FALSE(led_config->bytes_per_pixel == 3 || led_config->bytes_per_pixel == 4, ESP_ERR_INVALID_ARG, err, TAG, "invalid led_pixel bytes"); - uint8_t bytes_per_pixel = led_config->bytes_per_pixel; + // by default, each pixel should have at least 3 color components: R, G, B + uint8_t num_color_components = 3; + // but sometimes, there can be different number of components, e.g. RGBW + if (led_config->num_color_components) { + num_color_components = led_config->num_color_components; + } + // TODO: we assume each color component is 8 bits, may need to support other configurations in the future, e.g. 10bits per color component? + uint8_t bytes_per_pixel = num_color_components; uint32_t mem_caps = MALLOC_CAP_DEFAULT; if (spi_config->flags.with_dma) { // DMA buffer must be placed in internal SRAM @@ -185,7 +190,8 @@ esp_err_t led_strip_new_spi_device(const led_strip_config_t *led_config, const l // clock_resolution between 2.2MHz to 2.8MHz is supported ESP_GOTO_ON_FALSE((clock_resolution_khz < LED_STRIP_SPI_DEFAULT_RESOLUTION / 1000 + 300) && (clock_resolution_khz > LED_STRIP_SPI_DEFAULT_RESOLUTION / 1000 - 300), ESP_ERR_NOT_SUPPORTED, err, TAG, "unsupported clock resolution:%dKHz", clock_resolution_khz); - ESP_GOTO_ON_ERROR(led_strip_set_color_order(spi_strip->led_pixel_offset, led_config->pixel_order, bytes_per_pixel), err, TAG, "adjust color order failed"); + ESP_GOTO_ON_ERROR(led_strip_set_color_order(spi_strip->led_pixel_offset, led_config->color_component_order, num_color_components), + err, TAG, "set color component order failed"); spi_strip->bytes_per_pixel = bytes_per_pixel; spi_strip->strip_len = led_config->max_leds;