From a37b1914329c0333ff6bfd1ccafbe730bec153e8 Mon Sep 17 00:00:00 2001 From: Phillip Burgess Date: Sat, 27 Jun 2015 21:08:59 -0700 Subject: [PATCH] Arduino Zero support --- Adafruit_NeoPixel.cpp | 85 +++++++++++++++++++++++++++++++++++++++++-- esp8266.c | 2 +- library.properties | 2 +- 3 files changed, 83 insertions(+), 6 deletions(-) diff --git a/Adafruit_NeoPixel.cpp b/Adafruit_NeoPixel.cpp index b562fa44..3d5dc1d3 100644 --- a/Adafruit_NeoPixel.cpp +++ b/Adafruit_NeoPixel.cpp @@ -835,6 +835,83 @@ void Adafruit_NeoPixel::show(void) { #error "Sorry, only 48 MHz is supported, please set Tools > CPU Speed to 48 MHz" #endif // F_CPU == 48000000 +#elif defined(__SAMD21G18A__) // Arduino Zero + + // Tried this with a timer/counter, couldn't quite get adequate + // resolution. So yay, you get a load of goofball NOPs... + + uint8_t *ptr, *end, p, bitMask, portNum; + uint32_t pinMask; + + portNum = g_APinDescription[pin].ulPort; + pinMask = 1ul << g_APinDescription[pin].ulPin; + ptr = pixels; + end = ptr + numBytes - 1; + p = *ptr++; + bitMask = 0x80; + + 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 +#endif + for(;;) { + *set = pinMask; + asm("nop; nop; nop; nop; nop; nop; nop; nop;"); + if(p & bitMask) { + asm("nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop;"); + *clr = pinMask; + } else { + *clr = pinMask; + asm("nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop;"); + } + if(bitMask >>= 1) { + asm("nop; nop; nop; nop; nop; nop; nop; nop; nop;"); + } else { + if(ptr >= end) break; + p = *ptr++; + bitMask = 0x80; + } + } +#ifdef NEO_KHZ400 + } else { // 400 KHz bitstream + for(;;) { + *set = pinMask; + asm("nop; nop; nop; nop; nop; nop; nop; nop; nop; nop; nop;"); + if(p & bitMask) { + asm("nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop;"); + *clr = pinMask; + } else { + *clr = pinMask; + asm("nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop;"); + } + asm("nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;" + "nop; nop; nop; nop; nop; nop; nop; nop;"); + if(bitMask >>= 1) { + asm("nop; nop; nop; nop; nop; nop; nop;"); + } else { + if(ptr >= end) break; + p = *ptr++; + bitMask = 0x80; + } + } + } +#endif + + #else // Other ARM architecture -- Presumed Arduino Due #define SCALE VARIANT_MCK / 2UL / 1000000UL @@ -871,13 +948,13 @@ void Adafruit_NeoPixel::show(void) { #ifdef NEO_KHZ400 if((type & NEO_SPDMASK) == NEO_KHZ800) { // 800 KHz bitstream #endif - time0 = TIME_800_0; - time1 = TIME_800_1; + time0 = TIME_800_0; + time1 = TIME_800_1; period = PERIOD_800; #ifdef NEO_KHZ400 } else { // 400 KHz bitstream - time0 = TIME_400_0; - time1 = TIME_400_1; + time0 = TIME_400_0; + time1 = TIME_400_1; period = PERIOD_400; } #endif diff --git a/esp8266.c b/esp8266.c index 4814ca0a..e457edb7 100644 --- a/esp8266.c +++ b/esp8266.c @@ -44,7 +44,7 @@ void ICACHE_RAM_ATTR espShow( } else { // 400 KHz bitstream time0 = CYCLES_400_T0H; time1 = CYCLES_400_T1H; - period = CYCLES_800; + period = CYCLES_400; } #endif diff --git a/library.properties b/library.properties index e2be3c4b..a298ae45 100644 --- a/library.properties +++ b/library.properties @@ -1,5 +1,5 @@ name=Adafruit NeoPixel -version=1.0.1 +version=1.0.2 author=Adafruit maintainer=Adafruit sentence=Arduino library for controlling single-wire-based LED pixels and strip.