aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/dma/ste_dma40.c
diff options
context:
space:
mode:
authorRabin Vincent <rabin.vincent@stericsson.com>2011-01-25 05:18:35 -0500
committerDan Williams <dan.j.williams@intel.com>2011-01-31 01:27:21 -0500
commit0c842b551063c5f7382ac9b457992f3b34972801 (patch)
treeda606f382493e6cf3540b4245b86a784a78e46b4 /drivers/dma/ste_dma40.c
parent86eb5fb61125e4646c9447a1f2ce130817dab34e (diff)
dma40: cyclic xfer support
Support cyclic transfers, which are useful for ALSA drivers. Acked-by: Per Forlin <per.forlin@stericsson.com> Acked-by: Jonas Aaberg <jonas.aberg@stericsson.com> Signed-off-by: Rabin Vincent <rabin.vincent@stericsson.com> Signed-off-by: Linus Walleij <linus.walleij@stericsson.com> Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Diffstat (limited to 'drivers/dma/ste_dma40.c')
-rw-r--r--drivers/dma/ste_dma40.c172
1 files changed, 140 insertions, 32 deletions
diff --git a/drivers/dma/ste_dma40.c b/drivers/dma/ste_dma40.c
index 8fd0bb94e777..af955de035f4 100644
--- a/drivers/dma/ste_dma40.c
+++ b/drivers/dma/ste_dma40.c
@@ -115,6 +115,7 @@ struct d40_desc {
115 struct list_head node; 115 struct list_head node;
116 116
117 bool is_in_client_list; 117 bool is_in_client_list;
118 bool cyclic;
118}; 119};
119 120
120/** 121/**
@@ -527,17 +528,45 @@ static void d40_log_lli_to_lcxa(struct d40_chan *chan, struct d40_desc *desc)
527 struct d40_log_lli_bidir *lli = &desc->lli_log; 528 struct d40_log_lli_bidir *lli = &desc->lli_log;
528 int lli_current = desc->lli_current; 529 int lli_current = desc->lli_current;
529 int lli_len = desc->lli_len; 530 int lli_len = desc->lli_len;
531 bool cyclic = desc->cyclic;
530 int curr_lcla = -EINVAL; 532 int curr_lcla = -EINVAL;
533 int first_lcla = 0;
534 bool linkback;
531 535
532 if (lli_len - lli_current > 1) 536 /*
537 * We may have partially running cyclic transfers, in case we did't get
538 * enough LCLA entries.
539 */
540 linkback = cyclic && lli_current == 0;
541
542 /*
543 * For linkback, we need one LCLA even with only one link, because we
544 * can't link back to the one in LCPA space
545 */
546 if (linkback || (lli_len - lli_current > 1)) {
533 curr_lcla = d40_lcla_alloc_one(chan, desc); 547 curr_lcla = d40_lcla_alloc_one(chan, desc);
548 first_lcla = curr_lcla;
549 }
550
551 /*
552 * For linkback, we normally load the LCPA in the loop since we need to
553 * link it to the second LCLA and not the first. However, if we
554 * couldn't even get a first LCLA, then we have to run in LCPA and
555 * reload manually.
556 */
557 if (!linkback || curr_lcla == -EINVAL) {
558 unsigned int flags = 0;
534 559
535 d40_log_lli_lcpa_write(chan->lcpa, 560 if (curr_lcla == -EINVAL)
536 &lli->dst[lli_current], 561 flags |= LLI_TERM_INT;
537 &lli->src[lli_current],
538 curr_lcla);
539 562
540 lli_current++; 563 d40_log_lli_lcpa_write(chan->lcpa,
564 &lli->dst[lli_current],
565 &lli->src[lli_current],
566 curr_lcla,
567 flags);
568 lli_current++;
569 }
541 570
542 if (curr_lcla < 0) 571 if (curr_lcla < 0)
543 goto out; 572 goto out;
@@ -546,17 +575,33 @@ static void d40_log_lli_to_lcxa(struct d40_chan *chan, struct d40_desc *desc)
546 unsigned int lcla_offset = chan->phy_chan->num * 1024 + 575 unsigned int lcla_offset = chan->phy_chan->num * 1024 +
547 8 * curr_lcla * 2; 576 8 * curr_lcla * 2;
548 struct d40_log_lli *lcla = pool->base + lcla_offset; 577 struct d40_log_lli *lcla = pool->base + lcla_offset;
578 unsigned int flags = 0;
549 int next_lcla; 579 int next_lcla;
550 580
551 if (lli_current + 1 < lli_len) 581 if (lli_current + 1 < lli_len)
552 next_lcla = d40_lcla_alloc_one(chan, desc); 582 next_lcla = d40_lcla_alloc_one(chan, desc);
553 else 583 else
554 next_lcla = -EINVAL; 584 next_lcla = linkback ? first_lcla : -EINVAL;
585
586 if (cyclic || next_lcla == -EINVAL)
587 flags |= LLI_TERM_INT;
555 588
589 if (linkback && curr_lcla == first_lcla) {
590 /* First link goes in both LCPA and LCLA */
591 d40_log_lli_lcpa_write(chan->lcpa,
592 &lli->dst[lli_current],
593 &lli->src[lli_current],
594 next_lcla, flags);
595 }
596
597 /*
598 * One unused LCLA in the cyclic case if the very first
599 * next_lcla fails...
600 */
556 d40_log_lli_lcla_write(lcla, 601 d40_log_lli_lcla_write(lcla,
557 &lli->dst[lli_current], 602 &lli->dst[lli_current],
558 &lli->src[lli_current], 603 &lli->src[lli_current],
559 next_lcla); 604 next_lcla, flags);
560 605
561 dma_sync_single_range_for_device(chan->base->dev, 606 dma_sync_single_range_for_device(chan->base->dev,
562 pool->dma_addr, lcla_offset, 607 pool->dma_addr, lcla_offset,
@@ -565,7 +610,7 @@ static void d40_log_lli_to_lcxa(struct d40_chan *chan, struct d40_desc *desc)
565 610
566 curr_lcla = next_lcla; 611 curr_lcla = next_lcla;
567 612
568 if (curr_lcla == -EINVAL) { 613 if (curr_lcla == -EINVAL || curr_lcla == first_lcla) {
569 lli_current++; 614 lli_current++;
570 break; 615 break;
571 } 616 }
@@ -1074,17 +1119,36 @@ static void dma_tc_handle(struct d40_chan *d40c)
1074 if (d40d == NULL) 1119 if (d40d == NULL)
1075 return; 1120 return;
1076 1121
1077 d40_lcla_free_all(d40c, d40d); 1122 if (d40d->cyclic) {
1123 /*
1124 * If this was a paritially loaded list, we need to reloaded
1125 * it, and only when the list is completed. We need to check
1126 * for done because the interrupt will hit for every link, and
1127 * not just the last one.
1128 */
1129 if (d40d->lli_current < d40d->lli_len
1130 && !d40_tx_is_linked(d40c)
1131 && !d40_residue(d40c)) {
1132 d40_lcla_free_all(d40c, d40d);
1133 d40_desc_load(d40c, d40d);
1134 (void) d40_start(d40c);
1078 1135
1079 if (d40d->lli_current < d40d->lli_len) { 1136 if (d40d->lli_current == d40d->lli_len)
1080 d40_desc_load(d40c, d40d); 1137 d40d->lli_current = 0;
1081 /* Start dma job */ 1138 }
1082 (void) d40_start(d40c); 1139 } else {
1083 return; 1140 d40_lcla_free_all(d40c, d40d);
1084 }
1085 1141
1086 if (d40_queue_start(d40c) == NULL) 1142 if (d40d->lli_current < d40d->lli_len) {
1087 d40c->busy = false; 1143 d40_desc_load(d40c, d40d);
1144 /* Start dma job */
1145 (void) d40_start(d40c);
1146 return;
1147 }
1148
1149 if (d40_queue_start(d40c) == NULL)
1150 d40c->busy = false;
1151 }
1088 1152
1089 d40c->pending_tx++; 1153 d40c->pending_tx++;
1090 tasklet_schedule(&d40c->tasklet); 1154 tasklet_schedule(&d40c->tasklet);
@@ -1103,11 +1167,11 @@ static void dma_tasklet(unsigned long data)
1103 1167
1104 /* Get first active entry from list */ 1168 /* Get first active entry from list */
1105 d40d = d40_first_active_get(d40c); 1169 d40d = d40_first_active_get(d40c);
1106
1107 if (d40d == NULL) 1170 if (d40d == NULL)
1108 goto err; 1171 goto err;
1109 1172
1110 d40c->completed = d40d->txd.cookie; 1173 if (!d40d->cyclic)
1174 d40c->completed = d40d->txd.cookie;
1111 1175
1112 /* 1176 /*
1113 * If terminating a channel pending_tx is set to zero. 1177 * If terminating a channel pending_tx is set to zero.
@@ -1122,16 +1186,18 @@ static void dma_tasklet(unsigned long data)
1122 callback = d40d->txd.callback; 1186 callback = d40d->txd.callback;
1123 callback_param = d40d->txd.callback_param; 1187 callback_param = d40d->txd.callback_param;
1124 1188
1125 if (async_tx_test_ack(&d40d->txd)) { 1189 if (!d40d->cyclic) {
1126 d40_pool_lli_free(d40c, d40d); 1190 if (async_tx_test_ack(&d40d->txd)) {
1127 d40_desc_remove(d40d); 1191 d40_pool_lli_free(d40c, d40d);
1128 d40_desc_free(d40c, d40d);
1129 } else {
1130 if (!d40d->is_in_client_list) {
1131 d40_desc_remove(d40d); 1192 d40_desc_remove(d40d);
1132 d40_lcla_free_all(d40c, d40d); 1193 d40_desc_free(d40c, d40d);
1133 list_add_tail(&d40d->node, &d40c->client); 1194 } else {
1134 d40d->is_in_client_list = true; 1195 if (!d40d->is_in_client_list) {
1196 d40_desc_remove(d40d);
1197 d40_lcla_free_all(d40c, d40d);
1198 list_add_tail(&d40d->node, &d40c->client);
1199 d40d->is_in_client_list = true;
1200 }
1135 } 1201 }
1136 } 1202 }
1137 1203
@@ -1694,19 +1760,23 @@ d40_prep_sg_phy(struct d40_chan *chan, struct d40_desc *desc,
1694 struct stedma40_chan_cfg *cfg = &chan->dma_cfg; 1760 struct stedma40_chan_cfg *cfg = &chan->dma_cfg;
1695 struct stedma40_half_channel_info *src_info = &cfg->src_info; 1761 struct stedma40_half_channel_info *src_info = &cfg->src_info;
1696 struct stedma40_half_channel_info *dst_info = &cfg->dst_info; 1762 struct stedma40_half_channel_info *dst_info = &cfg->dst_info;
1763 unsigned long flags = 0;
1697 int ret; 1764 int ret;
1698 1765
1766 if (desc->cyclic)
1767 flags |= LLI_CYCLIC | LLI_TERM_INT;
1768
1699 ret = d40_phy_sg_to_lli(sg_src, sg_len, src_dev_addr, 1769 ret = d40_phy_sg_to_lli(sg_src, sg_len, src_dev_addr,
1700 desc->lli_phy.src, 1770 desc->lli_phy.src,
1701 virt_to_phys(desc->lli_phy.src), 1771 virt_to_phys(desc->lli_phy.src),
1702 chan->src_def_cfg, 1772 chan->src_def_cfg,
1703 src_info, dst_info); 1773 src_info, dst_info, flags);
1704 1774
1705 ret = d40_phy_sg_to_lli(sg_dst, sg_len, dst_dev_addr, 1775 ret = d40_phy_sg_to_lli(sg_dst, sg_len, dst_dev_addr,
1706 desc->lli_phy.dst, 1776 desc->lli_phy.dst,
1707 virt_to_phys(desc->lli_phy.dst), 1777 virt_to_phys(desc->lli_phy.dst),
1708 chan->dst_def_cfg, 1778 chan->dst_def_cfg,
1709 dst_info, src_info); 1779 dst_info, src_info, flags);
1710 1780
1711 dma_sync_single_for_device(chan->base->dev, desc->lli_pool.dma_addr, 1781 dma_sync_single_for_device(chan->base->dev, desc->lli_pool.dma_addr,
1712 desc->lli_pool.size, DMA_TO_DEVICE); 1782 desc->lli_pool.size, DMA_TO_DEVICE);
@@ -1789,12 +1859,16 @@ d40_prep_sg(struct dma_chan *dchan, struct scatterlist *sg_src,
1789 return NULL; 1859 return NULL;
1790 } 1860 }
1791 1861
1862
1792 spin_lock_irqsave(&chan->lock, flags); 1863 spin_lock_irqsave(&chan->lock, flags);
1793 1864
1794 desc = d40_prep_desc(chan, sg_src, sg_len, dma_flags); 1865 desc = d40_prep_desc(chan, sg_src, sg_len, dma_flags);
1795 if (desc == NULL) 1866 if (desc == NULL)
1796 goto err; 1867 goto err;
1797 1868
1869 if (sg_next(&sg_src[sg_len - 1]) == sg_src)
1870 desc->cyclic = true;
1871
1798 if (direction != DMA_NONE) { 1872 if (direction != DMA_NONE) {
1799 dma_addr_t dev_addr = d40_get_dev_addr(chan, direction); 1873 dma_addr_t dev_addr = d40_get_dev_addr(chan, direction);
1800 1874
@@ -2007,6 +2081,36 @@ static struct dma_async_tx_descriptor *d40_prep_slave_sg(struct dma_chan *chan,
2007 return d40_prep_sg(chan, sgl, sgl, sg_len, direction, dma_flags); 2081 return d40_prep_sg(chan, sgl, sgl, sg_len, direction, dma_flags);
2008} 2082}
2009 2083
2084static struct dma_async_tx_descriptor *
2085dma40_prep_dma_cyclic(struct dma_chan *chan, dma_addr_t dma_addr,
2086 size_t buf_len, size_t period_len,
2087 enum dma_data_direction direction)
2088{
2089 unsigned int periods = buf_len / period_len;
2090 struct dma_async_tx_descriptor *txd;
2091 struct scatterlist *sg;
2092 int i;
2093
2094 sg = kcalloc(periods + 1, sizeof(struct scatterlist), GFP_KERNEL);
2095 for (i = 0; i < periods; i++) {
2096 sg_dma_address(&sg[i]) = dma_addr;
2097 sg_dma_len(&sg[i]) = period_len;
2098 dma_addr += period_len;
2099 }
2100
2101 sg[periods].offset = 0;
2102 sg[periods].length = 0;
2103 sg[periods].page_link =
2104 ((unsigned long)sg | 0x01) & ~0x02;
2105
2106 txd = d40_prep_sg(chan, sg, sg, periods, direction,
2107 DMA_PREP_INTERRUPT);
2108
2109 kfree(sg);
2110
2111 return txd;
2112}
2113
2010static enum dma_status d40_tx_status(struct dma_chan *chan, 2114static enum dma_status d40_tx_status(struct dma_chan *chan,
2011 dma_cookie_t cookie, 2115 dma_cookie_t cookie,
2012 struct dma_tx_state *txstate) 2116 struct dma_tx_state *txstate)
@@ -2264,6 +2368,9 @@ static void d40_ops_init(struct d40_base *base, struct dma_device *dev)
2264 if (dma_has_cap(DMA_SG, dev->cap_mask)) 2368 if (dma_has_cap(DMA_SG, dev->cap_mask))
2265 dev->device_prep_dma_sg = d40_prep_memcpy_sg; 2369 dev->device_prep_dma_sg = d40_prep_memcpy_sg;
2266 2370
2371 if (dma_has_cap(DMA_CYCLIC, dev->cap_mask))
2372 dev->device_prep_dma_cyclic = dma40_prep_dma_cyclic;
2373
2267 dev->device_alloc_chan_resources = d40_alloc_chan_resources; 2374 dev->device_alloc_chan_resources = d40_alloc_chan_resources;
2268 dev->device_free_chan_resources = d40_free_chan_resources; 2375 dev->device_free_chan_resources = d40_free_chan_resources;
2269 dev->device_issue_pending = d40_issue_pending; 2376 dev->device_issue_pending = d40_issue_pending;
@@ -2282,6 +2389,7 @@ static int __init d40_dmaengine_init(struct d40_base *base,
2282 2389
2283 dma_cap_zero(base->dma_slave.cap_mask); 2390 dma_cap_zero(base->dma_slave.cap_mask);
2284 dma_cap_set(DMA_SLAVE, base->dma_slave.cap_mask); 2391 dma_cap_set(DMA_SLAVE, base->dma_slave.cap_mask);
2392 dma_cap_set(DMA_CYCLIC, base->dma_slave.cap_mask);
2285 2393
2286 d40_ops_init(base, &base->dma_slave); 2394 d40_ops_init(base, &base->dma_slave);
2287 2395
@@ -2316,9 +2424,9 @@ static int __init d40_dmaengine_init(struct d40_base *base,
2316 dma_cap_set(DMA_SLAVE, base->dma_both.cap_mask); 2424 dma_cap_set(DMA_SLAVE, base->dma_both.cap_mask);
2317 dma_cap_set(DMA_MEMCPY, base->dma_both.cap_mask); 2425 dma_cap_set(DMA_MEMCPY, base->dma_both.cap_mask);
2318 dma_cap_set(DMA_SG, base->dma_both.cap_mask); 2426 dma_cap_set(DMA_SG, base->dma_both.cap_mask);
2427 dma_cap_set(DMA_CYCLIC, base->dma_slave.cap_mask);
2319 2428
2320 d40_ops_init(base, &base->dma_both); 2429 d40_ops_init(base, &base->dma_both);
2321
2322 err = dma_async_device_register(&base->dma_both); 2430 err = dma_async_device_register(&base->dma_both);
2323 2431
2324 if (err) { 2432 if (err) {