diff options
| author | Joshua Diamond <josh@windowoffire.com> | 2020-06-19 22:59:28 -0400 |
|---|---|---|
| committer | GitHub <noreply@github.com> | 2020-06-20 12:59:28 +1000 |
| commit | 18e561b82cdb71bed0162b3ce4083a2d269e5336 (patch) | |
| tree | 6e016ac5a1e8cbca871b04646e477307dc366280 /drivers | |
| parent | c7851484457e5ab604962ae7c914b863b8de0cb2 (diff) | |
| download | qmk_firmware-18e561b82cdb71bed0162b3ce4083a2d269e5336.tar.gz qmk_firmware-18e561b82cdb71bed0162b3ce4083a2d269e5336.zip | |
Fix incorrect delay when setting WS2812 (and similar) leds (#9302)
* Fix incorrect delay when setting WS2812 (and similar) leds
* Add documentation for WS2812_DELAY_MICROSECONDS
* Remove improper cast to uint8_t
Co-authored-by: Sergey Vlasov <sigprof@gmail.com>
* Remove unneeded cast to uint8_t and correct math
Co-authored-by: Sergey Vlasov <sigprof@gmail.com>
* microseconds -> µs
Co-authored-by: Ryan <fauxpark@gmail.com>
* Make documentation better match the spec sheet.
Co-authored-by: Ryan <fauxpark@gmail.com>
* Rename macro to match spec sheet
* Further correction to the delay maths for the SPI case.
Co-authored-by: Joel Challis <git@zvecr.com>
* Move ws2812_common.h to the drivers directory
* Revert "Further correction to the delay maths for the SPI case."
This reverts commit e61b56a2cfc7dfec9992a7a3af92afa50e5b8ec0.
* Remove ws2812_setleds_pin(); consolidate ws2812.h
Co-authored-by: Sergey Vlasov <sigprof@gmail.com>
Co-authored-by: Ryan <fauxpark@gmail.com>
Co-authored-by: Joel Challis <git@zvecr.com>
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/avr/ws2812.c | 20 | ||||
| -rw-r--r-- | drivers/chibios/ws2812.c | 2 | ||||
| -rw-r--r-- | drivers/chibios/ws2812.h | 16 | ||||
| -rw-r--r-- | drivers/chibios/ws2812_pwm.c | 7 | ||||
| -rw-r--r-- | drivers/chibios/ws2812_spi.c | 2 | ||||
| -rw-r--r-- | drivers/ws2812.h (renamed from drivers/avr/ws2812.h) | 18 |
6 files changed, 19 insertions, 46 deletions
diff --git a/drivers/avr/ws2812.c b/drivers/avr/ws2812.c index 5c3d72dcb..dd2ef8991 100644 --- a/drivers/avr/ws2812.c +++ b/drivers/avr/ws2812.c | |||
| @@ -36,25 +36,15 @@ | |||
| 36 | 36 | ||
| 37 | static inline void ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t masklo, uint8_t maskhi); | 37 | static inline void ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t masklo, uint8_t maskhi); |
| 38 | 38 | ||
| 39 | // Setleds for standard RGB | 39 | void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds) { |
| 40 | void inline ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds) { | 40 | DDRx_ADDRESS(RGB_DI_PIN) |= pinmask(RGB_DI_PIN); |
| 41 | // wrap up usage of RGB_DI_PIN | ||
| 42 | ws2812_setleds_pin(ledarray, number_of_leds, RGB_DI_PIN); | ||
| 43 | } | ||
| 44 | |||
| 45 | void ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t number_of_leds, uint8_t pin) { | ||
| 46 | DDRx_ADDRESS(RGB_DI_PIN) |= pinmask(pin); | ||
| 47 | 41 | ||
| 48 | uint8_t masklo = ~(pinmask(pin)) & PORTx_ADDRESS(pin); | 42 | uint8_t masklo = ~(pinmask(RGB_DI_PIN)) & PORTx_ADDRESS(RGB_DI_PIN); |
| 49 | uint8_t maskhi = pinmask(pin) | PORTx_ADDRESS(pin); | 43 | uint8_t maskhi = pinmask(RGB_DI_PIN) | PORTx_ADDRESS(RGB_DI_PIN); |
| 50 | 44 | ||
| 51 | ws2812_sendarray_mask((uint8_t *)ledarray, number_of_leds * sizeof(LED_TYPE), masklo, maskhi); | 45 | ws2812_sendarray_mask((uint8_t *)ledarray, number_of_leds * sizeof(LED_TYPE), masklo, maskhi); |
| 52 | 46 | ||
| 53 | #ifdef RGBW | 47 | _delay_us(WS2812_TRST_US); |
| 54 | _delay_us(80); | ||
| 55 | #else | ||
| 56 | _delay_us(50); | ||
| 57 | #endif | ||
| 58 | } | 48 | } |
| 59 | 49 | ||
| 60 | /* | 50 | /* |
diff --git a/drivers/chibios/ws2812.c b/drivers/chibios/ws2812.c index 2c2d9fb2d..5917b7f0e 100644 --- a/drivers/chibios/ws2812.c +++ b/drivers/chibios/ws2812.c | |||
| @@ -51,7 +51,7 @@ | |||
| 51 | 51 | ||
| 52 | // The reset gap can be 6000 ns, but depending on the LED strip it may have to be increased | 52 | // The reset gap can be 6000 ns, but depending on the LED strip it may have to be increased |
| 53 | // to values like 600000 ns. If it is too small, the pixels will show nothing most of the time. | 53 | // to values like 600000 ns. If it is too small, the pixels will show nothing most of the time. |
| 54 | #define RES 10000 // Width of the low gap between bits to cause a frame to latch | 54 | #define RES (1000 * WS2812_TRST_US) // Width of the low gap between bits to cause a frame to latch |
| 55 | 55 | ||
| 56 | void sendByte(uint8_t byte) { | 56 | void sendByte(uint8_t byte) { |
| 57 | // WS2812 protocol wants most significant bits first | 57 | // WS2812 protocol wants most significant bits first |
diff --git a/drivers/chibios/ws2812.h b/drivers/chibios/ws2812.h deleted file mode 100644 index 41c22a00b..000000000 --- a/drivers/chibios/ws2812.h +++ /dev/null | |||
| @@ -1,16 +0,0 @@ | |||
| 1 | #pragma once | ||
| 2 | |||
| 3 | #include "quantum/color.h" | ||
| 4 | |||
| 5 | /* User Interface | ||
| 6 | * | ||
| 7 | * Input: | ||
| 8 | * ledarray: An array of GRB data describing the LED colors | ||
| 9 | * number_of_leds: The number of LEDs to write | ||
| 10 | * | ||
| 11 | * The functions will perform the following actions: | ||
| 12 | * - Set the data-out pin as output | ||
| 13 | * - Send out the LED data | ||
| 14 | * - Wait 50us to reset the LEDs | ||
| 15 | */ | ||
| 16 | void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds); | ||
diff --git a/drivers/chibios/ws2812_pwm.c b/drivers/chibios/ws2812_pwm.c index ba45d0042..7113db11e 100644 --- a/drivers/chibios/ws2812_pwm.c +++ b/drivers/chibios/ws2812_pwm.c | |||
| @@ -53,11 +53,10 @@ | |||
| 53 | /** | 53 | /** |
| 54 | * @brief Number of bit-periods to hold the data line low at the end of a frame | 54 | * @brief Number of bit-periods to hold the data line low at the end of a frame |
| 55 | * | 55 | * |
| 56 | * The reset period for each frame must be at least 50 uS; so we add in 50 bit-times | 56 | * The reset period for each frame is defined in WS2812_TRST_US. |
| 57 | * of zeroes at the end. (50 bits)*(1.25 uS/bit) = 62.5 uS, which gives us some | 57 | * Calculate the number of zeroes to add at the end assuming 1.25 uS/bit: |
| 58 | * slack in the timing requirements | ||
| 59 | */ | 58 | */ |
| 60 | #define WS2812_RESET_BIT_N (50) | 59 | #define WS2812_RESET_BIT_N (1000 * WS2812_TRST_US / 1250) |
| 61 | #define WS2812_COLOR_BIT_N (RGBLED_NUM * 24) /**< Number of data bits */ | 60 | #define WS2812_COLOR_BIT_N (RGBLED_NUM * 24) /**< Number of data bits */ |
| 62 | #define WS2812_BIT_N (WS2812_COLOR_BIT_N + WS2812_RESET_BIT_N) /**< Total number of bits in a frame */ | 61 | #define WS2812_BIT_N (WS2812_COLOR_BIT_N + WS2812_RESET_BIT_N) /**< Total number of bits in a frame */ |
| 63 | 62 | ||
diff --git a/drivers/chibios/ws2812_spi.c b/drivers/chibios/ws2812_spi.c index 3bbada7fe..7a1d2f05d 100644 --- a/drivers/chibios/ws2812_spi.c +++ b/drivers/chibios/ws2812_spi.c | |||
| @@ -36,7 +36,7 @@ | |||
| 36 | #define NB_COLORS 3 | 36 | #define NB_COLORS 3 |
| 37 | #define BYTES_FOR_LED (BYTES_FOR_LED_BYTE * NB_COLORS) | 37 | #define BYTES_FOR_LED (BYTES_FOR_LED_BYTE * NB_COLORS) |
| 38 | #define DATA_SIZE (BYTES_FOR_LED * RGBLED_NUM) | 38 | #define DATA_SIZE (BYTES_FOR_LED * RGBLED_NUM) |
| 39 | #define RESET_SIZE 200 | 39 | #define RESET_SIZE (1000 * WS2812_TRST_US / (2 * 1250)) |
| 40 | #define PREAMBLE_SIZE 4 | 40 | #define PREAMBLE_SIZE 4 |
| 41 | 41 | ||
| 42 | static uint8_t txbuf[PREAMBLE_SIZE + DATA_SIZE + RESET_SIZE] = {0}; | 42 | static uint8_t txbuf[PREAMBLE_SIZE + DATA_SIZE + RESET_SIZE] = {0}; |
diff --git a/drivers/avr/ws2812.h b/drivers/ws2812.h index 88eb08189..370b14f3e 100644 --- a/drivers/avr/ws2812.h +++ b/drivers/ws2812.h | |||
| @@ -1,11 +1,4 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * light weight WS2812 lib include | ||
| 3 | * | ||
| 4 | * Version 2.3 - Nev 29th 2015 | ||
| 5 | * Author: Tim (cpldcpu@gmail.com) | ||
| 6 | * | ||
| 7 | * Please do not change this file! All configuration is handled in "ws2812_config.h" | ||
| 8 | * | ||
| 9 | * This program is free software: you can redistribute it and/or modify | 2 | * This program is free software: you can redistribute it and/or modify |
| 10 | * it under the terms of the GNU General Public License as published by | 3 | * it under the terms of the GNU General Public License as published by |
| 11 | * the Free Software Foundation, either version 2 of the License, or | 4 | * the Free Software Foundation, either version 2 of the License, or |
| @@ -24,12 +17,20 @@ | |||
| 24 | 17 | ||
| 25 | #include "quantum/color.h" | 18 | #include "quantum/color.h" |
| 26 | 19 | ||
| 20 | /* | ||
| 21 | * Older WS2812s can handle a reset time (TRST) of 50us, but recent | ||
| 22 | * component revisions require a minimum of 280us. | ||
| 23 | */ | ||
| 24 | |||
| 25 | #if !defined(WS2812_TRST_US) | ||
| 26 | #define WS2812_TRST_US 280 | ||
| 27 | #endif | ||
| 28 | |||
| 27 | /* User Interface | 29 | /* User Interface |
| 28 | * | 30 | * |
| 29 | * Input: | 31 | * Input: |
| 30 | * ledarray: An array of GRB data describing the LED colors | 32 | * ledarray: An array of GRB data describing the LED colors |
| 31 | * number_of_leds: The number of LEDs to write | 33 | * number_of_leds: The number of LEDs to write |
| 32 | * pin (optional): A pin_t definition for the line to drive | ||
| 33 | * | 34 | * |
| 34 | * The functions will perform the following actions: | 35 | * The functions will perform the following actions: |
| 35 | * - Set the data-out pin as output | 36 | * - Set the data-out pin as output |
| @@ -37,4 +38,3 @@ | |||
| 37 | * - Wait 50us to reset the LEDs | 38 | * - Wait 50us to reset the LEDs |
| 38 | */ | 39 | */ |
| 39 | void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds); | 40 | void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds); |
| 40 | void ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t number_of_leds, uint8_t pin); | ||
