From 8852374738b2a52a1e0883ff91b331dd2b5676b2 Mon Sep 17 00:00:00 2001 From: Phillip Burgess Date: Mon, 10 Aug 2015 20:43:48 -0700 Subject: [PATCH] Initial RGBW support --- Adafruit_NeoPixel.cpp | 160 ++++++++++++++++++++++++++---------------- Adafruit_NeoPixel.h | 118 +++++++++++++++++++++++++------ esp8266.c | 4 +- library.properties | 2 +- 4 files changed, 199 insertions(+), 85 deletions(-) diff --git a/Adafruit_NeoPixel.cpp b/Adafruit_NeoPixel.cpp index f0b851c5..2f6106f8 100644 --- a/Adafruit_NeoPixel.cpp +++ b/Adafruit_NeoPixel.cpp @@ -2,7 +2,7 @@ Arduino library to control a wide variety of WS2811- and WS2812-based RGB LED devices such as Adafruit FLORA RGB Smart Pixels and NeoPixel strips. Currently handles 400 and 800 KHz bitstreams on 8, 12 and 16 MHz ATmega - MCUs, with LEDs wired for RGB or GRB color order. 8 MHz MCUs provide + MCUs, with LEDs wired for various color orders. 8 MHz MCUs provide output on PORTB and PORTD, while 16 MHz chips can handle most output pins (possible exception with upper PORT registers on the Arduino Mega). @@ -35,22 +35,27 @@ #include "Adafruit_NeoPixel.h" // Constructor when length, pin and type are known at compile-time: -Adafruit_NeoPixel::Adafruit_NeoPixel(uint16_t n, uint8_t p, uint8_t t) : +Adafruit_NeoPixel::Adafruit_NeoPixel(uint16_t n, uint8_t p, neoPixelType t) : brightness(0), pixels(NULL), endTime(0), begun(false) { + updateType(t); updateLength(n); setPin(p); - updateType(t); } // via Michael Vogt/neophob: empty constructor is used when strand length // isn't known at compile-time; situations where program config might be // read from internal flash memory or an SD card, or arrive via serial -// command. If using this constructor, MUST follow up with updateLength(), -// etc. to establish the strand length, type and pin number! +// command. If using this constructor, MUST follow up with updateType(), +// updateLength(), etc. to establish the strand type, length and pin number! Adafruit_NeoPixel::Adafruit_NeoPixel() : pin(-1), brightness(0), pixels(NULL), endTime(0), begun(false), - numLEDs(0), numBytes(0), type(NEO_GRB + NEO_KHZ800) { } + numLEDs(0), numBytes(0), rOffset(1), gOffset(0), bOffset(2), wOffset(1) +#ifdef NEO_KHZ400 + , is800KHz(true) +#endif +{ +} Adafruit_NeoPixel::~Adafruit_NeoPixel() { if(pixels) free(pixels); @@ -69,7 +74,7 @@ void Adafruit_NeoPixel::updateLength(uint16_t n) { if(pixels) free(pixels); // Free existing data (if any) // Allocate new data -- note: ALL PIXELS ARE CLEARED - numBytes = n * 3; + numBytes = n * ((wOffset == rOffset) ? 3 : 4); if((pixels = (uint8_t *)malloc(numBytes))) { memset(pixels, 0, numBytes); numLEDs = n; @@ -78,28 +83,25 @@ void Adafruit_NeoPixel::updateLength(uint16_t n) { } } -void Adafruit_NeoPixel::updateType(uint8_t t) { - type = t; - if(t & NEO_GRB) { // GRB vs RGB; might add others if needed - rOffset = 1; - gOffset = 0; - bOffset = 2; - } else if (t & NEO_BRG) { - rOffset = 1; - gOffset = 2; - bOffset = 0; - } else if (t & NEO_RBG) { - rOffset = 0; - gOffset = 2; - bOffset = 1; - } else { - rOffset = 0; - gOffset = 1; - bOffset = 2; +void Adafruit_NeoPixel::updateType(neoPixelType t) { + boolean oldThreeBytesPerPixel = (wOffset == rOffset); // false if RGBW + + wOffset = (t >> 6) & 0b11; // See notes in header file + rOffset = (t >> 4) & 0b11; // regarding R/G/B/W offsets + gOffset = (t >> 2) & 0b11; + bOffset = t & 0b11; +#ifdef NEO_KHZ400 + is800KHz = (t < 256); // 400 KHz flag is 1<<8 +#endif + + // If bytes-per-pixel has changed (and pixel data was previously + // allocated), re-allocate to new size. Will clear any data. + if(pixels) { + boolean newThreeBytesPerPixel = (wOffset == rOffset); + if(newThreeBytesPerPixel != oldThreeBytesPerPixel) updateLength(numLEDs); } } - #ifdef ESP8266 // ESP8266 show() is external to enforce ICACHE_RAM_ATTR execution extern "C" void ICACHE_RAM_ATTR espShow( @@ -162,8 +164,8 @@ void Adafruit_NeoPixel::show(void) { // 8 MHz(ish) AVR --------------------------------------------------------- #if (F_CPU >= 7400000UL) && (F_CPU <= 9500000UL) -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif volatile uint8_t n1, n2 = 0; // First, next bits out @@ -420,8 +422,8 @@ void Adafruit_NeoPixel::show(void) { // 12 MHz(ish) AVR -------------------------------------------------------- #elif (F_CPU >= 11100000UL) && (F_CPU <= 14300000UL) -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif // In the 12 MHz case, an optimized 800 KHz datastream (no dead time @@ -600,8 +602,8 @@ void Adafruit_NeoPixel::show(void) { // 16 MHz(ish) AVR -------------------------------------------------------- #elif (F_CPU >= 15400000UL) && (F_CPU <= 19000000L) -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif // WS2811 and WS2812 have different hi/lo duty cycles; this is @@ -737,8 +739,8 @@ void Adafruit_NeoPixel::show(void) { ARM_DEMCR |= ARM_DEMCR_TRCENA; ARM_DWT_CTRL |= ARM_DWT_CTRL_CYCCNTENA; -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif cyc = ARM_DWT_CYCCNT + CYCLES_800; while(p < end) { @@ -881,8 +883,8 @@ void Adafruit_NeoPixel::show(void) { volatile uint32_t *set = &(PORT->Group[portNum].OUTSET.reg), *clr = &(PORT->Group[portNum].OUTCLR.reg); -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif for(;;) { *set = pinMask; @@ -973,8 +975,8 @@ void Adafruit_NeoPixel::show(void) { pix = *p++; mask = 0x80; -#ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream +#ifdef NEO_KHZ400 // 800 KHz check needed only if 400 KHz support enabled + if(is800KHz) { #endif time0 = TIME_800_0; time1 = TIME_800_1; @@ -1013,7 +1015,7 @@ void Adafruit_NeoPixel::show(void) { // ESP8266 ---------------------------------------------------------------- // ESP8266 show() is external to enforce ICACHE_RAM_ATTR execution - espShow(pin, pixels, numBytes, type); + espShow(pin, pixels, numBytes, is800KHz); #endif // ESP8266 @@ -1044,14 +1046,21 @@ void Adafruit_NeoPixel::setPin(uint8_t p) { // Set pixel color from separate R,G,B components: void Adafruit_NeoPixel::setPixelColor( uint16_t n, uint8_t r, uint8_t g, uint8_t b) { + if(n < numLEDs) { if(brightness) { // See notes in setBrightness() r = (r * brightness) >> 8; g = (g * brightness) >> 8; b = (b * brightness) >> 8; } - uint8_t *p = &pixels[n * 3]; - p[rOffset] = r; + uint8_t *p; + if(wOffset == rOffset) { // Is an RGB-type strip + p = &pixels[n * 3]; // 3 bytes per pixel + } else { // Is a WRGB-type strip + p = &pixels[n * 4]; // 4 bytes per pixel + p[wOffset] = 0; // But only R,G,B passed -- set W to 0 + } + p[rOffset] = r; // R,G,B always stored p[gOffset] = g; p[bOffset] = b; } @@ -1060,7 +1069,7 @@ void Adafruit_NeoPixel::setPixelColor( // Set pixel color from 'packed' 32-bit RGB color: void Adafruit_NeoPixel::setPixelColor(uint16_t n, uint32_t c) { if(n < numLEDs) { - uint8_t + uint8_t *p, r = (uint8_t)(c >> 16), g = (uint8_t)(c >> 8), b = (uint8_t)c; @@ -1069,7 +1078,13 @@ void Adafruit_NeoPixel::setPixelColor(uint16_t n, uint32_t c) { g = (g * brightness) >> 8; b = (b * brightness) >> 8; } - uint8_t *p = &pixels[n * 3]; + if(wOffset == rOffset) { + p = &pixels[n * 3]; + } else { + p = &pixels[n * 4]; + uint8_t w = (uint8_t)(c >> 24); + p[wOffset] = brightness ? ((w * brightness) >> 8) : w; + } p[rOffset] = r; p[gOffset] = g; p[bOffset] = b; @@ -1082,31 +1097,54 @@ uint32_t Adafruit_NeoPixel::Color(uint8_t r, uint8_t g, uint8_t b) { return ((uint32_t)r << 16) | ((uint32_t)g << 8) | b; } +// Convert separate R,G,B,W into packed 32-bit WRGB color. +// Packed format is always WRGB, regardless of LED strand color order. +uint32_t Adafruit_NeoPixel::Color(uint8_t r, uint8_t g, uint8_t b, uint8_t w) { + return ((uint32_t)w << 24) | ((uint32_t)r << 16) | ((uint32_t)g << 8) | b; +} + // Query color from previously-set pixel (returns packed 32-bit RGB value) uint32_t Adafruit_NeoPixel::getPixelColor(uint16_t n) const { - if(n >= numLEDs) { - // Out of bounds, return no color. - return 0; - } - uint8_t *p = &pixels[n * 3]; - uint32_t c = ((uint32_t)p[rOffset] << 16) | - ((uint32_t)p[gOffset] << 8) | - (uint32_t)p[bOffset]; - // Adjust this back up to the true color, as setting a pixel color will - // scale it back down again. - if(brightness) { // See notes in setBrightness() - //Cast the color to a byte array - uint8_t * c_ptr =reinterpret_cast(&c); - c_ptr[0] = (c_ptr[0] << 8)/brightness; - c_ptr[1] = (c_ptr[1] << 8)/brightness; - c_ptr[2] = (c_ptr[2] << 8)/brightness; + if(n >= numLEDs) return 0; // Out of bounds, return no color. + + uint8_t *p; + + if(wOffset == rOffset) { // Is RGB-type device + p = &pixels[n * 3]; + if(brightness) { + // Stored color was decimated by setBrightness(). Returned value + // attempts to scale back to an approximation of the original 24-bit + // value used when setting the pixel color, but there will always be + // some error -- those bits are simply gone. Issue is most + // pronounced at low brightness levels. + return (((uint32_t)(p[rOffset] << 8) / brightness) << 16) | + (((uint32_t)(p[gOffset] << 8) / brightness) << 8) | + ( (uint32_t)(p[bOffset] << 8) / brightness ); + } else { + // No brightness adjustment has been made -- return 'raw' color + return ((uint32_t)p[rOffset] << 16) | + ((uint32_t)p[gOffset] << 8) | + (uint32_t)p[bOffset]; + } + } else { // Is RGBW-type device + p = &pixels[n * 4]; + if(brightness) { // Return scaled color + return (((uint32_t)(p[wOffset] << 8) / brightness) << 24) | + (((uint32_t)(p[rOffset] << 8) / brightness) << 16) | + (((uint32_t)(p[gOffset] << 8) / brightness) << 8) | + ( (uint32_t)(p[bOffset] << 8) / brightness ); + } else { // Return raw color + return ((uint32_t)p[wOffset] << 24) | + ((uint32_t)p[rOffset] << 16) | + ((uint32_t)p[gOffset] << 8) | + (uint32_t)p[bOffset]; + } } - return c; // Pixel # is out of bounds } // Returns pointer to pixels[] array. Pixel data is stored in device- // native format and is not translated here. Application will need to be -// aware whether pixels are RGB vs. GRB and handle colors appropriately. +// aware of specific pixel data format and handle colors appropriately. uint8_t *Adafruit_NeoPixel::getPixels(void) const { return pixels; } diff --git a/Adafruit_NeoPixel.h b/Adafruit_NeoPixel.h index f4b5fa5d..c32e3bf0 100644 --- a/Adafruit_NeoPixel.h +++ b/Adafruit_NeoPixel.h @@ -26,20 +26,91 @@ #include #endif -// 'type' flags for LED pixels (third parameter to constructor): -#define NEO_RGB 0x00 // Wired for RGB data order -#define NEO_GRB 0x01 // Wired for GRB data order -#define NEO_BRG 0x04 -#define NEO_RBG 0x08 - -#define NEO_COLMASK 0x01 -#define NEO_KHZ800 0x02 // 800 KHz datastream -#define NEO_SPDMASK 0x02 -// Trinket flash space is tight, v1 NeoPixels aren't handled by default. -// Remove the ifndef/endif to add support -- but code will be bigger. -// Conversely, can comment out the #defines to save space on other MCUs. +// The order of primary colors in the NeoPixel data stream can vary +// among device types, manufacturers and even different revisions of +// the same item. The third parameter to the Adafruit_NeoPixel +// constructor encodes the per-pixel byte offsets of the red, green +// and blue primaries (plus white, if present) in the data stream -- +// the following #defines provide an easier-to-use named version for +// each permutation. e.g. NEO_GRB indicates a NeoPixel-compatible +// device expecting three bytes per pixel, with the first byte +// containing the green value, second containing red and third +// containing blue. The in-memory representation of a chain of +// NeoPixels is the same as the data-stream order; no re-ordering of +// bytes is required when issuing data to the chain. + +// Bits 5,4 of this value are the offset (0-3) from the first byte of +// a pixel to the location of the red color byte. Bits 3,2 are the +// green offset and 1,0 are the blue offset. If it is an RGBW-type +// device (supporting a white primary in addition to R,G,B), bits 7,6 +// are the offset to the white byte...otherwise, bits 7,6 are set to +// the same value as 5,4 (red) to indicate an RGB (not RGBW) device. +// i.e. binary representation: +// 0bWWRRGGBB for RGBW devices +// 0bRRRRGGBB for RGB + +// RGB NeoPixel permutations; white and red offsets are always same +// Offset: W R G B +#define NEO_RGB ((0 << 6) | (0 << 4) | (1 << 2) | (2)) +#define NEO_RBG ((0 << 6) | (0 << 4) | (2 << 2) | (1)) +#define NEO_GRB ((1 << 6) | (1 << 4) | (0 << 2) | (2)) +#define NEO_GBR ((2 << 6) | (2 << 4) | (0 << 2) | (1)) +#define NEO_BRG ((1 << 6) | (1 << 4) | (2 << 2) | (0)) +#define NEO_BGR ((2 << 6) | (2 << 4) | (1 << 2) | (0)) + +// RGBW NeoPixel permutations; all 4 offsets are distinct +// Offset: W R G B +#define NEO_WRGB ((0 << 6) | (1 << 4) | (2 << 2) | (3)) +#define NEO_WRBG ((0 << 6) | (1 << 4) | (3 << 2) | (2)) +#define NEO_WGRB ((0 << 6) | (2 << 4) | (1 << 2) | (3)) +#define NEO_WGBR ((0 << 6) | (3 << 4) | (1 << 2) | (2)) +#define NEO_WBRG ((0 << 6) | (2 << 4) | (3 << 2) | (1)) +#define NEO_WBGR ((0 << 6) | (3 << 4) | (2 << 2) | (1)) + +#define NEO_RWGB ((1 << 6) | (0 << 4) | (2 << 2) | (3)) +#define NEO_RWBG ((1 << 6) | (0 << 4) | (3 << 2) | (2)) +#define NEO_RGWB ((2 << 6) | (0 << 4) | (1 << 2) | (3)) +#define NEO_RGBW ((3 << 6) | (0 << 4) | (1 << 2) | (2)) +#define NEO_RBWG ((2 << 6) | (0 << 4) | (3 << 2) | (1)) +#define NEO_RBGW ((3 << 6) | (0 << 4) | (2 << 2) | (1)) + +#define NEO_GWRB ((1 << 6) | (2 << 4) | (0 << 2) | (3)) +#define NEO_GWBR ((1 << 6) | (3 << 4) | (0 << 2) | (2)) +#define NEO_GRWB ((2 << 6) | (1 << 4) | (0 << 2) | (3)) +#define NEO_GRBW ((3 << 6) | (1 << 4) | (0 << 2) | (2)) +#define NEO_GBWR ((2 << 6) | (3 << 4) | (0 << 2) | (1)) +#define NEO_GBRW ((3 << 6) | (2 << 4) | (0 << 2) | (1)) + +#define NEO_BWRG ((1 << 6) | (2 << 4) | (3 << 2) | (0)) +#define NEO_BWGR ((1 << 6) | (3 << 4) | (2 << 2) | (0)) +#define NEO_BRWG ((2 << 6) | (1 << 4) | (3 << 2) | (0)) +#define NEO_BRGW ((3 << 6) | (1 << 4) | (2 << 2) | (0)) +#define NEO_BGWR ((2 << 6) | (3 << 4) | (1 << 2) | (0)) +#define NEO_BGRW ((3 << 6) | (2 << 4) | (1 << 2) | (0)) + +// Add NEO_KHZ400 to the color order value to indicate a 400 KHz +// device. All but the earliest v1 NeoPixels expect an 800 KHz data +// stream, this is the default if unspecified. Because flash space +// is very limited on ATtiny devices (e.g. Trinket, Gemma), v1 +// NeoPixels aren't handled by default on those chips, though it can +// be enabled by removing the ifndef/endif below -- but code will be +// bigger. Conversely, can disable the NEO_KHZ400 line on other MCUs +// to remove v1 support and save a little space. + +#define NEO_KHZ800 0x0000 // 800 KHz datastream #ifndef __AVR_ATtiny85__ -#define NEO_KHZ400 0x00 // 400 KHz datastream +#define NEO_KHZ400 0x0100 // 400 KHz datastream +#endif + +// If 400 KHz support is enabled, the third parameter to the constructor +// requires a 16-bit value (in order to select 400 vs 800 KHz speed). +// If only 800 KHz is enabled (as is default on ATtiny), an 8-bit value +// is sufficient to encode pixel color order, saving some space. + +#ifdef NEO_KHZ400 +typedef uint8_t neoPixelType; +#else +typedef uint16_t neoPixelType; #endif class Adafruit_NeoPixel { @@ -47,7 +118,7 @@ class Adafruit_NeoPixel { public: // Constructor: number of LEDs, pin number, LED type - Adafruit_NeoPixel(uint16_t n, uint8_t p=6, uint8_t t=NEO_GRB + NEO_KHZ800); + Adafruit_NeoPixel(uint16_t n, uint8_t p=6, neoPixelType t=NEO_GRB + NEO_KHZ800); Adafruit_NeoPixel(void); ~Adafruit_NeoPixel(); @@ -56,18 +127,20 @@ class Adafruit_NeoPixel { show(void), setPin(uint8_t p), setPixelColor(uint16_t n, uint8_t r, uint8_t g, uint8_t b), + setPixelColor(uint16_t n, uint8_t r, uint8_t g, uint8_t b, uint8_t w), setPixelColor(uint16_t n, uint32_t c), setBrightness(uint8_t), clear(), updateLength(uint16_t n), - updateType(uint8_t t); + updateType(neoPixelType t); uint8_t *getPixels(void) const, getBrightness(void) const; uint16_t numPixels(void) const; static uint32_t - Color(uint8_t r, uint8_t g, uint8_t b); + Color(uint8_t r, uint8_t g, uint8_t b), + Color(uint8_t r, uint8_t g, uint8_t b, uint8_t w); uint32_t getPixelColor(uint16_t n) const; inline bool @@ -76,19 +149,22 @@ class Adafruit_NeoPixel { private: boolean +#ifdef NEO_KHZ400 // If 400 KHz NeoPixel support enabled... + is800KHz, // ...true if 800 KHz pixels +#endif begun; // true if begin() previously called uint16_t numLEDs, // Number of RGB LEDs in strip - numBytes; // Size of 'pixels' buffer below + numBytes; // Size of 'pixels' buffer below (3 or 4 bytes/pixel) int8_t pin; // Output pin number (-1 if not yet set) uint8_t - type, // Pixel flags (400 vs 800 KHz, RGB vs GRB color) brightness, - *pixels, // Holds LED color values (3 bytes each) - rOffset, // Index of red byte within each 3-byte pixel + *pixels, // Holds LED color values (3 or 4 bytes each) + rOffset, // Index of red byte within each 3- or 4-byte pixel gOffset, // Index of green byte - bOffset; // Index of blue byte + bOffset, // Index of blue byte + wOffset; // Index of white byte (same as rOffset if no white) uint32_t endTime; // Latch timing reference #ifdef __AVR__ diff --git a/esp8266.c b/esp8266.c index e457edb7..1ae3ba0f 100644 --- a/esp8266.c +++ b/esp8266.c @@ -15,7 +15,7 @@ static inline uint32_t _getCycleCount(void) { } void ICACHE_RAM_ATTR espShow( - uint8_t pin, uint8_t *pixels, uint32_t numBytes, uint8_t type) { + uint8_t pin, uint8_t *pixels, uint32_t numBytes, boolean is800KHz) { #define CYCLES_800_T0H (F_CPU / 2500000) // 0.4us #define CYCLES_800_T1H (F_CPU / 1250000) // 0.8us @@ -35,7 +35,7 @@ void ICACHE_RAM_ATTR espShow( startTime = 0; #ifdef NEO_KHZ400 - if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream + if(is800KHz) { #endif time0 = CYCLES_800_T0H; time1 = CYCLES_800_T1H; diff --git a/library.properties b/library.properties index a298ae45..be53f8f5 100644 --- a/library.properties +++ b/library.properties @@ -1,5 +1,5 @@ name=Adafruit NeoPixel -version=1.0.2 +version=1.0.3 author=Adafruit maintainer=Adafruit sentence=Arduino library for controlling single-wire-based LED pixels and strip.