2023-06-21 19:00:59 +08:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "soc/soc_caps.h"
|
|
|
|
#include "hal/assert.h"
|
|
|
|
#include "hal/gdma_hal_ahb.h"
|
|
|
|
#include "hal/gdma_ll.h"
|
|
|
|
|
|
|
|
static gdma_hal_priv_data_t gdma_ahb_hal_priv_data = {
|
|
|
|
.m2m_free_periph_mask = GDMA_LL_M2M_FREE_PERIPH_ID_MASK,
|
|
|
|
};
|
|
|
|
|
|
|
|
void gdma_ahb_hal_start_with_desc(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, intptr_t desc_base_addr)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_set_desc_addr(hal->dev, chan_id, desc_base_addr);
|
|
|
|
gdma_ll_rx_start(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_set_desc_addr(hal->dev, chan_id, desc_base_addr);
|
|
|
|
gdma_ll_tx_start(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_stop(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_stop(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_stop(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_append(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_restart(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_restart(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_reset(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_reset_channel(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_reset_channel(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_set_priority(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, uint32_t priority)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_set_priority(hal->dev, chan_id, priority);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_set_priority(hal->dev, chan_id, priority);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_connect_peri(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, gdma_trigger_peripheral_t periph, int periph_sub_id)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_reset_channel(hal->dev, chan_id); // reset channel
|
|
|
|
gdma_ll_rx_connect_to_periph(hal->dev, chan_id, periph, periph_sub_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_reset_channel(hal->dev, chan_id); // reset channel
|
|
|
|
gdma_ll_tx_connect_to_periph(hal->dev, chan_id, periph, periph_sub_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_disconnect_peri(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_disconnect_from_periph(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_disconnect_from_periph(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_enable_burst(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, bool en_data_burst, bool en_desc_burst)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_enable_data_burst(hal->dev, chan_id, en_data_burst);
|
|
|
|
gdma_ll_rx_enable_descriptor_burst(hal->dev, chan_id, en_desc_burst);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_enable_data_burst(hal->dev, chan_id, en_data_burst);
|
|
|
|
gdma_ll_tx_enable_descriptor_burst(hal->dev, chan_id, en_desc_burst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if SOC_AHB_GDMA_SUPPORT_PSRAM
|
|
|
|
void gdma_ahb_hal_set_ext_mem_align(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, uint8_t align)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_set_ext_mem_block_size(hal->dev, chan_id, align);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_set_ext_mem_block_size(hal->dev, chan_id, align);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2024-03-27 16:33:42 +08:00
|
|
|
void gdma_ahb_hal_set_strategy(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, bool en_owner_check, bool en_desc_write_back, bool eof_till_popped)
|
2023-06-21 19:00:59 +08:00
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_enable_owner_check(hal->dev, chan_id, en_owner_check);
|
2023-09-25 11:36:16 +08:00
|
|
|
// RX direction always has the descriptor write-back feature enabled
|
2024-03-27 16:33:42 +08:00
|
|
|
// RX direction don't need config eof_mode
|
2023-06-21 19:00:59 +08:00
|
|
|
} else {
|
|
|
|
gdma_ll_tx_enable_owner_check(hal->dev, chan_id, en_owner_check);
|
|
|
|
gdma_ll_tx_enable_auto_write_back(hal->dev, chan_id, en_desc_write_back);
|
2024-03-27 16:33:42 +08:00
|
|
|
gdma_ll_tx_set_eof_mode(hal->dev, chan_id, eof_till_popped);
|
2023-06-21 19:00:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_enable_intr(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, uint32_t intr_event_mask, bool en_or_dis)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_enable_interrupt(hal->dev, chan_id, intr_event_mask, en_or_dis);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_enable_interrupt(hal->dev, chan_id, intr_event_mask, en_or_dis);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gdma_ahb_hal_clear_intr(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, uint32_t intr_event_mask)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_clear_interrupt_status(hal->dev, chan_id, intr_event_mask);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_clear_interrupt_status(hal->dev, chan_id, intr_event_mask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-15 16:33:48 +08:00
|
|
|
uint32_t gdma_ahb_hal_read_intr_status(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, bool raw)
|
2023-06-21 19:00:59 +08:00
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
2023-11-15 16:33:48 +08:00
|
|
|
return gdma_ll_rx_get_interrupt_status(hal->dev, chan_id, raw);
|
2023-06-21 19:00:59 +08:00
|
|
|
} else {
|
2023-11-15 16:33:48 +08:00
|
|
|
return gdma_ll_tx_get_interrupt_status(hal->dev, chan_id, raw);
|
2023-06-21 19:00:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t gdma_ahb_hal_get_intr_status_reg(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
return (uint32_t)gdma_ll_rx_get_interrupt_status_reg(hal->dev, chan_id);
|
|
|
|
} else {
|
|
|
|
return (uint32_t)gdma_ll_tx_get_interrupt_status_reg(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-24 12:22:31 +08:00
|
|
|
uint32_t gdma_ahb_hal_get_eof_desc_addr(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, bool is_success)
|
2023-06-21 19:00:59 +08:00
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
2023-07-24 12:22:31 +08:00
|
|
|
if (is_success) {
|
|
|
|
return gdma_ll_rx_get_success_eof_desc_addr(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
return gdma_ll_rx_get_error_eof_desc_addr(hal->dev, chan_id);
|
2023-06-21 19:00:59 +08:00
|
|
|
} else {
|
2023-07-24 12:22:31 +08:00
|
|
|
// The TX direction only has success EOF, parameter 'is_success' is ignored
|
2023-06-21 19:00:59 +08:00
|
|
|
return gdma_ll_tx_get_eof_desc_addr(hal->dev, chan_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-06 11:48:04 +08:00
|
|
|
#if SOC_GDMA_SUPPORT_ETM
|
|
|
|
void gdma_ahb_hal_enable_etm_task(gdma_hal_context_t *hal, int chan_id, gdma_channel_direction_t dir, bool en_or_dis)
|
|
|
|
{
|
|
|
|
if (dir == GDMA_CHANNEL_DIRECTION_RX) {
|
|
|
|
gdma_ll_rx_enable_etm_task(hal->dev, chan_id, en_or_dis);
|
|
|
|
} else {
|
|
|
|
gdma_ll_tx_enable_etm_task(hal->dev, chan_id, en_or_dis);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // SOC_GDMA_SUPPORT_ETM
|
|
|
|
|
2023-06-21 19:00:59 +08:00
|
|
|
void gdma_ahb_hal_init(gdma_hal_context_t *hal, const gdma_hal_config_t *config)
|
|
|
|
{
|
|
|
|
hal->dev = GDMA_LL_GET_HW(config->group_id - GDMA_LL_AHB_GROUP_START_ID);
|
|
|
|
hal->start_with_desc = gdma_ahb_hal_start_with_desc;
|
|
|
|
hal->stop = gdma_ahb_hal_stop;
|
|
|
|
hal->append = gdma_ahb_hal_append;
|
|
|
|
hal->reset = gdma_ahb_hal_reset;
|
|
|
|
hal->set_priority = gdma_ahb_hal_set_priority;
|
|
|
|
hal->connect_peri = gdma_ahb_hal_connect_peri;
|
|
|
|
hal->disconnect_peri = gdma_ahb_hal_disconnect_peri;
|
|
|
|
hal->enable_burst = gdma_ahb_hal_enable_burst;
|
|
|
|
hal->set_strategy = gdma_ahb_hal_set_strategy;
|
|
|
|
hal->enable_intr = gdma_ahb_hal_enable_intr;
|
|
|
|
hal->clear_intr = gdma_ahb_hal_clear_intr;
|
|
|
|
hal->read_intr_status = gdma_ahb_hal_read_intr_status;
|
|
|
|
hal->get_intr_status_reg = gdma_ahb_hal_get_intr_status_reg;
|
|
|
|
hal->get_eof_desc_addr = gdma_ahb_hal_get_eof_desc_addr;
|
2023-12-06 11:48:04 +08:00
|
|
|
#if SOC_GDMA_SUPPORT_ETM
|
|
|
|
hal->enable_etm_task = gdma_ahb_hal_enable_etm_task;
|
|
|
|
#endif
|
2023-06-21 19:00:59 +08:00
|
|
|
#if SOC_AHB_GDMA_SUPPORT_PSRAM
|
|
|
|
hal->set_ext_mem_align = gdma_ahb_hal_set_ext_mem_align;
|
|
|
|
#endif // SOC_AHB_GDMA_SUPPORT_PSRAM
|
|
|
|
hal->priv_data = &gdma_ahb_hal_priv_data;
|
|
|
|
}
|