Skip to content

Commit

Permalink
drivers: dma: esp32: remove block size limitation
Browse files Browse the repository at this point in the history
Removes block size limitation during dma rx/tx configuration

Signed-off-by: Marcio Ribeiro <[email protected]>
  • Loading branch information
wmrsouza committed Dec 17, 2024
1 parent 798122e commit f7572c4
Showing 1 changed file with 58 additions and 71 deletions.
129 changes: 58 additions & 71 deletions drivers/dma/dma_esp32_gdma.c
Original file line number Diff line number Diff line change
Expand Up @@ -137,46 +137,82 @@ static void IRAM_ATTR dma_esp32_isr_handle(const struct device *dev, uint8_t rx_
}
#endif

static int dma_esp32_config_rx_descriptor(struct dma_esp32_channel *dma_channel,
struct dma_block_config *block)
static int dma_esp32_config_descriptor(struct dma_esp32_channel *dma_channel,
struct dma_block_config *block, bool tx_dir)
{
if (!block) {
LOG_ERR("At least one dma block is required");
return -EINVAL;
}

if (!esp_ptr_dma_capable((uint32_t *)block->dest_address)
uint32_t target_address = 0, block_size = 0;
dma_descriptor_t *desc_iter = dma_channel->desc_list;

for (int i = 0; i < CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM; ++i) {
if (block_size == 0) {
if (tx_dir) {
target_address = block->source_address;
} else {
target_address = block->dest_address;
}

if (!esp_ptr_dma_capable((uint32_t *)target_address)
#if defined(CONFIG_ESP_SPIRAM)
&& !esp_ptr_dma_ext_capable((uint32_t *)block->dest_address)
&& !esp_ptr_dma_ext_capable((uint32_t *)target_address)
#endif
) {
LOG_ERR("Rx buffer not in DMA capable memory: %p", (uint32_t *)block->dest_address);
return -EINVAL;
}
) {
if (tx_dir) {
LOG_ERR("Tx buffer not in DMA capable memory: %p",
(uint32_t *)target_address);
} else {
LOG_ERR("Rx buffer not in DMA capable memory: %p",
(uint32_t *)target_address);
}

return -EINVAL;
}

dma_descriptor_t *desc_iter = dma_channel->desc_list;
block_size = block->block_size;
}

for (int i = 0; i < CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM; ++i) {
if (block->block_size > DMA_DESCRIPTOR_BUFFER_MAX_SIZE) {
LOG_ERR("Size of block %d is too large", i);
return -EINVAL;
uint32_t buffer_size;

if (block_size > DMA_DESCRIPTOR_BUFFER_MAX_SIZE) {
buffer_size = DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
} else {
buffer_size = block_size;
}

memset(desc_iter, 0, sizeof(dma_descriptor_t));
desc_iter->buffer = (void *)block->dest_address;
desc_iter->dw0.size = block->block_size;
desc_iter->buffer = (void *)target_address;
desc_iter->dw0.size = buffer_size;
if (tx_dir) {
desc_iter->dw0.length = buffer_size;
}
desc_iter->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
if (!block->next_block) {
desc_iter->next = NULL;
break;

target_address += buffer_size;
block_size -= buffer_size;

if (!block_size) {
if (block->next_block) {
block = block->next_block;
} else {
desc_iter->next = NULL;
if (tx_dir) {
desc_iter->dw0.suc_eof = 1;
}
break;
}
}

desc_iter->next = desc_iter + 1;
desc_iter += 1;
block = block->next_block;
}

if (desc_iter->next) {
memset(dma_channel->desc_list, 0, sizeof(dma_channel->desc_list));
LOG_ERR("Too many dma blocks. Increase CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM");
LOG_ERR("Run out of descriptors. Increase CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM");
return -EINVAL;
}

Expand Down Expand Up @@ -217,56 +253,7 @@ static int dma_esp32_config_rx(const struct device *dev, struct dma_esp32_channe
gdma_ll_rx_enable_interrupt(data->hal.dev, dma_channel->channel_id, UINT32_MAX,
config_dma->dma_callback != NULL);

return dma_esp32_config_rx_descriptor(dma_channel, config_dma->head_block);
}

static int dma_esp32_config_tx_descriptor(struct dma_esp32_channel *dma_channel,
struct dma_block_config *block)
{
if (!block) {
LOG_ERR("At least one dma block is required");
return -EINVAL;
}

if (!esp_ptr_dma_capable((uint32_t *)block->source_address)
#if defined(CONFIG_ESP_SPIRAM)
&& !esp_ptr_dma_ext_capable((uint32_t *)block->source_address)
#endif
) {
LOG_ERR("Tx buffer not in DMA capable memory: %p",
(uint32_t *)block->source_address);
return -EINVAL;
}

dma_descriptor_t *desc_iter = dma_channel->desc_list;

for (int i = 0; i < CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM; ++i) {
if (block->block_size > DMA_DESCRIPTOR_BUFFER_MAX_SIZE) {
LOG_ERR("Size of block %d is too large", i);
return -EINVAL;
}
memset(desc_iter, 0, sizeof(dma_descriptor_t));
desc_iter->buffer = (void *)block->source_address;
desc_iter->dw0.size = block->block_size;
desc_iter->dw0.length = block->block_size;
desc_iter->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
if (!block->next_block) {
desc_iter->next = NULL;
desc_iter->dw0.suc_eof = 1;
break;
}
desc_iter->next = desc_iter + 1;
desc_iter += 1;
block = block->next_block;
}

if (desc_iter->next) {
memset(dma_channel->desc_list, 0, sizeof(dma_channel->desc_list));
LOG_ERR("Too many dma blocks. Increase CONFIG_DMA_ESP32_MAX_DESCRIPTOR_NUM");
return -EINVAL;
}

return 0;
return dma_esp32_config_descriptor(dma_channel, config_dma->head_block, false);
}

static int dma_esp32_config_tx(const struct device *dev, struct dma_esp32_channel *dma_channel,
Expand Down Expand Up @@ -301,7 +288,7 @@ static int dma_esp32_config_tx(const struct device *dev, struct dma_esp32_channe
gdma_ll_tx_enable_interrupt(data->hal.dev, dma_channel->channel_id, GDMA_LL_EVENT_TX_EOF,
config_dma->dma_callback != NULL);

return dma_esp32_config_tx_descriptor(dma_channel, config_dma->head_block);
return dma_esp32_config_descriptor(dma_channel, config_dma->head_block, true);
}

static int dma_esp32_config(const struct device *dev, uint32_t channel,
Expand Down

0 comments on commit f7572c4

Please sign in to comment.