forked from Makuna/NeoPixelBus
Mbi6033 (#719)
* two wire protocol * due to unique reset using clock pin, supports bit bang only methods.
This commit is contained in:
@@ -35,6 +35,7 @@ License along with NeoPixel. If not, see
|
|||||||
#include "methods/P9813GenericMethod.h"
|
#include "methods/P9813GenericMethod.h"
|
||||||
#include "methods/Tlc5947GenericMethod.h"
|
#include "methods/Tlc5947GenericMethod.h"
|
||||||
#include "methods/Sm16716GenericMethod.h"
|
#include "methods/Sm16716GenericMethod.h"
|
||||||
|
#include "methods/Mbi6033GenericMethod.h"
|
||||||
|
|
||||||
// Platform specific and One Wire (data) methods
|
// Platform specific and One Wire (data) methods
|
||||||
//
|
//
|
||||||
|
207
src/internal/methods/Mbi6033GenericMethod.h
Normal file
207
src/internal/methods/Mbi6033GenericMethod.h
Normal file
@@ -0,0 +1,207 @@
|
|||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
NeoPixel library helper functions for Mbi6033s using general Pins.
|
||||||
|
|
||||||
|
Written by Michael C. Miller.
|
||||||
|
|
||||||
|
I invest time and resources providing this open source code,
|
||||||
|
please support me by dontating (see https://github.com/Makuna/NeoPixelBus)
|
||||||
|
|
||||||
|
-------------------------------------------------------------------------
|
||||||
|
This file is part of the Makuna/NeoPixelBus library.
|
||||||
|
|
||||||
|
NeoPixelBus 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 3 of
|
||||||
|
the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
NeoPixelBus 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 NeoPixel. If not, see
|
||||||
|
<http://www.gnu.org/licenses/>.
|
||||||
|
-------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
// must also check for arm due to Teensy incorrectly having ARDUINO_ARCH_AVR set
|
||||||
|
#if defined(ARDUINO_ARCH_AVR) && !defined(__arm__)
|
||||||
|
#include "TwoWireBitBangImpleAvr.h"
|
||||||
|
#else
|
||||||
|
#include "TwoWireBitBangImple.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
template<typename T_TWOWIRE> class Mbi6033MethodBase
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
typedef typename T_TWOWIRE::SettingsObject SettingsObject;
|
||||||
|
|
||||||
|
Mbi6033MethodBase(uint8_t pinClock, uint8_t pinData, uint16_t pixelCount, size_t elementSize, size_t settingsSize) :
|
||||||
|
_countChips(NeoUtil::RoundUp(pixelCount * elementSize, c_countBytesPerChip) / c_countBytesPerChip),
|
||||||
|
_sizeData(_countChips * c_countBytesPerChip + settingsSize),
|
||||||
|
_pinClock(pinClock),
|
||||||
|
_wire(pinClock, pinData)
|
||||||
|
{
|
||||||
|
_data = static_cast<uint8_t*>(malloc(_sizeData));
|
||||||
|
// data cleared later in Begin()
|
||||||
|
}
|
||||||
|
|
||||||
|
#if !defined(__AVR_ATtiny85__) && !defined(ARDUINO_attiny)
|
||||||
|
Mbi6033MethodBase(uint16_t pixelCount, size_t elementSize, size_t settingsSize) :
|
||||||
|
Mbi6033MethodBase(SCK, MOSI, pixelCount, elementSize, settingsSize)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
~Mbi6033MethodBase()
|
||||||
|
{
|
||||||
|
free(_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsReadyToUpdate() const
|
||||||
|
{
|
||||||
|
return true; // clock driven chips don't have a required delay
|
||||||
|
}
|
||||||
|
|
||||||
|
#if defined(ARDUINO_ARCH_ESP32)
|
||||||
|
void Initialize(int8_t sck, int8_t miso, int8_t mosi, int8_t ss)
|
||||||
|
{
|
||||||
|
_pinClock = sck;
|
||||||
|
_wire.begin(sck, miso, mosi, ss);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void Initialize()
|
||||||
|
{
|
||||||
|
_wire.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Update(bool)
|
||||||
|
{
|
||||||
|
// non current header format:
|
||||||
|
// 8 bits: command (0xf3) = non current header mode
|
||||||
|
// 14 bits: sync (0x0000)
|
||||||
|
// 14 bits: length (count chips - 1)
|
||||||
|
// 8 bits: configuration (0x02)
|
||||||
|
// bits (6-4) refresh rate divider bits (0 for fastest)
|
||||||
|
// bit (1) on or off
|
||||||
|
// 4 bits: header code name X1 (0x0)
|
||||||
|
const uint16_t chipLength = _countChips - 1;
|
||||||
|
const uint8_t headerFrame[6] = { 0xf3,
|
||||||
|
0x00,
|
||||||
|
static_cast<uint8_t>(chipLength >> 12),
|
||||||
|
static_cast<uint8_t>((chipLength >> 4) & 0xff),
|
||||||
|
static_cast<uint8_t>((chipLength << 4) & 0xff),
|
||||||
|
0x20 };
|
||||||
|
|
||||||
|
// prefix protocol, >21us clock low, clock high, >21us clock low
|
||||||
|
// expecting at least 21us since last call to show
|
||||||
|
// but using hardware SPI won't allow messing with clock
|
||||||
|
// directly like this...
|
||||||
|
//delayMicroseconds(c_usResetTime);
|
||||||
|
//digitalWrite(_pinClock, HIGH);
|
||||||
|
//digitalWrite(_pinClock, LOW);
|
||||||
|
//delayMicroseconds(c_usResetTime);
|
||||||
|
|
||||||
|
_wire.beginTransaction();
|
||||||
|
|
||||||
|
// reset by toggle of clock
|
||||||
|
delayMicroseconds(c_usResetTime);
|
||||||
|
_wire.transmitBit(0); // Our Two Wire BitBang supports this
|
||||||
|
delayMicroseconds(c_usResetTime);
|
||||||
|
|
||||||
|
// header frame
|
||||||
|
_wire.transmitBytes(headerFrame, sizeof(headerFrame));
|
||||||
|
|
||||||
|
// data:
|
||||||
|
// 24 bytes per chip of data (12 16 bit values)
|
||||||
|
_wire.transmitBytes(_data, _sizeData);
|
||||||
|
|
||||||
|
_wire.endTransaction();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AlwaysUpdate()
|
||||||
|
{
|
||||||
|
// this method requires update to be called only if changes to buffer
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t* getData() const
|
||||||
|
{
|
||||||
|
return _data;
|
||||||
|
};
|
||||||
|
|
||||||
|
size_t getDataSize() const
|
||||||
|
{
|
||||||
|
return _sizeData;
|
||||||
|
};
|
||||||
|
|
||||||
|
void applySettings([[maybe_unused]] const SettingsObject& settings)
|
||||||
|
{
|
||||||
|
_wire.applySettings(settings);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
// while spec states 4 RGB * 16 bit values,
|
||||||
|
// its really 12 channels * 16 bit values, as they could
|
||||||
|
// be wired how ever the circuit is built, like 3 RGBW
|
||||||
|
static const uint16_t c_countBytesPerChip = 24; // twelve 16 bit values
|
||||||
|
static const uint16_t c_usResetTime = 21;
|
||||||
|
|
||||||
|
const uint16_t _countChips; // not pixels, driver chips
|
||||||
|
const size_t _sizeData; // Size of '_data' buffer below
|
||||||
|
|
||||||
|
uint8_t _pinClock;
|
||||||
|
T_TWOWIRE _wire;
|
||||||
|
uint8_t* _data; // Holds LED color values
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef Mbi6033MethodBase<TwoWireBitBangImple> Mbi6033Method;
|
||||||
|
|
||||||
|
/* Due to reset model by these chips needing to control clock, we can't use hardware SPI
|
||||||
|
* as neither the normal SPI exposes a single bit send nor does it allow direct clock pulses
|
||||||
|
* using digitalWrite. If a generalized solution could be found, then this could be enabled
|
||||||
|
*
|
||||||
|
#if !defined(__AVR_ATtiny85__) && !defined(ARDUINO_attiny)
|
||||||
|
#include "TwoWireSpiImple.h"
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed40Mhz>> Mbi6033Spi40MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed20Mhz>> Mbi6033Spi20MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed10Mhz>> Mbi6033Spi10MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed5Mhz>> Mbi6033Spi5MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed2Mhz>> Mbi6033Spi2MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed1Mhz>> Mbi6033Spi1MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed500Khz>> Mbi6033Spi500KhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeedHz>> Mbi6033SpiHzMethod;
|
||||||
|
|
||||||
|
typedef Mbi6033Spi10MhzMethod Mbi6033SpiMethod;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(ARDUINO_ARCH_ESP32)
|
||||||
|
// Give option to use Vspi alias of Spi class if wanting to specify which SPI peripheral is used on the ESP32
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed40Mhz>> Mbi6033Esp32Vspi40MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed20Mhz>> Mbi6033Esp32Vspi20MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed10Mhz>> Mbi6033Esp32Vspi10MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed5Mhz>> Mbi6033Esp32Vspi5MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed2Mhz>> Mbi6033Esp32Vspi2MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed1Mhz>> Mbi6033Esp32Vspi1MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeed500Khz>> Mbi6033Esp32Vspi500KhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireSpiImple<SpiSpeedHz>> Mbi6033Esp32VspiHzMethod;
|
||||||
|
|
||||||
|
typedef Mbi6033Spi10MhzMethod Mbi6033Esp32VspiMethod;
|
||||||
|
|
||||||
|
#include "TwoWireHspiImple.h"
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed40Mhz>> Mbi6033Esp32Hspi40MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed20Mhz>> Mbi6033Esp32Hspi20MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed10Mhz>> Mbi6033Esp32Hspi10MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed5Mhz>> Mbi6033Esp32Hspi5MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed2Mhz>> Mbi6033Esp32Hspi2MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed1Mhz>> Mbi6033Esp32Hspi1MhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeed500Khz>> Mbi6033Esp32Hspi500KhzMethod;
|
||||||
|
typedef Mbi6033MethodBase<TwoWireHspiImple<SpiSpeedHz>> Mbi6033Esp32HspiHzMethod;
|
||||||
|
|
||||||
|
typedef Mbi6033Esp32Hspi10MhzMethod Mbi6033Esp32HspiMethod;
|
||||||
|
#endif
|
||||||
|
*/
|
Reference in New Issue
Block a user