mirror of
https://github.com/espressif/esp-idf.git
synced 2025-11-04 22:22:27 +00:00
feat(esp_eth): removed SPI Ethernet and PHY drivers from IDF
This commit is contained in:
@@ -1,23 +1,145 @@
|
||||
/* Ethernet Basic Example
|
||||
|
||||
This example code is in the Public Domain (or CC0 licensed, at your option.)
|
||||
|
||||
Unless required by applicable law or agreed to in writing, this
|
||||
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||||
CONDITIONS OF ANY KIND, either express or implied.
|
||||
*/
|
||||
/*
|
||||
* SPDX-FileCopyrightText: 2025 Espressif Systems (Shanghai) CO LTD
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
*/
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include "freertos/FreeRTOS.h"
|
||||
#include "freertos/task.h"
|
||||
#include "esp_netif.h"
|
||||
#include "esp_eth.h"
|
||||
#include "esp_event.h"
|
||||
#include "esp_log.h"
|
||||
#include "ethernet_init.h"
|
||||
#include "esp_check.h"
|
||||
#include "sdkconfig.h"
|
||||
|
||||
static const char *TAG = "eth_example";
|
||||
static const char *TAG = "eth_basic_example";
|
||||
|
||||
/**
|
||||
* @brief Initialize Ethernet driver with generic PHY (all IEEE 802.3 compliant PHYs)
|
||||
*
|
||||
*
|
||||
* @param[out] eth_handle_out initialized Ethernet driver handle
|
||||
* @return
|
||||
* - ESP_OK on success
|
||||
* - ESP_ERR_INVALID_ARG when passed invalid pointer
|
||||
* - ESP_FAIL on any other failure
|
||||
*/
|
||||
static esp_err_t eth_init(esp_eth_handle_t *eth_handle_out)
|
||||
{
|
||||
if (eth_handle_out == NULL) {
|
||||
ESP_LOGE(TAG, "invalid argument: eth_handle_out cannot be NULL");
|
||||
return ESP_ERR_INVALID_ARG;
|
||||
}
|
||||
|
||||
// Init common MAC and PHY configs to default
|
||||
eth_mac_config_t mac_config = ETH_MAC_DEFAULT_CONFIG();
|
||||
eth_phy_config_t phy_config = ETH_PHY_DEFAULT_CONFIG();
|
||||
|
||||
// Update PHY config based on board specific configuration
|
||||
phy_config.phy_addr = CONFIG_EXAMPLE_ETH_PHY_ADDR;
|
||||
phy_config.reset_gpio_num = CONFIG_EXAMPLE_ETH_PHY_RST_GPIO;
|
||||
#if CONFIG_EXAMPLE_ETH_PHY_RST_TIMING_EN
|
||||
phy_config.hw_reset_assert_time_us = CONFIG_EXAMPLE_ETH_PHY_RST_ASSERT_TIME_US;
|
||||
phy_config.post_hw_reset_delay_ms = CONFIG_EXAMPLE_ETH_PHY_RST_DELAY_MS;
|
||||
#endif // CONFIG_EXAMPLE_ETH_PHY_RST_TIMING_EN
|
||||
|
||||
// Init vendor specific MAC config to default
|
||||
eth_esp32_emac_config_t esp32_emac_config = ETH_ESP32_EMAC_DEFAULT_CONFIG();
|
||||
// Update vendor specific MAC config based on board configuration
|
||||
esp32_emac_config.smi_gpio.mdc_num = CONFIG_EXAMPLE_ETH_MDC_GPIO;
|
||||
esp32_emac_config.smi_gpio.mdio_num = CONFIG_EXAMPLE_ETH_MDIO_GPIO;
|
||||
|
||||
#if CONFIG_EXAMPLE_ETH_PHY_INTERFACE_RMII
|
||||
// Configure RMII based on Kconfig when non-default configuration selected
|
||||
esp32_emac_config.interface = EMAC_DATA_INTERFACE_RMII;
|
||||
|
||||
// Configure RMII clock mode and GPIO
|
||||
#if CONFIG_EXAMPLE_ETH_RMII_CLK_INPUT
|
||||
esp32_emac_config.clock_config.rmii.clock_mode = EMAC_CLK_EXT_IN;
|
||||
#else // CONFIG_EXAMPLE_ETH_RMII_CLK_OUTPUT
|
||||
esp32_emac_config.clock_config.rmii.clock_mode = EMAC_CLK_OUT;
|
||||
#endif
|
||||
esp32_emac_config.clock_config.rmii.clock_gpio = CONFIG_EXAMPLE_ETH_RMII_CLK_GPIO;
|
||||
|
||||
#if CONFIG_EXAMPLE_ETH_RMII_CLK_EXT_LOOPBACK_EN
|
||||
esp32_emac_config.clock_config.rmii.clock_loopback_gpio = CONFIG_EXAMPLE_ETH_RMII_CLK_EXT_LOOPBACK_IN_GPIO;
|
||||
#endif
|
||||
|
||||
#if SOC_EMAC_USE_MULTI_IO_MUX
|
||||
// Configure RMII datapane GPIOs
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.tx_en_num = CONFIG_EXAMPLE_ETH_RMII_TX_EN_GPIO;
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.txd0_num = CONFIG_EXAMPLE_ETH_RMII_TXD0_GPIO;
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.txd1_num = CONFIG_EXAMPLE_ETH_RMII_TXD1_GPIO;
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.crs_dv_num = CONFIG_EXAMPLE_ETH_RMII_CRS_DV_GPIO;
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.rxd0_num = CONFIG_EXAMPLE_ETH_RMII_RXD0_GPIO;
|
||||
esp32_emac_config.emac_dataif_gpio.rmii.rxd1_num = CONFIG_EXAMPLE_ETH_RMII_RXD1_GPIO;
|
||||
#endif // SOC_EMAC_USE_MULTI_IO_MUX
|
||||
#endif // CONFIG_EXAMPLE_ETH_PHY_INTERFACE_RMII
|
||||
|
||||
// Create new ESP32 Ethernet MAC instance
|
||||
esp_eth_mac_t *mac = esp_eth_mac_new_esp32(&esp32_emac_config, &mac_config);
|
||||
if (mac == NULL) {
|
||||
ESP_LOGE(TAG, "create MAC instance failed");
|
||||
return ESP_FAIL;
|
||||
}
|
||||
|
||||
// Create new generic PHY instance
|
||||
esp_eth_phy_t *phy = esp_eth_phy_new_generic(&phy_config);
|
||||
if (phy == NULL) {
|
||||
ESP_LOGE(TAG, "create PHY instance failed");
|
||||
mac->del(mac);
|
||||
return ESP_FAIL;
|
||||
}
|
||||
|
||||
// Init Ethernet driver to default and install it
|
||||
esp_eth_handle_t eth_handle = NULL;
|
||||
esp_eth_config_t config = ETH_DEFAULT_CONFIG(mac, phy);
|
||||
if (esp_eth_driver_install(&config, ð_handle) != ESP_OK) {
|
||||
ESP_LOGE(TAG, "Ethernet driver install failed");
|
||||
mac->del(mac);
|
||||
phy->del(phy);
|
||||
return ESP_FAIL;
|
||||
}
|
||||
|
||||
*eth_handle_out = eth_handle;
|
||||
|
||||
return ESP_OK;
|
||||
}
|
||||
|
||||
#if CONFIG_EXAMPLE_ETH_DEINIT_AFTER_S >= 0
|
||||
/**
|
||||
* @brief De-initialize Ethernet driver
|
||||
* @note Ethernet driver must be stopped prior calling this function.
|
||||
*
|
||||
* @param[in] eth_handle Ethernet driver to be de-initialized
|
||||
* @return
|
||||
* - ESP_OK on success
|
||||
* - ESP_ERR_INVALID_ARG when passed invalid pointer
|
||||
*/
|
||||
static esp_err_t eth_deinit(esp_eth_handle_t eth_handle)
|
||||
{
|
||||
ESP_RETURN_ON_FALSE(eth_handle != NULL, ESP_ERR_INVALID_ARG, TAG, "Ethernet handle cannot be NULL");
|
||||
|
||||
esp_eth_mac_t *mac = NULL;
|
||||
esp_eth_phy_t *phy = NULL;
|
||||
esp_eth_get_mac_instance(eth_handle, &mac);
|
||||
esp_eth_get_phy_instance(eth_handle, &phy);
|
||||
ESP_RETURN_ON_ERROR(esp_eth_driver_uninstall(eth_handle), TAG, "Ethernet %p uninstall failed", eth_handle);
|
||||
|
||||
if (mac != NULL) {
|
||||
mac->del(mac);
|
||||
}
|
||||
if (phy != NULL) {
|
||||
phy->del(phy);
|
||||
}
|
||||
|
||||
return ESP_OK;
|
||||
}
|
||||
#endif // CONFIG_EXAMPLE_ETH_DEINIT_AFTER_S >= 0
|
||||
|
||||
/** Event handler for Ethernet events */
|
||||
static void eth_event_handler(void *arg, esp_event_base_t event_base,
|
||||
@@ -66,73 +188,40 @@ static void got_ip_event_handler(void *arg, esp_event_base_t event_base,
|
||||
void app_main(void)
|
||||
{
|
||||
// Initialize Ethernet driver
|
||||
uint8_t eth_port_cnt = 0;
|
||||
esp_eth_handle_t *eth_handles;
|
||||
ESP_ERROR_CHECK(example_eth_init(ð_handles, ð_port_cnt));
|
||||
esp_eth_handle_t eth_handle;
|
||||
ESP_ERROR_CHECK(eth_init(ð_handle));
|
||||
|
||||
// Initialize TCP/IP network interface aka the esp-netif (should be called only once in application)
|
||||
ESP_ERROR_CHECK(esp_netif_init());
|
||||
// Create default event loop that running in background
|
||||
ESP_ERROR_CHECK(esp_event_loop_create_default());
|
||||
|
||||
esp_netif_t *eth_netifs[eth_port_cnt];
|
||||
esp_eth_netif_glue_handle_t eth_netif_glues[eth_port_cnt];
|
||||
|
||||
// Create instance(s) of esp-netif for Ethernet(s)
|
||||
if (eth_port_cnt == 1) {
|
||||
// Use ESP_NETIF_DEFAULT_ETH when just one Ethernet interface is used and you don't need to modify
|
||||
// default esp-netif configuration parameters.
|
||||
esp_netif_config_t cfg = ESP_NETIF_DEFAULT_ETH();
|
||||
eth_netifs[0] = esp_netif_new(&cfg);
|
||||
eth_netif_glues[0] = esp_eth_new_netif_glue(eth_handles[0]);
|
||||
// Attach Ethernet driver to TCP/IP stack
|
||||
ESP_ERROR_CHECK(esp_netif_attach(eth_netifs[0], eth_netif_glues[0]));
|
||||
} else {
|
||||
// Use ESP_NETIF_INHERENT_DEFAULT_ETH when multiple Ethernet interfaces are used and so you need to modify
|
||||
// esp-netif configuration parameters for each interface (name, priority, etc.).
|
||||
esp_netif_inherent_config_t esp_netif_config = ESP_NETIF_INHERENT_DEFAULT_ETH();
|
||||
esp_netif_config_t cfg_spi = {
|
||||
.base = &esp_netif_config,
|
||||
.stack = ESP_NETIF_NETSTACK_DEFAULT_ETH
|
||||
};
|
||||
char if_key_str[10];
|
||||
char if_desc_str[10];
|
||||
char num_str[3];
|
||||
for (int i = 0; i < eth_port_cnt; i++) {
|
||||
itoa(i, num_str, 10);
|
||||
strcat(strcpy(if_key_str, "ETH_"), num_str);
|
||||
strcat(strcpy(if_desc_str, "eth"), num_str);
|
||||
esp_netif_config.if_key = if_key_str;
|
||||
esp_netif_config.if_desc = if_desc_str;
|
||||
esp_netif_config.route_prio -= i*5;
|
||||
eth_netifs[i] = esp_netif_new(&cfg_spi);
|
||||
eth_netif_glues[i] = esp_eth_new_netif_glue(eth_handles[i]);
|
||||
// Attach Ethernet driver to TCP/IP stack
|
||||
ESP_ERROR_CHECK(esp_netif_attach(eth_netifs[i], eth_netif_glues[i]));
|
||||
}
|
||||
}
|
||||
// Create instance of esp-netif for Ethernet
|
||||
// Use ESP_NETIF_DEFAULT_ETH when just one Ethernet interface is used and you don't need to modify
|
||||
// default esp-netif configuration parameters.
|
||||
esp_netif_config_t cfg = ESP_NETIF_DEFAULT_ETH();
|
||||
esp_netif_t *eth_netif = esp_netif_new(&cfg);
|
||||
esp_eth_netif_glue_handle_t eth_netif_glue = esp_eth_new_netif_glue(eth_handle);
|
||||
// Attach Ethernet driver to TCP/IP stack
|
||||
ESP_ERROR_CHECK(esp_netif_attach(eth_netif, eth_netif_glue));
|
||||
|
||||
// Register user defined event handlers
|
||||
ESP_ERROR_CHECK(esp_event_handler_register(ETH_EVENT, ESP_EVENT_ANY_ID, ð_event_handler, NULL));
|
||||
ESP_ERROR_CHECK(esp_event_handler_register(IP_EVENT, IP_EVENT_ETH_GOT_IP, &got_ip_event_handler, NULL));
|
||||
|
||||
// Start Ethernet driver state machine
|
||||
for (int i = 0; i < eth_port_cnt; i++) {
|
||||
ESP_ERROR_CHECK(esp_eth_start(eth_handles[i]));
|
||||
}
|
||||
ESP_ERROR_CHECK(esp_eth_start(eth_handle));
|
||||
|
||||
#if CONFIG_EXAMPLE_ETH_DEINIT_AFTER_S >= 0
|
||||
// For demonstration purposes, wait and then deinit Ethernet network
|
||||
vTaskDelay(pdMS_TO_TICKS(CONFIG_EXAMPLE_ETH_DEINIT_AFTER_S * 1000));
|
||||
ESP_LOGI(TAG, "stop and deinitialize Ethernet network...");
|
||||
// Stop Ethernet driver state machine and destroy netif
|
||||
for (int i = 0; i < eth_port_cnt; i++) {
|
||||
ESP_ERROR_CHECK(esp_eth_stop(eth_handles[i]));
|
||||
ESP_ERROR_CHECK(esp_eth_del_netif_glue(eth_netif_glues[i]));
|
||||
esp_netif_destroy(eth_netifs[i]);
|
||||
}
|
||||
ESP_ERROR_CHECK(esp_eth_stop(eth_handle));
|
||||
ESP_ERROR_CHECK(esp_eth_del_netif_glue(eth_netif_glue));
|
||||
esp_netif_destroy(eth_netif);
|
||||
esp_netif_deinit();
|
||||
ESP_ERROR_CHECK(example_eth_deinit(eth_handles, eth_port_cnt));
|
||||
ESP_ERROR_CHECK(eth_deinit(eth_handle));
|
||||
ESP_ERROR_CHECK(esp_event_handler_unregister(IP_EVENT, IP_EVENT_ETH_GOT_IP, got_ip_event_handler));
|
||||
ESP_ERROR_CHECK(esp_event_handler_unregister(ETH_EVENT, ESP_EVENT_ANY_ID, eth_event_handler));
|
||||
ESP_ERROR_CHECK(esp_event_loop_delete_default());
|
||||
|
||||
Reference in New Issue
Block a user