import logging from esphome import pins import esphome.codegen as cg from esphome.components.esp32 import add_idf_sdkconfig_option, get_esp32_variant from esphome.components.esp32.const import ( VARIANT_ESP32C3, VARIANT_ESP32S2, VARIANT_ESP32S3, ) from esphome.components.network import IPAddress from esphome.components.spi import CONF_INTERFACE_INDEX, get_spi_interface import esphome.config_validation as cv from esphome.const import ( CONF_ADDRESS, CONF_CLK_PIN, CONF_CS_PIN, CONF_DNS1, CONF_DNS2, CONF_DOMAIN, CONF_GATEWAY, CONF_ID, CONF_INTERRUPT_PIN, CONF_MANUAL_IP, CONF_MISO_PIN, CONF_MOSI_PIN, CONF_PAGE_ID, CONF_POLLING_INTERVAL, CONF_RESET_PIN, CONF_SPI, CONF_STATIC_IP, CONF_SUBNET, CONF_TYPE, CONF_USE_ADDRESS, CONF_VALUE, KEY_CORE, KEY_FRAMEWORK_VERSION, ) from esphome.core import CORE, TimePeriodMilliseconds, coroutine_with_priority import esphome.final_validate as fv CONFLICTS_WITH = ["wifi"] DEPENDENCIES = ["esp32"] AUTO_LOAD = ["network"] LOGGER = logging.getLogger(__name__) ethernet_ns = cg.esphome_ns.namespace("ethernet") PHYRegister = ethernet_ns.struct("PHYRegister") CONF_PHY_ADDR = "phy_addr" CONF_MDC_PIN = "mdc_pin" CONF_MDIO_PIN = "mdio_pin" CONF_CLK_MODE = "clk_mode" CONF_POWER_PIN = "power_pin" CONF_PHY_REGISTERS = "phy_registers" CONF_CLOCK_SPEED = "clock_speed" EthernetType = ethernet_ns.enum("EthernetType") ETHERNET_TYPES = { "LAN8720": EthernetType.ETHERNET_TYPE_LAN8720, "RTL8201": EthernetType.ETHERNET_TYPE_RTL8201, "DP83848": EthernetType.ETHERNET_TYPE_DP83848, "IP101": EthernetType.ETHERNET_TYPE_IP101, "JL1101": EthernetType.ETHERNET_TYPE_JL1101, "KSZ8081": EthernetType.ETHERNET_TYPE_KSZ8081, "KSZ8081RNA": EthernetType.ETHERNET_TYPE_KSZ8081RNA, "W5500": EthernetType.ETHERNET_TYPE_W5500, "OPENETH": EthernetType.ETHERNET_TYPE_OPENETH, } SPI_ETHERNET_TYPES = ["W5500"] SPI_ETHERNET_DEFAULT_POLLING_INTERVAL = TimePeriodMilliseconds(milliseconds=10) emac_rmii_clock_mode_t = cg.global_ns.enum("emac_rmii_clock_mode_t") emac_rmii_clock_gpio_t = cg.global_ns.enum("emac_rmii_clock_gpio_t") CLK_MODES = { "GPIO0_IN": ( emac_rmii_clock_mode_t.EMAC_CLK_EXT_IN, emac_rmii_clock_gpio_t.EMAC_CLK_IN_GPIO, ), "GPIO0_OUT": ( emac_rmii_clock_mode_t.EMAC_CLK_OUT, emac_rmii_clock_gpio_t.EMAC_APPL_CLK_OUT_GPIO, ), "GPIO16_OUT": ( emac_rmii_clock_mode_t.EMAC_CLK_OUT, emac_rmii_clock_gpio_t.EMAC_CLK_OUT_GPIO, ), "GPIO17_OUT": ( emac_rmii_clock_mode_t.EMAC_CLK_OUT, emac_rmii_clock_gpio_t.EMAC_CLK_OUT_180_GPIO, ), } MANUAL_IP_SCHEMA = cv.Schema( { cv.Required(CONF_STATIC_IP): cv.ipv4address, cv.Required(CONF_GATEWAY): cv.ipv4address, cv.Required(CONF_SUBNET): cv.ipv4address, cv.Optional(CONF_DNS1, default="0.0.0.0"): cv.ipv4address, cv.Optional(CONF_DNS2, default="0.0.0.0"): cv.ipv4address, } ) EthernetComponent = ethernet_ns.class_("EthernetComponent", cg.Component) ManualIP = ethernet_ns.struct("ManualIP") def _is_framework_spi_polling_mode_supported(): # SPI Ethernet without IRQ feature is added in # esp-idf >= (5.3+ ,5.2.1+, 5.1.4) and arduino-esp32 >= 3.0.0 framework_version = CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION] if CORE.using_esp_idf: if framework_version >= cv.Version(5, 3, 0): return True if cv.Version(5, 3, 0) > framework_version >= cv.Version(5, 2, 1): return True if cv.Version(5, 2, 0) > framework_version >= cv.Version(5, 1, 4): return True return False if CORE.using_arduino: return framework_version >= cv.Version(3, 0, 0) # fail safe: Unknown framework return False def _validate(config): if CONF_USE_ADDRESS not in config: if CONF_MANUAL_IP in config: use_address = str(config[CONF_MANUAL_IP][CONF_STATIC_IP]) else: use_address = CORE.name + config[CONF_DOMAIN] config[CONF_USE_ADDRESS] = use_address if config[CONF_TYPE] in SPI_ETHERNET_TYPES: if _is_framework_spi_polling_mode_supported(): if CONF_POLLING_INTERVAL in config and CONF_INTERRUPT_PIN in config: raise cv.Invalid( f"Cannot specify more than one of {CONF_INTERRUPT_PIN}, {CONF_POLLING_INTERVAL}" ) if CONF_POLLING_INTERVAL not in config and CONF_INTERRUPT_PIN not in config: config[CONF_POLLING_INTERVAL] = SPI_ETHERNET_DEFAULT_POLLING_INTERVAL else: if CONF_POLLING_INTERVAL in config: raise cv.Invalid( "In this version of the framework " f"({CORE.target_framework} {CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]}), " f"'{CONF_POLLING_INTERVAL}' is not supported." ) if CONF_INTERRUPT_PIN not in config: raise cv.Invalid( "In this version of the framework " f"({CORE.target_framework} {CORE.data[KEY_CORE][KEY_FRAMEWORK_VERSION]}), " f"'{CONF_INTERRUPT_PIN}' is a required option for [ethernet]." ) return config BASE_SCHEMA = cv.Schema( { cv.GenerateID(): cv.declare_id(EthernetComponent), cv.Optional(CONF_MANUAL_IP): MANUAL_IP_SCHEMA, cv.Optional(CONF_DOMAIN, default=".local"): cv.domain_name, cv.Optional(CONF_USE_ADDRESS): cv.string_strict, cv.Optional("enable_mdns"): cv.invalid( "This option has been removed. Please use the [disabled] option under the " "new mdns component instead." ), } ).extend(cv.COMPONENT_SCHEMA) PHY_REGISTER_SCHEMA = cv.Schema( { cv.Required(CONF_ADDRESS): cv.hex_int, cv.Required(CONF_VALUE): cv.hex_int, cv.Optional(CONF_PAGE_ID): cv.hex_int, } ) RMII_SCHEMA = BASE_SCHEMA.extend( cv.Schema( { cv.Required(CONF_MDC_PIN): pins.internal_gpio_output_pin_number, cv.Required(CONF_MDIO_PIN): pins.internal_gpio_output_pin_number, cv.Optional(CONF_CLK_MODE, default="GPIO0_IN"): cv.enum( CLK_MODES, upper=True, space="_" ), cv.Optional(CONF_PHY_ADDR, default=0): cv.int_range(min=0, max=31), cv.Optional(CONF_POWER_PIN): pins.internal_gpio_output_pin_number, cv.Optional(CONF_PHY_REGISTERS): cv.ensure_list(PHY_REGISTER_SCHEMA), } ) ) SPI_SCHEMA = BASE_SCHEMA.extend( cv.Schema( { cv.Required(CONF_CLK_PIN): pins.internal_gpio_output_pin_number, cv.Required(CONF_MISO_PIN): pins.internal_gpio_input_pin_number, cv.Required(CONF_MOSI_PIN): pins.internal_gpio_output_pin_number, cv.Required(CONF_CS_PIN): pins.internal_gpio_output_pin_number, cv.Optional(CONF_INTERRUPT_PIN): pins.internal_gpio_input_pin_number, cv.Optional(CONF_RESET_PIN): pins.internal_gpio_output_pin_number, cv.Optional(CONF_CLOCK_SPEED, default="26.67MHz"): cv.All( cv.frequency, cv.int_range(int(8e6), int(80e6)) ), # Set default value (SPI_ETHERNET_DEFAULT_POLLING_INTERVAL) at _validate() cv.Optional(CONF_POLLING_INTERVAL): cv.All( cv.positive_time_period_milliseconds, cv.Range(min=TimePeriodMilliseconds(milliseconds=1)), ), } ), ) CONFIG_SCHEMA = cv.All( cv.typed_schema( { "LAN8720": RMII_SCHEMA, "RTL8201": RMII_SCHEMA, "DP83848": RMII_SCHEMA, "IP101": RMII_SCHEMA, "JL1101": RMII_SCHEMA, "KSZ8081": RMII_SCHEMA, "KSZ8081RNA": RMII_SCHEMA, "W5500": SPI_SCHEMA, "OPENETH": BASE_SCHEMA, }, upper=True, ), _validate, ) def _final_validate(config): if config[CONF_TYPE] not in SPI_ETHERNET_TYPES: return if spi_configs := fv.full_config.get().get(CONF_SPI): variant = get_esp32_variant() if variant in (VARIANT_ESP32C3, VARIANT_ESP32S2, VARIANT_ESP32S3): spi_host = "SPI2_HOST" else: spi_host = "SPI3_HOST" for spi_conf in spi_configs: if (index := spi_conf.get(CONF_INTERFACE_INDEX)) is not None: interface = get_spi_interface(index) if interface == spi_host: raise cv.Invalid( f"`spi` component is using interface '{interface}'. " f"To use {config[CONF_TYPE]}, you must change the `interface` on the `spi` component.", ) FINAL_VALIDATE_SCHEMA = _final_validate def manual_ip(config): return cg.StructInitializer( ManualIP, ("static_ip", IPAddress(str(config[CONF_STATIC_IP]))), ("gateway", IPAddress(str(config[CONF_GATEWAY]))), ("subnet", IPAddress(str(config[CONF_SUBNET]))), ("dns1", IPAddress(str(config[CONF_DNS1]))), ("dns2", IPAddress(str(config[CONF_DNS2]))), ) def phy_register(address: int, value: int, page: int): return cg.StructInitializer( PHYRegister, ("address", address), ("value", value), ("page", page), ) @coroutine_with_priority(60.0) async def to_code(config): var = cg.new_Pvariable(config[CONF_ID]) await cg.register_component(var, config) if config[CONF_TYPE] == "W5500": cg.add(var.set_clk_pin(config[CONF_CLK_PIN])) cg.add(var.set_miso_pin(config[CONF_MISO_PIN])) cg.add(var.set_mosi_pin(config[CONF_MOSI_PIN])) cg.add(var.set_cs_pin(config[CONF_CS_PIN])) if CONF_INTERRUPT_PIN in config: cg.add(var.set_interrupt_pin(config[CONF_INTERRUPT_PIN])) else: cg.add(var.set_polling_interval(config[CONF_POLLING_INTERVAL])) if _is_framework_spi_polling_mode_supported(): cg.add_define("USE_ETHERNET_SPI_POLLING_SUPPORT") if CONF_RESET_PIN in config: cg.add(var.set_reset_pin(config[CONF_RESET_PIN])) cg.add(var.set_clock_speed(config[CONF_CLOCK_SPEED])) cg.add_define("USE_ETHERNET_SPI") if CORE.using_esp_idf: add_idf_sdkconfig_option("CONFIG_ETH_USE_SPI_ETHERNET", True) add_idf_sdkconfig_option("CONFIG_ETH_SPI_ETHERNET_W5500", True) elif config[CONF_TYPE] == "OPENETH": cg.add_define("USE_ETHERNET_OPENETH") add_idf_sdkconfig_option("CONFIG_ETH_USE_OPENETH", True) else: cg.add(var.set_phy_addr(config[CONF_PHY_ADDR])) cg.add(var.set_mdc_pin(config[CONF_MDC_PIN])) cg.add(var.set_mdio_pin(config[CONF_MDIO_PIN])) cg.add(var.set_clk_mode(*CLK_MODES[config[CONF_CLK_MODE]])) if CONF_POWER_PIN in config: cg.add(var.set_power_pin(config[CONF_POWER_PIN])) for register_value in config.get(CONF_PHY_REGISTERS, []): reg = phy_register( register_value.get(CONF_ADDRESS), register_value.get(CONF_VALUE), register_value.get(CONF_PAGE_ID), ) cg.add(var.add_phy_register(reg)) cg.add(var.set_type(ETHERNET_TYPES[config[CONF_TYPE]])) cg.add(var.set_use_address(config[CONF_USE_ADDRESS])) if CONF_MANUAL_IP in config: cg.add(var.set_manual_ip(manual_ip(config[CONF_MANUAL_IP]))) cg.add_define("USE_ETHERNET") if CORE.using_arduino: cg.add_library("WiFi", None)