|
|
@@ -64,6 +64,9 @@ typedef struct {
|
|
|
#endif
|
|
|
} emac_esp32_t;
|
|
|
|
|
|
+static esp_err_t esp_emac_alloc_driver_obj(const eth_mac_config_t *config, emac_esp32_t **emac_out_hdl, void **out_descriptors);
|
|
|
+static void esp_emac_free_driver_obj(emac_esp32_t *emac, void *descriptors);
|
|
|
+
|
|
|
static esp_err_t emac_esp32_set_mediator(esp_eth_mac_t *mac, esp_eth_mediator_t *eth)
|
|
|
{
|
|
|
esp_err_t ret = ESP_OK;
|
|
|
@@ -289,10 +292,6 @@ static esp_err_t emac_esp32_init(esp_eth_mac_t *mac)
|
|
|
esp_err_t ret = ESP_OK;
|
|
|
emac_esp32_t *emac = __containerof(mac, emac_esp32_t, parent);
|
|
|
esp_eth_mediator_t *eth = emac->eth;
|
|
|
- /* enable peripheral clock */
|
|
|
- periph_module_enable(PERIPH_EMAC_MODULE);
|
|
|
- /* init clock, config gpio, etc */
|
|
|
- emac_hal_lowlevel_init(&emac->hal);
|
|
|
/* init gpio used by smi interface */
|
|
|
emac_esp32_init_smi_gpio(emac);
|
|
|
MAC_CHECK(eth->on_state_changed(eth, ETH_STATE_LLINIT, NULL) == ESP_OK, "lowlevel init failed", err, ESP_FAIL);
|
|
|
@@ -337,7 +336,6 @@ static esp_err_t emac_esp32_deinit(esp_eth_mac_t *mac)
|
|
|
#endif
|
|
|
emac_hal_stop(&emac->hal);
|
|
|
eth->on_state_changed(eth, ETH_STATE_DEINIT, NULL);
|
|
|
- periph_module_disable(PERIPH_EMAC_MODULE);
|
|
|
return ESP_OK;
|
|
|
}
|
|
|
|
|
|
@@ -358,22 +356,8 @@ static esp_err_t emac_esp32_stop(esp_eth_mac_t *mac)
|
|
|
static esp_err_t emac_esp32_del(esp_eth_mac_t *mac)
|
|
|
{
|
|
|
emac_esp32_t *emac = __containerof(mac, emac_esp32_t, parent);
|
|
|
- esp_intr_free(emac->intr_hdl);
|
|
|
-#ifdef CONFIG_PM_ENABLE
|
|
|
- if (emac->pm_lock) {
|
|
|
- esp_pm_lock_delete(emac->pm_lock);
|
|
|
- }
|
|
|
-#endif
|
|
|
- vTaskDelete(emac->rx_task_hdl);
|
|
|
- int i = 0;
|
|
|
- for (i = 0; i < CONFIG_ETH_DMA_RX_BUFFER_NUM; i++) {
|
|
|
- free(emac->hal.rx_buf[i]);
|
|
|
- }
|
|
|
- for (i = 0; i < CONFIG_ETH_DMA_TX_BUFFER_NUM; i++) {
|
|
|
- free(emac->hal.tx_buf[i]);
|
|
|
- }
|
|
|
- free(emac->hal.descriptors);
|
|
|
- free(emac);
|
|
|
+ esp_emac_free_driver_obj(emac, emac->hal.descriptors);
|
|
|
+ periph_module_disable(PERIPH_EMAC_MODULE);
|
|
|
return ESP_OK;
|
|
|
}
|
|
|
|
|
|
@@ -389,40 +373,106 @@ IRAM_ATTR void emac_esp32_isr_handler(void *args)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-esp_eth_mac_t *esp_eth_mac_new_esp32(const eth_mac_config_t *config)
|
|
|
+static void esp_emac_free_driver_obj(emac_esp32_t *emac, void *descriptors)
|
|
|
{
|
|
|
- esp_err_t ret_code = ESP_OK;
|
|
|
- esp_eth_mac_t *ret = NULL;
|
|
|
- void *descriptors = NULL;
|
|
|
+ if (emac) {
|
|
|
+ if (emac->rx_task_hdl) {
|
|
|
+ vTaskDelete(emac->rx_task_hdl);
|
|
|
+ }
|
|
|
+ if (emac->intr_hdl) {
|
|
|
+ esp_intr_free(emac->intr_hdl);
|
|
|
+ }
|
|
|
+ for (int i = 0; i < CONFIG_ETH_DMA_TX_BUFFER_NUM; i++) {
|
|
|
+ free(emac->tx_buf[i]);
|
|
|
+ }
|
|
|
+ for (int i = 0; i < CONFIG_ETH_DMA_RX_BUFFER_NUM; i++) {
|
|
|
+ free(emac->rx_buf[i]);
|
|
|
+ }
|
|
|
+#ifdef CONFIG_PM_ENABLE
|
|
|
+ if (emac->pm_lock) {
|
|
|
+ esp_pm_lock_delete(emac->pm_lock);
|
|
|
+ }
|
|
|
+#endif
|
|
|
+ free(emac);
|
|
|
+ }
|
|
|
+ if (descriptors) {
|
|
|
+ free(descriptors);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static esp_err_t esp_emac_alloc_driver_obj(const eth_mac_config_t *config, emac_esp32_t **emac_out_hdl, void **out_descriptors)
|
|
|
+{
|
|
|
+ esp_err_t ret = ESP_OK;
|
|
|
emac_esp32_t *emac = NULL;
|
|
|
- MAC_CHECK(config, "can't set mac config to null", err, NULL);
|
|
|
+ void *descriptors = NULL;
|
|
|
if (config->flags & ETH_MAC_FLAG_WORK_WITH_CACHE_DISABLE) {
|
|
|
emac = heap_caps_calloc(1, sizeof(emac_esp32_t), MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT);
|
|
|
} else {
|
|
|
emac = calloc(1, sizeof(emac_esp32_t));
|
|
|
}
|
|
|
- MAC_CHECK(emac, "calloc emac failed", err, NULL);
|
|
|
+ MAC_CHECK(emac, "no mem for esp emac object", err, ESP_ERR_NO_MEM);
|
|
|
/* alloc memory for ethernet dma descriptor */
|
|
|
uint32_t desc_size = CONFIG_ETH_DMA_RX_BUFFER_NUM * sizeof(eth_dma_rx_descriptor_t) +
|
|
|
CONFIG_ETH_DMA_TX_BUFFER_NUM * sizeof(eth_dma_tx_descriptor_t);
|
|
|
descriptors = heap_caps_calloc(1, desc_size, MALLOC_CAP_DMA);
|
|
|
- MAC_CHECK(descriptors, "calloc descriptors failed", err, NULL);
|
|
|
- int i = 0;
|
|
|
+ MAC_CHECK(descriptors, "no mem for descriptors", err, ESP_ERR_NO_MEM);
|
|
|
/* alloc memory for ethernet dma buffer */
|
|
|
- for (i = 0; i < CONFIG_ETH_DMA_RX_BUFFER_NUM; i++) {
|
|
|
+ for (int i = 0; i < CONFIG_ETH_DMA_RX_BUFFER_NUM; i++) {
|
|
|
emac->rx_buf[i] = heap_caps_calloc(1, CONFIG_ETH_DMA_BUFFER_SIZE, MALLOC_CAP_DMA);
|
|
|
- if (!(emac->rx_buf[i])) {
|
|
|
- goto err;
|
|
|
- }
|
|
|
+ MAC_CHECK(emac->rx_buf[i], "no mem for RX DMA buffers", err, ESP_ERR_NO_MEM);
|
|
|
}
|
|
|
- for (i = 0; i < CONFIG_ETH_DMA_TX_BUFFER_NUM; i++) {
|
|
|
+ for (int i = 0; i < CONFIG_ETH_DMA_TX_BUFFER_NUM; i++) {
|
|
|
emac->tx_buf[i] = heap_caps_calloc(1, CONFIG_ETH_DMA_BUFFER_SIZE, MALLOC_CAP_DMA);
|
|
|
- if (!(emac->tx_buf[i])) {
|
|
|
- goto err;
|
|
|
- }
|
|
|
+ MAC_CHECK(emac->tx_buf[i], "no mem for TX DMA buffers", err, ESP_ERR_NO_MEM);
|
|
|
}
|
|
|
+ /* alloc PM lock */
|
|
|
+#ifdef CONFIG_PM_ENABLE
|
|
|
+ MAC_CHECK(esp_pm_lock_create(ESP_PM_APB_FREQ_MAX, 0, "emac_esp32", &emac->pm_lock) == ESP_OK,
|
|
|
+ "create pm lock failed", err, ESP_FAIL);
|
|
|
+#endif
|
|
|
+ /* create rx task */
|
|
|
+ BaseType_t core_num = tskNO_AFFINITY;
|
|
|
+ if (config->flags & ETH_MAC_FLAG_PIN_TO_CORE) {
|
|
|
+ core_num = cpu_hal_get_core_id();
|
|
|
+ }
|
|
|
+ BaseType_t xReturned = xTaskCreatePinnedToCore(emac_esp32_rx_task, "emac_rx", config->rx_task_stack_size, emac,
|
|
|
+ config->rx_task_prio, &emac->rx_task_hdl, core_num);
|
|
|
+ MAC_CHECK(xReturned == pdPASS, "create emac_rx task failed", err, ESP_FAIL);
|
|
|
+
|
|
|
+ *out_descriptors = descriptors;
|
|
|
+ *emac_out_hdl = emac;
|
|
|
+ return ESP_OK;
|
|
|
+err:
|
|
|
+ esp_emac_free_driver_obj(emac, descriptors);
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+esp_eth_mac_t *esp_eth_mac_new_esp32(const eth_mac_config_t *config)
|
|
|
+{
|
|
|
+ esp_err_t ret_code = ESP_OK;
|
|
|
+ esp_eth_mac_t *ret = NULL;
|
|
|
+ void *descriptors = NULL;
|
|
|
+ emac_esp32_t *emac = NULL;
|
|
|
+ MAC_CHECK(config, "can't set mac config to null", err, NULL);
|
|
|
+ ret_code = esp_emac_alloc_driver_obj(config, &emac, &descriptors);
|
|
|
+ MAC_CHECK(ret_code == ESP_OK, "alloc driver object failed", err, NULL);
|
|
|
+
|
|
|
+ /* enable APB to access Ethernet peripheral registers */
|
|
|
+ periph_module_enable(PERIPH_EMAC_MODULE);
|
|
|
/* initialize hal layer driver */
|
|
|
emac_hal_init(&emac->hal, descriptors, emac->rx_buf, emac->tx_buf);
|
|
|
+ // config emac data interface
|
|
|
+ emac_hal_lowlevel_init(&emac->hal);
|
|
|
+ /* alloc interrupt */
|
|
|
+ if (config->flags & ETH_MAC_FLAG_WORK_WITH_CACHE_DISABLE) {
|
|
|
+ ret_code = esp_intr_alloc(ETS_ETH_MAC_INTR_SOURCE, ESP_INTR_FLAG_IRAM,
|
|
|
+ emac_esp32_isr_handler, &emac->hal, &(emac->intr_hdl));
|
|
|
+ } else {
|
|
|
+ ret_code = esp_intr_alloc(ETS_ETH_MAC_INTR_SOURCE, 0,
|
|
|
+ emac_esp32_isr_handler, &emac->hal, &(emac->intr_hdl));
|
|
|
+ }
|
|
|
+ MAC_CHECK(ret_code == ESP_OK, "alloc emac interrupt failed", err_intr, NULL);
|
|
|
+
|
|
|
emac->sw_reset_timeout_ms = config->sw_reset_timeout_ms;
|
|
|
emac->smi_mdc_gpio_num = config->smi_mdc_gpio_num;
|
|
|
emac->smi_mdio_gpio_num = config->smi_mdio_gpio_num;
|
|
|
@@ -442,53 +492,12 @@ esp_eth_mac_t *esp_eth_mac_new_esp32(const eth_mac_config_t *config)
|
|
|
emac->parent.set_promiscuous = emac_esp32_set_promiscuous;
|
|
|
emac->parent.transmit = emac_esp32_transmit;
|
|
|
emac->parent.receive = emac_esp32_receive;
|
|
|
- /* Interrupt configuration */
|
|
|
- if (config->flags & ETH_MAC_FLAG_WORK_WITH_CACHE_DISABLE) {
|
|
|
- ret_code = esp_intr_alloc(ETS_ETH_MAC_INTR_SOURCE, ESP_INTR_FLAG_IRAM,
|
|
|
- emac_esp32_isr_handler, &emac->hal, &(emac->intr_hdl));
|
|
|
- } else {
|
|
|
- ret_code = esp_intr_alloc(ETS_ETH_MAC_INTR_SOURCE, 0,
|
|
|
- emac_esp32_isr_handler, &emac->hal, &(emac->intr_hdl));
|
|
|
- }
|
|
|
- MAC_CHECK(ret_code == ESP_OK, "alloc emac interrupt failed", err, NULL);
|
|
|
-#ifdef CONFIG_PM_ENABLE
|
|
|
- MAC_CHECK(esp_pm_lock_create(ESP_PM_APB_FREQ_MAX, 0, "emac_esp32", &emac->pm_lock) == ESP_OK,
|
|
|
- "create pm lock failed", err, NULL);
|
|
|
-#endif
|
|
|
- /* create rx task */
|
|
|
- BaseType_t core_num = tskNO_AFFINITY;
|
|
|
- if (config->flags & ETH_MAC_FLAG_PIN_TO_CORE) {
|
|
|
- core_num = cpu_hal_get_core_id();
|
|
|
- }
|
|
|
- BaseType_t xReturned = xTaskCreatePinnedToCore(emac_esp32_rx_task, "emac_rx", config->rx_task_stack_size, emac,
|
|
|
- config->rx_task_prio, &emac->rx_task_hdl, core_num);
|
|
|
- MAC_CHECK(xReturned == pdPASS, "create emac_rx task failed", err, NULL);
|
|
|
return &(emac->parent);
|
|
|
|
|
|
+err_intr:
|
|
|
+ periph_module_disable(PERIPH_EMAC_MODULE);
|
|
|
err:
|
|
|
- if (emac) {
|
|
|
- if (emac->rx_task_hdl) {
|
|
|
- vTaskDelete(emac->rx_task_hdl);
|
|
|
- }
|
|
|
- if (emac->intr_hdl) {
|
|
|
- esp_intr_free(emac->intr_hdl);
|
|
|
- }
|
|
|
- for (int i = 0; i < CONFIG_ETH_DMA_TX_BUFFER_NUM; i++) {
|
|
|
- free(emac->tx_buf[i]);
|
|
|
- }
|
|
|
- for (int i = 0; i < CONFIG_ETH_DMA_RX_BUFFER_NUM; i++) {
|
|
|
- free(emac->rx_buf[i]);
|
|
|
- }
|
|
|
-#ifdef CONFIG_PM_ENABLE
|
|
|
- if (emac->pm_lock) {
|
|
|
- esp_pm_lock_delete(emac->pm_lock);
|
|
|
- }
|
|
|
-#endif
|
|
|
- free(emac);
|
|
|
- }
|
|
|
- if (descriptors) {
|
|
|
- free(descriptors);
|
|
|
- }
|
|
|
+ esp_emac_free_driver_obj(emac, descriptors);
|
|
|
return ret;
|
|
|
}
|
|
|
|