mirror of
https://github.com/mfulz/qmk_firmware.git
synced 2025-10-24 11:09:57 +02:00
Fix AVR ws2812 when ADDRESS_BASE is non zero (#8646)
* Fix AVR ws2812 when ADDRESS_BASE is non zero * fix port * remove unused function defs
This commit is contained in:
parent
2b427f774a
commit
31fd0cbc1c
@ -20,12 +20,13 @@
|
|||||||
* You should have received a copy of the GNU General Public License
|
* You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include "ws2812.h"
|
#include "ws2812.h"
|
||||||
#include <avr/interrupt.h>
|
#include <avr/interrupt.h>
|
||||||
#include <avr/io.h>
|
#include <avr/io.h>
|
||||||
#include <util/delay.h>
|
#include <util/delay.h>
|
||||||
|
|
||||||
|
#define pinmask(pin) (_BV((pin)&0xF))
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Forward declare internal functions
|
* Forward declare internal functions
|
||||||
*
|
*
|
||||||
@ -33,20 +34,21 @@
|
|||||||
* The length is the number of bytes to send - three per LED.
|
* The length is the number of bytes to send - three per LED.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
void ws2812_sendarray(uint8_t *array, uint16_t length);
|
static inline void ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t masklo, uint8_t maskhi);
|
||||||
void ws2812_sendarray_mask(uint8_t *array, uint16_t length, uint8_t pinmask);
|
|
||||||
|
|
||||||
// Setleds for standard RGB
|
// Setleds for standard RGB
|
||||||
void inline ws2812_setleds(LED_TYPE *ledarray, uint16_t leds) {
|
void inline ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds) {
|
||||||
// ws2812_setleds_pin(ledarray,leds, _BV(ws2812_pin));
|
// wrap up usage of RGB_DI_PIN
|
||||||
ws2812_setleds_pin(ledarray, leds, _BV(RGB_DI_PIN & 0xF));
|
ws2812_setleds_pin(ledarray, number_of_leds, RGB_DI_PIN);
|
||||||
}
|
}
|
||||||
|
|
||||||
void inline ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t leds, uint8_t pinmask) {
|
void ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t number_of_leds, uint8_t pin) {
|
||||||
// new universal format (DDR)
|
DDRx_ADDRESS(RGB_DI_PIN) |= pinmask(pin);
|
||||||
_SFR_IO8((RGB_DI_PIN >> 4) + 1) |= pinmask;
|
|
||||||
|
|
||||||
ws2812_sendarray_mask((uint8_t *)ledarray, leds * sizeof(LED_TYPE), pinmask);
|
uint8_t masklo = ~(pinmask(pin)) & PORTx_ADDRESS(pin);
|
||||||
|
uint8_t maskhi = pinmask(pin) | PORTx_ADDRESS(pin);
|
||||||
|
|
||||||
|
ws2812_sendarray_mask((uint8_t *)ledarray, number_of_leds * sizeof(LED_TYPE), masklo, maskhi);
|
||||||
|
|
||||||
#ifdef RGBW
|
#ifdef RGBW
|
||||||
_delay_us(80);
|
_delay_us(80);
|
||||||
@ -55,8 +57,6 @@ void inline ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t leds, uint8_t pinmas
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void ws2812_sendarray(uint8_t *data, uint16_t datlen) { ws2812_sendarray_mask(data, datlen, _BV(RGB_DI_PIN & 0xF)); }
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This routine writes an array of bytes with RGB values to the Dataout pin
|
This routine writes an array of bytes with RGB values to the Dataout pin
|
||||||
using the fast 800kHz clockless WS2811/2812 protocol.
|
using the fast 800kHz clockless WS2811/2812 protocol.
|
||||||
@ -118,14 +118,9 @@ void ws2812_sendarray(uint8_t *data, uint16_t datlen) { ws2812_sendarray_mask(da
|
|||||||
#define w_nop8 w_nop4 w_nop4
|
#define w_nop8 w_nop4 w_nop4
|
||||||
#define w_nop16 w_nop8 w_nop8
|
#define w_nop16 w_nop8 w_nop8
|
||||||
|
|
||||||
void inline ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t maskhi) {
|
static inline void ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t masklo, uint8_t maskhi) {
|
||||||
uint8_t curbyte, ctr, masklo;
|
uint8_t curbyte, ctr, sreg_prev;
|
||||||
uint8_t sreg_prev;
|
|
||||||
|
|
||||||
// masklo =~maskhi&ws2812_PORTREG;
|
|
||||||
// maskhi |= ws2812_PORTREG;
|
|
||||||
masklo = ~maskhi & _SFR_IO8((RGB_DI_PIN >> 4) + 2);
|
|
||||||
maskhi |= _SFR_IO8((RGB_DI_PIN >> 4) + 2);
|
|
||||||
sreg_prev = SREG;
|
sreg_prev = SREG;
|
||||||
cli();
|
cli();
|
||||||
|
|
||||||
@ -188,7 +183,7 @@ void inline ws2812_sendarray_mask(uint8_t *data, uint16_t datlen, uint8_t maskhi
|
|||||||
" dec %0 \n\t" // '1' [+2] '0' [+2]
|
" dec %0 \n\t" // '1' [+2] '0' [+2]
|
||||||
" brne loop%=\n\t" // '1' [+3] '0' [+4]
|
" brne loop%=\n\t" // '1' [+3] '0' [+4]
|
||||||
: "=&d"(ctr)
|
: "=&d"(ctr)
|
||||||
: "r"(curbyte), "I"(_SFR_IO_ADDR(_SFR_IO8((RGB_DI_PIN >> 4) + 2))), "r"(maskhi), "r"(masklo));
|
: "r"(curbyte), "I"(_SFR_IO_ADDR(PORTx_ADDRESS(RGB_DI_PIN))), "r"(maskhi), "r"(masklo));
|
||||||
}
|
}
|
||||||
|
|
||||||
SREG = sreg_prev;
|
SREG = sreg_prev;
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
* Input:
|
* Input:
|
||||||
* ledarray: An array of GRB data describing the LED colors
|
* ledarray: An array of GRB data describing the LED colors
|
||||||
* number_of_leds: The number of LEDs to write
|
* number_of_leds: The number of LEDs to write
|
||||||
* pinmask (optional): Bitmask describing the output bin. e.g. _BV(PB0)
|
* pin (optional): A pin_t definition for the line to drive
|
||||||
*
|
*
|
||||||
* The functions will perform the following actions:
|
* The functions will perform the following actions:
|
||||||
* - Set the data-out pin as output
|
* - Set the data-out pin as output
|
||||||
@ -37,4 +37,4 @@
|
|||||||
* - Wait 50us to reset the LEDs
|
* - Wait 50us to reset the LEDs
|
||||||
*/
|
*/
|
||||||
void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds);
|
void ws2812_setleds(LED_TYPE *ledarray, uint16_t number_of_leds);
|
||||||
void ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t number_of_leds, uint8_t pinmask);
|
void ws2812_setleds_pin(LED_TYPE *ledarray, uint16_t number_of_leds, uint8_t pin);
|
||||||
|
@ -27,19 +27,6 @@
|
|||||||
|
|
||||||
extern rgblight_config_t rgblight_config;
|
extern rgblight_config_t rgblight_config;
|
||||||
|
|
||||||
/*
|
|
||||||
* Forward declare internal functions
|
|
||||||
*
|
|
||||||
* The functions take a byte-array and send to the data output as WS2812 bitstream.
|
|
||||||
* The length is the number of bytes to send - three per LED.
|
|
||||||
*/
|
|
||||||
|
|
||||||
void ws2812_sendarray(uint8_t *array, uint16_t length);
|
|
||||||
void ws2812_sendarray_mask(uint8_t *array, uint16_t length, uint8_t pinmask);
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void rgblight_set(void) {
|
void rgblight_set(void) {
|
||||||
if (!rgblight_config.enable) {
|
if (!rgblight_config.enable) {
|
||||||
for (uint8_t i = 0; i < RGBLED_NUM; i++) {
|
for (uint8_t i = 0; i < RGBLED_NUM; i++) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user