forked from Makuna/NeoPixelBus
Applied Mem attribute to show method
This should fix the initial "off" color pixel.
This commit is contained in:
124
NeoPixelBus.cpp
124
NeoPixelBus.cpp
@@ -31,6 +31,13 @@ License along with NeoPixel. If not, see
|
|||||||
|
|
||||||
#include "NeoPixelBus.h"
|
#include "NeoPixelBus.h"
|
||||||
|
|
||||||
|
#if defined(ESP8266)
|
||||||
|
// due to linker overriding the ICACHE_RAM_ATTR for cpp files, these methods are
|
||||||
|
// moved into a C file so the attribute will be applied correctly
|
||||||
|
extern "C" void ICACHE_RAM_ATTR send_pixels_800(uint8_t* pixels, uint8_t* end, uint8_t pin);
|
||||||
|
extern "C" void ICACHE_RAM_ATTR send_pixels_400(uint8_t* pixels, uint8_t* end, uint8_t pin);
|
||||||
|
#endif
|
||||||
|
|
||||||
NeoPixelBus::NeoPixelBus(uint16_t n, uint8_t p, uint8_t t) :
|
NeoPixelBus::NeoPixelBus(uint16_t n, uint8_t p, uint8_t t) :
|
||||||
_countPixels(n),
|
_countPixels(n),
|
||||||
_sizePixels(n * 3),
|
_sizePixels(n * 3),
|
||||||
@@ -73,122 +80,6 @@ void NeoPixelBus::Begin(void)
|
|||||||
Dirty();
|
Dirty();
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(ESP8266)
|
|
||||||
|
|
||||||
|
|
||||||
#define CYCLES_800_T0H (F_CPU / 2500000 - 4) // 0.4us
|
|
||||||
#define CYCLES_800_T1H (F_CPU / 1250000 - 4) // 0.8us
|
|
||||||
#define CYCLES_800 (F_CPU / 800000 - 4) // 1.25us per bit
|
|
||||||
#define CYCLES_400_T0H (F_CPU / 2000000 - 4)
|
|
||||||
#define CYCLES_400_T1H (F_CPU / 833333 - 4)
|
|
||||||
#define CYCLES_400 (F_CPU / 400000 - 4)
|
|
||||||
|
|
||||||
static inline void send_pixels_800(uint8_t* pixels, uint8_t* end, uint8_t pin)
|
|
||||||
{
|
|
||||||
const uint32_t pinRegister = _BV(pin);
|
|
||||||
uint8_t mask;
|
|
||||||
uint8_t subpix;
|
|
||||||
uint32_t cyclesStart;
|
|
||||||
|
|
||||||
// this set low will help cleanup the first bit
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
|
||||||
__asm__ __volatile__("isync");
|
|
||||||
|
|
||||||
cyclesStart = ESP.getCycleCount() + CYCLES_800;
|
|
||||||
while (pixels < end)
|
|
||||||
{
|
|
||||||
subpix = *pixels++;
|
|
||||||
for (mask = 0x80; mask; mask >>= 1)
|
|
||||||
{
|
|
||||||
// do the check here while we are waiting on time to pass
|
|
||||||
bool nextBit = (subpix & mask);
|
|
||||||
uint32_t cyclesNext = cyclesStart;
|
|
||||||
|
|
||||||
// after we have done as much work as needed for this next bit
|
|
||||||
// now wait for the HIGH
|
|
||||||
do
|
|
||||||
{
|
|
||||||
// cache and use this count so we don't incur another
|
|
||||||
// instruction before we turn the bit high
|
|
||||||
cyclesStart = ESP.getCycleCount();
|
|
||||||
}
|
|
||||||
while ((cyclesStart - cyclesNext) < CYCLES_800);
|
|
||||||
|
|
||||||
// set high
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TS_ADDRESS, pinRegister);
|
|
||||||
|
|
||||||
// wait for the LOW
|
|
||||||
if (nextBit)
|
|
||||||
{
|
|
||||||
while ((ESP.getCycleCount() - cyclesStart) < CYCLES_800_T1H);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
while ((ESP.getCycleCount() - cyclesStart) < CYCLES_800_T0H);
|
|
||||||
}
|
|
||||||
|
|
||||||
// set low
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// while accurate, this isn't needed due to the delays at the
|
|
||||||
// top of Show() to enforce between update timing
|
|
||||||
// while ((ESP.getCycleCount() - cyclesStart) < CYCLES_800);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void send_pixels_400(uint8_t* pixels, uint8_t* end, uint8_t pin)
|
|
||||||
{
|
|
||||||
const uint32_t pinRegister = _BV(pin);
|
|
||||||
uint8_t mask;
|
|
||||||
uint8_t subpix;
|
|
||||||
uint32_t cyclesStart;
|
|
||||||
|
|
||||||
// this set low will help cleanup the first bit
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
|
||||||
__asm__ __volatile__("isync");
|
|
||||||
|
|
||||||
cyclesStart = ESP.getCycleCount() + CYCLES_400;
|
|
||||||
while (pixels < end)
|
|
||||||
{
|
|
||||||
subpix = *pixels++;
|
|
||||||
for (mask = 0x80; mask; mask >>= 1)
|
|
||||||
{
|
|
||||||
bool nextBit = (subpix & mask);
|
|
||||||
uint32_t cyclesNext = cyclesStart;
|
|
||||||
|
|
||||||
// after we have done as much work as needed for this next bit
|
|
||||||
// now wait for the HIGH
|
|
||||||
do
|
|
||||||
{
|
|
||||||
cyclesStart = ESP.getCycleCount();
|
|
||||||
} while ((cyclesStart - cyclesNext) < CYCLES_400);
|
|
||||||
|
|
||||||
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TS_ADDRESS, pinRegister);
|
|
||||||
|
|
||||||
// wait for the LOW
|
|
||||||
if (nextBit)
|
|
||||||
{
|
|
||||||
while ((ESP.getCycleCount() - cyclesStart) < CYCLES_400_T1H);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
while ((ESP.getCycleCount() - cyclesStart) < CYCLES_400_T0H);
|
|
||||||
}
|
|
||||||
|
|
||||||
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// while accurate, this isn't needed due to the delays at the
|
|
||||||
// top of Show() to enforce between update timing
|
|
||||||
// while ((ESP.getCycleCount() - cyclesStart) < CYCLES_400);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
void NeoPixelBus::Show(void)
|
void NeoPixelBus::Show(void)
|
||||||
{
|
{
|
||||||
if (!_pixels)
|
if (!_pixels)
|
||||||
@@ -824,7 +715,6 @@ void NeoPixelBus::Show(void)
|
|||||||
|
|
||||||
#elif defined(ESP8266)
|
#elif defined(ESP8266)
|
||||||
|
|
||||||
|
|
||||||
uint8_t* p = _pixels;
|
uint8_t* p = _pixels;
|
||||||
uint8_t* end = p + _sizePixels;
|
uint8_t* end = p + _sizePixels;
|
||||||
|
|
||||||
|
131
NeoPixelesp8266.c
Normal file
131
NeoPixelesp8266.c
Normal file
@@ -0,0 +1,131 @@
|
|||||||
|
/*
|
||||||
|
NeoPixelEsp8266.h - NeoPixel library helper functions for Esp8266 using cycle count
|
||||||
|
Copyright (c) 2015 Michael C. Miller. All right reserved.
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 2.1 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <Arduino.h>
|
||||||
|
#include <eagle_soc.h>
|
||||||
|
|
||||||
|
#if defined(ESP8266)
|
||||||
|
|
||||||
|
inline uint32_t _getCycleCount()
|
||||||
|
{
|
||||||
|
uint32_t ccount;
|
||||||
|
__asm__ __volatile__("rsr %0,ccount":"=a" (ccount));
|
||||||
|
return ccount;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define CYCLES_800_T0H (F_CPU / 2500000) // 0.4us
|
||||||
|
#define CYCLES_800_T1H (F_CPU / 1250000) // 0.8us
|
||||||
|
#define CYCLES_800 (F_CPU / 800000) // 1.25us per bit
|
||||||
|
#define CYCLES_400_T0H (F_CPU / 2000000)
|
||||||
|
#define CYCLES_400_T1H (F_CPU / 833333)
|
||||||
|
#define CYCLES_400 (F_CPU / 400000)
|
||||||
|
|
||||||
|
void ICACHE_RAM_ATTR send_pixels_800(uint8_t* pixels, uint8_t* end, uint8_t pin)
|
||||||
|
{
|
||||||
|
const uint32_t pinRegister = _BV(pin);
|
||||||
|
uint8_t mask;
|
||||||
|
uint8_t subpix;
|
||||||
|
uint32_t cyclesStart;
|
||||||
|
|
||||||
|
// trigger emediately
|
||||||
|
cyclesStart = _getCycleCount() - CYCLES_800;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
subpix = *pixels++;
|
||||||
|
for (mask = 0x80; mask != 0; mask >>= 1)
|
||||||
|
{
|
||||||
|
// do the checks here while we are waiting on time to pass
|
||||||
|
uint32_t cyclesBit = ((subpix & mask)) ? CYCLES_800_T1H : CYCLES_800_T0H;
|
||||||
|
uint32_t cyclesNext = cyclesStart;
|
||||||
|
uint32_t delta;
|
||||||
|
|
||||||
|
// after we have done as much work as needed for this next bit
|
||||||
|
// now wait for the HIGH
|
||||||
|
do
|
||||||
|
{
|
||||||
|
// cache and use this count so we don't incur another
|
||||||
|
// instruction before we turn the bit high
|
||||||
|
cyclesStart = _getCycleCount();
|
||||||
|
} while ((cyclesStart - cyclesNext) < CYCLES_800);
|
||||||
|
|
||||||
|
// set high
|
||||||
|
GPIO_REG_WRITE(GPIO_OUT_W1TS_ADDRESS, pinRegister);
|
||||||
|
|
||||||
|
// wait for the LOW
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cyclesNext = _getCycleCount();
|
||||||
|
} while ((cyclesNext - cyclesStart) < cyclesBit);
|
||||||
|
|
||||||
|
// set low
|
||||||
|
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
||||||
|
}
|
||||||
|
} while (pixels < end);
|
||||||
|
|
||||||
|
// while accurate, this isn't needed due to the delays at the
|
||||||
|
// top of Show() to enforce between update timing
|
||||||
|
// while ((_getCycleCount() - cyclesStart) < CYCLES_800);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ICACHE_RAM_ATTR send_pixels_400(uint8_t* pixels, uint8_t* end, uint8_t pin)
|
||||||
|
{
|
||||||
|
const uint32_t pinRegister = _BV(pin);
|
||||||
|
uint8_t mask;
|
||||||
|
uint8_t subpix;
|
||||||
|
uint32_t cyclesStart;
|
||||||
|
|
||||||
|
// trigger emediately
|
||||||
|
cyclesStart = _getCycleCount() - CYCLES_400;
|
||||||
|
while (pixels < end)
|
||||||
|
{
|
||||||
|
subpix = *pixels++;
|
||||||
|
for (mask = 0x80; mask; mask >>= 1)
|
||||||
|
{
|
||||||
|
uint32_t cyclesBit = ((subpix & mask)) ? CYCLES_400_T1H : CYCLES_400_T0H;
|
||||||
|
uint32_t cyclesNext = cyclesStart;
|
||||||
|
|
||||||
|
// after we have done as much work as needed for this next bit
|
||||||
|
// now wait for the HIGH
|
||||||
|
do
|
||||||
|
{
|
||||||
|
// cache and use this count so we don't incur another
|
||||||
|
// instruction before we turn the bit high
|
||||||
|
cyclesStart = _getCycleCount();
|
||||||
|
} while ((cyclesStart - cyclesNext) < CYCLES_400);
|
||||||
|
|
||||||
|
// set high
|
||||||
|
GPIO_REG_WRITE(GPIO_OUT_W1TS_ADDRESS, pinRegister);
|
||||||
|
|
||||||
|
// wait for the LOW
|
||||||
|
do
|
||||||
|
{
|
||||||
|
cyclesNext = _getCycleCount();
|
||||||
|
} while ((cyclesNext - cyclesStart) < cyclesBit);
|
||||||
|
|
||||||
|
// set low
|
||||||
|
GPIO_REG_WRITE(GPIO_OUT_W1TC_ADDRESS, pinRegister);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// while accurate, this isn't needed due to the delays at the
|
||||||
|
// top of Show() to enforce between update timing
|
||||||
|
// while ((_getCycleCount() - cyclesStart) < CYCLES_400);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
Reference in New Issue
Block a user