mirror of
https://github.com/esphome/esphome.git
synced 2024-12-02 11:44:13 +01:00
f3a969d35c
* Add ESP32-S3 support in NeoPixelBus component * Update NeoPixelBus version in platformio.ini
421 lines
14 KiB
Python
421 lines
14 KiB
Python
from dataclasses import dataclass
|
|
from typing import Any
|
|
import esphome.codegen as cg
|
|
import esphome.config_validation as cv
|
|
from esphome.const import (
|
|
CONF_CHANNEL,
|
|
CONF_CLOCK_PIN,
|
|
CONF_DATA_PIN,
|
|
CONF_METHOD,
|
|
CONF_PIN,
|
|
CONF_SPEED,
|
|
)
|
|
from esphome.components.esp32 import get_esp32_variant
|
|
from esphome.components.esp32.const import (
|
|
VARIANT_ESP32,
|
|
VARIANT_ESP32C3,
|
|
VARIANT_ESP32S2,
|
|
VARIANT_ESP32S3,
|
|
)
|
|
from esphome.core import CORE
|
|
from .const import (
|
|
CONF_ASYNC,
|
|
CONF_BUS,
|
|
CHIP_400KBPS,
|
|
CHIP_800KBPS,
|
|
CHIP_APA106,
|
|
CHIP_DOTSTAR,
|
|
CHIP_LC8812,
|
|
CHIP_LPD6803,
|
|
CHIP_LPD8806,
|
|
CHIP_P9813,
|
|
CHIP_SK6812,
|
|
CHIP_TM1814,
|
|
CHIP_TM1829,
|
|
CHIP_TM1914,
|
|
CHIP_WS2801,
|
|
CHIP_WS2811,
|
|
CHIP_WS2812,
|
|
CHIP_WS2812X,
|
|
CHIP_WS2813,
|
|
ONE_WIRE_CHIPS,
|
|
TWO_WIRE_CHIPS,
|
|
)
|
|
|
|
METHOD_BIT_BANG = "bit_bang"
|
|
METHOD_ESP8266_UART = "esp8266_uart"
|
|
METHOD_ESP8266_DMA = "esp8266_dma"
|
|
METHOD_ESP32_RMT = "esp32_rmt"
|
|
METHOD_ESP32_I2S = "esp32_i2s"
|
|
METHOD_SPI = "spi"
|
|
|
|
CHANNEL_DYNAMIC = "dynamic"
|
|
BUS_DYNAMIC = "dynamic"
|
|
SPI_BUS_VSPI = "vspi"
|
|
SPI_BUS_HSPI = "hspi"
|
|
SPI_SPEEDS = [40e6, 20e6, 10e6, 5e6, 2e6, 1e6, 500e3]
|
|
|
|
|
|
def _esp32_rmt_default_channel():
|
|
return {
|
|
VARIANT_ESP32C3: 1,
|
|
VARIANT_ESP32S2: 1,
|
|
VARIANT_ESP32S3: 1,
|
|
}.get(get_esp32_variant(), 6)
|
|
|
|
|
|
def _validate_esp32_rmt_channel(value):
|
|
if isinstance(value, str) and value.lower() == CHANNEL_DYNAMIC:
|
|
value = CHANNEL_DYNAMIC
|
|
else:
|
|
value = cv.int_(value)
|
|
variant_channels = {
|
|
VARIANT_ESP32: [0, 1, 2, 3, 4, 5, 6, 7, CHANNEL_DYNAMIC],
|
|
VARIANT_ESP32C3: [0, 1, CHANNEL_DYNAMIC],
|
|
VARIANT_ESP32S2: [0, 1, 2, 3, CHANNEL_DYNAMIC],
|
|
VARIANT_ESP32S3: [0, 1, 2, 3, CHANNEL_DYNAMIC],
|
|
}
|
|
variant = get_esp32_variant()
|
|
if variant not in variant_channels:
|
|
raise cv.Invalid(f"{variant} does not support the rmt method")
|
|
if value not in variant_channels[variant]:
|
|
raise cv.Invalid(f"{variant} does not support rmt channel {value}")
|
|
return value
|
|
|
|
|
|
def _esp32_i2s_default_bus():
|
|
return {
|
|
VARIANT_ESP32: 1,
|
|
VARIANT_ESP32S2: 0,
|
|
}.get(get_esp32_variant(), 0)
|
|
|
|
|
|
def _validate_esp32_i2s_bus(value):
|
|
if isinstance(value, str) and value.lower() == BUS_DYNAMIC:
|
|
value = BUS_DYNAMIC
|
|
else:
|
|
value = cv.int_(value)
|
|
variant_buses = {
|
|
VARIANT_ESP32: [0, 1, BUS_DYNAMIC],
|
|
VARIANT_ESP32S2: [0, BUS_DYNAMIC],
|
|
}
|
|
variant = get_esp32_variant()
|
|
if variant not in variant_buses:
|
|
raise cv.Invalid(f"{variant} does not support the i2s method")
|
|
if value not in variant_buses[variant]:
|
|
raise cv.Invalid(f"{variant} does not support i2s bus {value}")
|
|
return value
|
|
|
|
|
|
neo_ns = cg.global_ns
|
|
|
|
|
|
def _bit_bang_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/NeoEspBitBangMethod.h
|
|
# Some chips are only aliases
|
|
chip = {
|
|
CHIP_WS2813: CHIP_WS2812X,
|
|
CHIP_LC8812: CHIP_SK6812,
|
|
CHIP_TM1914: CHIP_TM1814,
|
|
CHIP_WS2812: CHIP_800KBPS,
|
|
}.get(chip, chip)
|
|
|
|
lookup = {
|
|
CHIP_WS2811: (neo_ns.NeoEspBitBangSpeedWs2811, False),
|
|
CHIP_WS2812X: (neo_ns.NeoEspBitBangSpeedWs2812x, False),
|
|
CHIP_SK6812: (neo_ns.NeoEspBitBangSpeedSk6812, False),
|
|
CHIP_TM1814: (neo_ns.NeoEspBitBangSpeedTm1814, True),
|
|
CHIP_TM1829: (neo_ns.NeoEspBitBangSpeedTm1829, True),
|
|
CHIP_800KBPS: (neo_ns.NeoEspBitBangSpeed800Kbps, False),
|
|
CHIP_400KBPS: (neo_ns.NeoEspBitBangSpeed400Kbps, False),
|
|
CHIP_APA106: (neo_ns.NeoEspBitBangSpeedApa106, False),
|
|
}
|
|
# For tm variants opposite of inverted is needed
|
|
speed, pinset_inverted = lookup[chip]
|
|
pinset = {
|
|
False: neo_ns.NeoEspPinset,
|
|
True: neo_ns.NeoEspPinsetInverted,
|
|
}[inverted != pinset_inverted]
|
|
return neo_ns.NeoEspBitBangMethodBase.template(speed, pinset)
|
|
|
|
|
|
def _bit_bang_extra_validate(config):
|
|
pin = config[CONF_PIN]
|
|
if CORE.is_esp8266 and not (0 <= pin <= 15):
|
|
# Due to use of w1ts
|
|
raise cv.Invalid("Bit bang only supports pins GPIO0-GPIO15 on ESP8266")
|
|
if CORE.is_esp32 and not (0 <= pin <= 31):
|
|
raise cv.Invalid("Bit bang only supports pins GPIO0-GPIO31 on ESP32")
|
|
|
|
|
|
def _esp8266_uart_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/NeoEsp8266UartMethod.h
|
|
uart_context, uart_base = {
|
|
False: (neo_ns.NeoEsp8266UartContext, neo_ns.NeoEsp8266Uart),
|
|
True: (neo_ns.NeoEsp8266UartInterruptContext, neo_ns.NeoEsp8266AsyncUart),
|
|
}[config[CONF_ASYNC]]
|
|
uart_feature = {
|
|
0: neo_ns.UartFeature0,
|
|
1: neo_ns.UartFeature1,
|
|
}[config[CONF_BUS]]
|
|
# Some chips are only aliases
|
|
chip = {
|
|
CHIP_WS2811: CHIP_WS2812X,
|
|
CHIP_WS2813: CHIP_WS2812X,
|
|
CHIP_LC8812: CHIP_SK6812,
|
|
CHIP_TM1914: CHIP_TM1814,
|
|
CHIP_WS2812: CHIP_800KBPS,
|
|
}.get(chip, chip)
|
|
|
|
lookup = {
|
|
CHIP_WS2812X: (neo_ns.NeoEsp8266UartSpeedWs2812x, False),
|
|
CHIP_SK6812: (neo_ns.NeoEsp8266UartSpeedSk6812, False),
|
|
CHIP_TM1814: (neo_ns.NeoEsp8266UartSpeedTm1814, True),
|
|
CHIP_TM1829: (neo_ns.NeoEsp8266UartSpeedTm1829, True),
|
|
CHIP_800KBPS: (neo_ns.NeoEsp8266UartSpeed800Kbps, False),
|
|
CHIP_400KBPS: (neo_ns.NeoEsp8266UartSpeed400Kbps, False),
|
|
CHIP_APA106: (neo_ns.NeoEsp8266UartSpeedApa106, False),
|
|
}
|
|
speed, uart_inverted = lookup[chip]
|
|
# For tm variants opposite of inverted is needed
|
|
inv = {
|
|
False: neo_ns.NeoEsp8266UartNotInverted,
|
|
True: neo_ns.NeoEsp8266UartInverted,
|
|
}[inverted != uart_inverted]
|
|
return neo_ns.NeoEsp8266UartMethodBase.template(
|
|
speed, uart_base.template(uart_feature, uart_context), inv
|
|
)
|
|
|
|
|
|
def _esp8266_uart_extra_validate(config):
|
|
pin = config[CONF_PIN]
|
|
bus = config[CONF_METHOD][CONF_BUS]
|
|
right_pin = {
|
|
0: 1, # U0TXD
|
|
1: 2, # U1TXD
|
|
}[bus]
|
|
if pin != right_pin:
|
|
raise cv.Invalid(f"ESP8266 uart bus {bus} only supports pin GPIO{right_pin}")
|
|
|
|
|
|
def _esp8266_dma_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/NeoEsp8266DmaMethod.h
|
|
# Some chips are only aliases
|
|
chip = {
|
|
CHIP_WS2811: CHIP_WS2812X,
|
|
CHIP_WS2813: CHIP_WS2812X,
|
|
CHIP_LC8812: CHIP_SK6812,
|
|
CHIP_TM1914: CHIP_TM1814,
|
|
CHIP_WS2812: CHIP_800KBPS,
|
|
}.get(chip, chip)
|
|
|
|
lookup = {
|
|
(CHIP_WS2812X, False): neo_ns.NeoEsp8266DmaSpeedWs2812x,
|
|
(CHIP_SK6812, False): neo_ns.NeoEsp8266DmaSpeedSk6812,
|
|
(CHIP_TM1814, True): neo_ns.NeoEsp8266DmaInvertedSpeedTm1814,
|
|
(CHIP_TM1829, True): neo_ns.NeoEsp8266DmaInvertedSpeedTm1829,
|
|
(CHIP_800KBPS, False): neo_ns.NeoEsp8266DmaSpeed800Kbps,
|
|
(CHIP_400KBPS, False): neo_ns.NeoEsp8266DmaSpeed400Kbps,
|
|
(CHIP_APA106, False): neo_ns.NeoEsp8266DmaSpeedApa106,
|
|
(CHIP_WS2812X, True): neo_ns.NeoEsp8266DmaInvertedSpeedWs2812x,
|
|
(CHIP_SK6812, True): neo_ns.NeoEsp8266DmaInvertedSpeedSk6812,
|
|
(CHIP_TM1814, False): neo_ns.NeoEsp8266DmaSpeedTm1814,
|
|
(CHIP_TM1829, False): neo_ns.NeoEsp8266DmaSpeedTm1829,
|
|
(CHIP_800KBPS, True): neo_ns.NeoEsp8266DmaInvertedSpeed800Kbps,
|
|
(CHIP_400KBPS, True): neo_ns.NeoEsp8266DmaInvertedSpeed400Kbps,
|
|
(CHIP_APA106, True): neo_ns.NeoEsp8266DmaInvertedSpeedApa106,
|
|
}
|
|
speed = lookup[(chip, inverted)]
|
|
return neo_ns.NeoEsp8266DmaMethodBase.template(speed)
|
|
|
|
|
|
def _esp8266_dma_extra_validate(config):
|
|
if config[CONF_PIN] != 3:
|
|
raise cv.Invalid("ESP8266 dma method only supports pin GPIO3")
|
|
|
|
|
|
def _esp32_rmt_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/NeoEsp32RmtMethod.h
|
|
channel = {
|
|
0: neo_ns.NeoEsp32RmtChannel0,
|
|
1: neo_ns.NeoEsp32RmtChannel1,
|
|
2: neo_ns.NeoEsp32RmtChannel2,
|
|
3: neo_ns.NeoEsp32RmtChannel3,
|
|
4: neo_ns.NeoEsp32RmtChannel4,
|
|
5: neo_ns.NeoEsp32RmtChannel5,
|
|
6: neo_ns.NeoEsp32RmtChannel6,
|
|
7: neo_ns.NeoEsp32RmtChannel7,
|
|
CHANNEL_DYNAMIC: neo_ns.NeoEsp32RmtChannelN,
|
|
}[config[CONF_CHANNEL]]
|
|
# Some chips are only aliases
|
|
chip = {
|
|
CHIP_WS2813: CHIP_WS2812X,
|
|
CHIP_LC8812: CHIP_SK6812,
|
|
CHIP_WS2812: CHIP_800KBPS,
|
|
}.get(chip, chip)
|
|
|
|
lookup = {
|
|
(CHIP_WS2811, False): neo_ns.NeoEsp32RmtSpeedWs2811,
|
|
(CHIP_WS2812X, False): neo_ns.NeoEsp32RmtSpeedWs2812x,
|
|
(CHIP_SK6812, False): neo_ns.NeoEsp32RmtSpeedSk6812,
|
|
(CHIP_TM1814, False): neo_ns.NeoEsp32RmtSpeedTm1814,
|
|
(CHIP_TM1829, False): neo_ns.NeoEsp32RmtSpeedTm1829,
|
|
(CHIP_TM1914, False): neo_ns.NeoEsp32RmtSpeedTm1914,
|
|
(CHIP_800KBPS, False): neo_ns.NeoEsp32RmtSpeed800Kbps,
|
|
(CHIP_400KBPS, False): neo_ns.NeoEsp32RmtSpeed400Kbps,
|
|
(CHIP_APA106, False): neo_ns.NeoEsp32RmtSpeedApa106,
|
|
(CHIP_WS2811, True): neo_ns.NeoEsp32RmtInvertedSpeedWs2811,
|
|
(CHIP_WS2812X, True): neo_ns.NeoEsp32RmtInvertedSpeedWs2812x,
|
|
(CHIP_SK6812, True): neo_ns.NeoEsp32RmtInvertedSpeedSk6812,
|
|
(CHIP_TM1814, True): neo_ns.NeoEsp32RmtInvertedSpeedTm1814,
|
|
(CHIP_TM1829, True): neo_ns.NeoEsp32RmtInvertedSpeedTm1829,
|
|
(CHIP_TM1914, True): neo_ns.NeoEsp32RmtInvertedSpeedTm1914,
|
|
(CHIP_800KBPS, True): neo_ns.NeoEsp32RmtInvertedSpeed800Kbps,
|
|
(CHIP_400KBPS, True): neo_ns.NeoEsp32RmtInvertedSpeed400Kbps,
|
|
(CHIP_APA106, True): neo_ns.NeoEsp32RmtInvertedSpeedApa106,
|
|
}
|
|
speed = lookup[(chip, inverted)]
|
|
return neo_ns.NeoEsp32RmtMethodBase.template(speed, channel)
|
|
|
|
|
|
def _esp32_i2s_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/NeoEsp32I2sMethod.h
|
|
bus = {
|
|
0: neo_ns.NeoEsp32I2sBusZero,
|
|
1: neo_ns.NeoEsp32I2sBusOne,
|
|
BUS_DYNAMIC: neo_ns.NeoEsp32I2sBusN,
|
|
}[config[CONF_BUS]]
|
|
# Some chips are only aliases
|
|
chip = {
|
|
CHIP_WS2811: CHIP_WS2812X,
|
|
CHIP_WS2813: CHIP_WS2812X,
|
|
CHIP_LC8812: CHIP_SK6812,
|
|
CHIP_WS2812: CHIP_800KBPS,
|
|
}.get(chip, chip)
|
|
|
|
lookup = {
|
|
CHIP_WS2812X: (neo_ns.NeoEsp32I2sSpeedWs2812x, False),
|
|
CHIP_SK6812: (neo_ns.NeoEsp32I2sSpeedSk6812, False),
|
|
CHIP_TM1814: (neo_ns.NeoEsp32I2sSpeedTm1814, True),
|
|
CHIP_TM1914: (neo_ns.NeoEsp32I2sSpeedTm1914, True),
|
|
CHIP_TM1829: (neo_ns.NeoEsp32I2sSpeedTm1829, True),
|
|
CHIP_800KBPS: (neo_ns.NeoEsp32I2sSpeed800Kbps, False),
|
|
CHIP_400KBPS: (neo_ns.NeoEsp32I2sSpeed400Kbps, False),
|
|
CHIP_APA106: (neo_ns.NeoEsp32I2sSpeedApa106, False),
|
|
}
|
|
speed, inv_inverted = lookup[chip]
|
|
# For tm variants opposite of inverted is needed
|
|
inv = {
|
|
False: neo_ns.NeoEsp32I2sNotInverted,
|
|
True: neo_ns.NeoEsp32I2sInverted,
|
|
}[inverted != inv_inverted]
|
|
return neo_ns.NeoEsp32I2sMethodBase.template(speed, bus, inv)
|
|
|
|
|
|
def _spi_to_code(config, chip: str, inverted: bool):
|
|
# https://github.com/Makuna/NeoPixelBus/blob/master/src/internal/TwoWireSpiImple.h
|
|
spi_imple = {
|
|
None: neo_ns.TwoWireSpiImple,
|
|
SPI_BUS_VSPI: neo_ns.TwoWireSpiImple,
|
|
SPI_BUS_HSPI: neo_ns.TwoWireHspiImple,
|
|
}[config.get(CONF_BUS)]
|
|
spi_speed = {
|
|
40e6: neo_ns.SpiSpeed40Mhz,
|
|
20e6: neo_ns.SpiSpeed20Mhz,
|
|
10e6: neo_ns.SpiSpeed10Mhz,
|
|
5e6: neo_ns.SpiSpeed5Mhz,
|
|
2e6: neo_ns.SpiSpeed2Mhz,
|
|
1e6: neo_ns.SpiSpeed1Mhz,
|
|
500e3: neo_ns.SpiSpeed500Khz,
|
|
}[config[CONF_SPEED]]
|
|
chip_method_base = {
|
|
CHIP_DOTSTAR: neo_ns.DotStarMethodBase,
|
|
CHIP_LPD6803: neo_ns.Lpd6803MethodBase,
|
|
CHIP_LPD8806: neo_ns.Lpd8806MethodBase,
|
|
CHIP_WS2801: neo_ns.Ws2801MethodBase,
|
|
CHIP_P9813: neo_ns.P9813MethodBase,
|
|
}[chip]
|
|
return chip_method_base.template(spi_imple.template(spi_speed))
|
|
|
|
|
|
def _spi_extra_validate(config):
|
|
if CORE.is_esp32:
|
|
return
|
|
|
|
if config[CONF_DATA_PIN] != 13 and config[CONF_CLOCK_PIN] != 14:
|
|
raise cv.Invalid(
|
|
"SPI only supports pins GPIO13 for data and GPIO14 for clock on ESP8266"
|
|
)
|
|
|
|
|
|
@dataclass
|
|
class MethodDescriptor:
|
|
method_schema: Any
|
|
to_code: Any
|
|
supported_chips: list[str]
|
|
extra_validate: Any = None
|
|
|
|
|
|
METHODS = {
|
|
METHOD_BIT_BANG: MethodDescriptor(
|
|
method_schema={},
|
|
to_code=_bit_bang_to_code,
|
|
extra_validate=_bit_bang_extra_validate,
|
|
supported_chips=ONE_WIRE_CHIPS,
|
|
),
|
|
METHOD_ESP8266_UART: MethodDescriptor(
|
|
method_schema=cv.All(
|
|
cv.only_on_esp8266,
|
|
{
|
|
cv.Optional(CONF_ASYNC, default=False): cv.boolean,
|
|
cv.Optional(CONF_BUS, default=1): cv.int_range(min=0, max=1),
|
|
},
|
|
),
|
|
extra_validate=_esp8266_uart_extra_validate,
|
|
to_code=_esp8266_uart_to_code,
|
|
supported_chips=ONE_WIRE_CHIPS,
|
|
),
|
|
METHOD_ESP8266_DMA: MethodDescriptor(
|
|
method_schema=cv.All(cv.only_on_esp8266, {}),
|
|
extra_validate=_esp8266_dma_extra_validate,
|
|
to_code=_esp8266_dma_to_code,
|
|
supported_chips=ONE_WIRE_CHIPS,
|
|
),
|
|
METHOD_ESP32_RMT: MethodDescriptor(
|
|
method_schema=cv.All(
|
|
cv.only_on_esp32,
|
|
{
|
|
cv.Optional(
|
|
CONF_CHANNEL, default=_esp32_rmt_default_channel
|
|
): _validate_esp32_rmt_channel,
|
|
},
|
|
),
|
|
to_code=_esp32_rmt_to_code,
|
|
supported_chips=ONE_WIRE_CHIPS,
|
|
),
|
|
METHOD_ESP32_I2S: MethodDescriptor(
|
|
method_schema=cv.All(
|
|
cv.only_on_esp32,
|
|
{
|
|
cv.Optional(
|
|
CONF_BUS, default=_esp32_i2s_default_bus
|
|
): _validate_esp32_i2s_bus,
|
|
},
|
|
),
|
|
to_code=_esp32_i2s_to_code,
|
|
supported_chips=ONE_WIRE_CHIPS,
|
|
),
|
|
METHOD_SPI: MethodDescriptor(
|
|
method_schema={
|
|
cv.Optional(CONF_BUS): cv.All(
|
|
cv.only_on_esp32, cv.one_of(SPI_BUS_VSPI, SPI_BUS_HSPI, lower=True)
|
|
),
|
|
cv.Optional(CONF_SPEED, default="10MHz"): cv.All(
|
|
cv.frequency, cv.one_of(*SPI_SPEEDS)
|
|
),
|
|
},
|
|
to_code=_spi_to_code,
|
|
extra_validate=_spi_extra_validate,
|
|
supported_chips=TWO_WIRE_CHIPS,
|
|
),
|
|
}
|