Commit 4e538578 authored by Maxime Ripard's avatar Maxime Ripard Committed by Vinod Koul

dmaengine: at_xdmac: handle numf > 1

Handle 'numf > 1' case for interleaved mode.
Signed-off-by: default avatarMaxime Ripard <maxime.ripard@free-electrons.com>
Signed-off-by: default avatarLudovic Desroches <ludovic.desroches@atmel.com>
Signed-off-by: default avatarVinod Koul <vinod.koul@intel.com>
parent 1a492ac2
...@@ -929,13 +929,19 @@ at_xdmac_prep_interleaved(struct dma_chan *chan, ...@@ -929,13 +929,19 @@ at_xdmac_prep_interleaved(struct dma_chan *chan,
{ {
struct at_xdmac_chan *atchan = to_at_xdmac_chan(chan); struct at_xdmac_chan *atchan = to_at_xdmac_chan(chan);
struct at_xdmac_desc *prev = NULL, *first = NULL; struct at_xdmac_desc *prev = NULL, *first = NULL;
struct data_chunk *chunk, *prev_chunk = NULL;
dma_addr_t dst_addr, src_addr; dma_addr_t dst_addr, src_addr;
size_t dst_skip, src_skip, len = 0; size_t src_skip = 0, dst_skip = 0, len = 0;
size_t prev_dst_icg = 0, prev_src_icg = 0; struct data_chunk *chunk;
int i; int i;
if (!xt || (xt->numf != 1) || (xt->dir != DMA_MEM_TO_MEM)) if (!xt || !xt->numf || (xt->dir != DMA_MEM_TO_MEM))
return NULL;
/*
* TODO: Handle the case where we have to repeat a chain of
* descriptors...
*/
if ((xt->numf > 1) && (xt->frame_size > 1))
return NULL; return NULL;
dev_dbg(chan2dev(chan), "%s: src=0x%08x, dest=0x%08x, numf=%d, frame_size=%d, flags=0x%lx\n", dev_dbg(chan2dev(chan), "%s: src=0x%08x, dest=0x%08x, numf=%d, frame_size=%d, flags=0x%lx\n",
...@@ -945,66 +951,56 @@ at_xdmac_prep_interleaved(struct dma_chan *chan, ...@@ -945,66 +951,56 @@ at_xdmac_prep_interleaved(struct dma_chan *chan,
src_addr = xt->src_start; src_addr = xt->src_start;
dst_addr = xt->dst_start; dst_addr = xt->dst_start;
for (i = 0; i < xt->frame_size; i++) { if (xt->numf > 1) {
struct at_xdmac_desc *desc; first = at_xdmac_interleaved_queue_desc(chan, atchan,
size_t src_icg, dst_icg; NULL,
src_addr, dst_addr,
chunk = xt->sgl + i; xt, xt->sgl);
for (i = 0; i < xt->numf; i++)
at_xdmac_increment_block_count(chan, first);
} else {
for (i = 0; i < xt->frame_size; i++) {
size_t src_icg = 0, dst_icg = 0;
struct at_xdmac_desc *desc;
dst_icg = dmaengine_get_dst_icg(xt, chunk); chunk = xt->sgl + i;
src_icg = dmaengine_get_src_icg(xt, chunk);
src_skip = chunk->size + src_icg; dst_icg = dmaengine_get_dst_icg(xt, chunk);
dst_skip = chunk->size + dst_icg; src_icg = dmaengine_get_src_icg(xt, chunk);
dev_dbg(chan2dev(chan), src_skip = chunk->size + src_icg;
"%s: chunk size=%d, src icg=%d, dst icg=%d\n", dst_skip = chunk->size + dst_icg;
__func__, chunk->size, src_icg, dst_icg);
/*
* Handle the case where we just have the same
* transfer to setup, we can just increase the
* block number and reuse the same descriptor.
*/
if (prev_chunk && prev &&
(prev_chunk->size == chunk->size) &&
(prev_src_icg == src_icg) &&
(prev_dst_icg == dst_icg)) {
dev_dbg(chan2dev(chan), dev_dbg(chan2dev(chan),
"%s: same configuration that the previous chunk, merging the descriptors...\n", "%s: chunk size=%d, src icg=%d, dst icg=%d\n",
__func__); __func__, chunk->size, src_icg, dst_icg);
at_xdmac_increment_block_count(chan, prev);
continue; desc = at_xdmac_interleaved_queue_desc(chan, atchan,
} prev,
src_addr, dst_addr,
desc = at_xdmac_interleaved_queue_desc(chan, atchan, xt, chunk);
prev, if (!desc) {
src_addr, dst_addr, list_splice_init(&first->descs_list,
xt, chunk); &atchan->free_descs_list);
if (!desc) { return NULL;
list_splice_init(&first->descs_list, }
&atchan->free_descs_list);
return NULL;
}
if (!first) if (!first)
first = desc; first = desc;
dev_dbg(chan2dev(chan), "%s: add desc 0x%p to descs_list 0x%p\n", dev_dbg(chan2dev(chan), "%s: add desc 0x%p to descs_list 0x%p\n",
__func__, desc, first); __func__, desc, first);
list_add_tail(&desc->desc_node, &first->descs_list); list_add_tail(&desc->desc_node, &first->descs_list);
if (xt->src_sgl) if (xt->src_sgl)
src_addr += src_skip; src_addr += src_skip;
if (xt->dst_sgl) if (xt->dst_sgl)
dst_addr += dst_skip; dst_addr += dst_skip;
len += chunk->size; len += chunk->size;
prev_chunk = chunk; prev = desc;
prev_dst_icg = dst_icg; }
prev_src_icg = src_icg;
prev = desc;
} }
first->tx_dma_desc.cookie = -EBUSY; first->tx_dma_desc.cookie = -EBUSY;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment