diff options
author | Linus Walleij <linus.walleij@stericsson.com> | 2010-03-26 19:44:01 -0400 |
---|---|---|
committer | Dan Williams <dan.j.williams@intel.com> | 2010-03-26 19:44:01 -0400 |
commit | c3635c78e500a52c9fcd55de381a72928d9e054d (patch) | |
tree | 87403f402227cd8b5572550e70facf81c9eaa0d9 /drivers/dma/coh901318.c | |
parent | 0f65169b1bf44220308e1ce1f6666ad03ddc27af (diff) |
DMAENGINE: generic slave control v2
Convert the device_terminate_all() operation on the
DMA engine to a generic device_control() operation
which can now optionally support also pausing and
resuming DMA on a certain channel. Implemented for the
COH 901 318 DMAC as an example.
[dan.j.williams@intel.com: update for timberdale]
Signed-off-by: Linus Walleij <linus.walleij@stericsson.com>
Acked-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
Cc: Maciej Sosnowski <maciej.sosnowski@intel.com>
Cc: Nicolas Ferre <nicolas.ferre@atmel.com>
Cc: Pavel Machek <pavel@ucw.cz>
Cc: Li Yang <leoli@freescale.com>
Cc: Guennadi Liakhovetski <g.liakhovetski@gmx.de>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: Haavard Skinnemoen <haavard.skinnemoen@atmel.com>
Cc: Magnus Damm <damm@opensource.se>
Cc: Liam Girdwood <lrg@slimlogic.co.uk>
Cc: Joe Perches <joe@perches.com>
Cc: Roland Dreier <rdreier@cisco.com>
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Diffstat (limited to 'drivers/dma/coh901318.c')
-rw-r--r-- | drivers/dma/coh901318.c | 42 |
1 files changed, 28 insertions, 14 deletions
diff --git a/drivers/dma/coh901318.c b/drivers/dma/coh901318.c index f636c4a87c7f..53c54e034aa3 100644 --- a/drivers/dma/coh901318.c +++ b/drivers/dma/coh901318.c | |||
@@ -506,10 +506,11 @@ u32 coh901318_get_bytes_left(struct dma_chan *chan) | |||
506 | EXPORT_SYMBOL(coh901318_get_bytes_left); | 506 | EXPORT_SYMBOL(coh901318_get_bytes_left); |
507 | 507 | ||
508 | 508 | ||
509 | /* Stops a transfer without losing data. Enables power save. | 509 | /* |
510 | Use this function in conjunction with coh901318_continue(..) | 510 | * Pauses a transfer without losing data. Enables power save. |
511 | */ | 511 | * Use this function in conjunction with coh901318_resume. |
512 | void coh901318_stop(struct dma_chan *chan) | 512 | */ |
513 | static void coh901318_pause(struct dma_chan *chan) | ||
513 | { | 514 | { |
514 | u32 val; | 515 | u32 val; |
515 | unsigned long flags; | 516 | unsigned long flags; |
@@ -550,12 +551,11 @@ void coh901318_stop(struct dma_chan *chan) | |||
550 | 551 | ||
551 | spin_unlock_irqrestore(&cohc->lock, flags); | 552 | spin_unlock_irqrestore(&cohc->lock, flags); |
552 | } | 553 | } |
553 | EXPORT_SYMBOL(coh901318_stop); | ||
554 | 554 | ||
555 | /* Continues a transfer that has been stopped via 300_dma_stop(..). | 555 | /* Resumes a transfer that has been stopped via 300_dma_stop(..). |
556 | Power save is handled. | 556 | Power save is handled. |
557 | */ | 557 | */ |
558 | void coh901318_continue(struct dma_chan *chan) | 558 | static void coh901318_resume(struct dma_chan *chan) |
559 | { | 559 | { |
560 | u32 val; | 560 | u32 val; |
561 | unsigned long flags; | 561 | unsigned long flags; |
@@ -581,7 +581,6 @@ void coh901318_continue(struct dma_chan *chan) | |||
581 | 581 | ||
582 | spin_unlock_irqrestore(&cohc->lock, flags); | 582 | spin_unlock_irqrestore(&cohc->lock, flags); |
583 | } | 583 | } |
584 | EXPORT_SYMBOL(coh901318_continue); | ||
585 | 584 | ||
586 | bool coh901318_filter_id(struct dma_chan *chan, void *chan_id) | 585 | bool coh901318_filter_id(struct dma_chan *chan, void *chan_id) |
587 | { | 586 | { |
@@ -945,7 +944,7 @@ coh901318_free_chan_resources(struct dma_chan *chan) | |||
945 | 944 | ||
946 | spin_unlock_irqrestore(&cohc->lock, flags); | 945 | spin_unlock_irqrestore(&cohc->lock, flags); |
947 | 946 | ||
948 | chan->device->device_terminate_all(chan); | 947 | chan->device->device_control(chan, DMA_TERMINATE_ALL); |
949 | } | 948 | } |
950 | 949 | ||
951 | 950 | ||
@@ -1179,16 +1178,29 @@ coh901318_issue_pending(struct dma_chan *chan) | |||
1179 | spin_unlock_irqrestore(&cohc->lock, flags); | 1178 | spin_unlock_irqrestore(&cohc->lock, flags); |
1180 | } | 1179 | } |
1181 | 1180 | ||
1182 | static void | 1181 | static int |
1183 | coh901318_terminate_all(struct dma_chan *chan) | 1182 | coh901318_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd) |
1184 | { | 1183 | { |
1185 | unsigned long flags; | 1184 | unsigned long flags; |
1186 | struct coh901318_chan *cohc = to_coh901318_chan(chan); | 1185 | struct coh901318_chan *cohc = to_coh901318_chan(chan); |
1187 | struct coh901318_desc *cohd; | 1186 | struct coh901318_desc *cohd; |
1188 | void __iomem *virtbase = cohc->base->virtbase; | 1187 | void __iomem *virtbase = cohc->base->virtbase; |
1189 | 1188 | ||
1190 | coh901318_stop(chan); | 1189 | if (cmd == DMA_PAUSE) { |
1190 | coh901318_pause(chan); | ||
1191 | return 0; | ||
1192 | } | ||
1193 | |||
1194 | if (cmd == DMA_RESUME) { | ||
1195 | coh901318_resume(chan); | ||
1196 | return 0; | ||
1197 | } | ||
1198 | |||
1199 | if (cmd != DMA_TERMINATE_ALL) | ||
1200 | return -ENXIO; | ||
1191 | 1201 | ||
1202 | /* The remainder of this function terminates the transfer */ | ||
1203 | coh901318_pause(chan); | ||
1192 | spin_lock_irqsave(&cohc->lock, flags); | 1204 | spin_lock_irqsave(&cohc->lock, flags); |
1193 | 1205 | ||
1194 | /* Clear any pending BE or TC interrupt */ | 1206 | /* Clear any pending BE or TC interrupt */ |
@@ -1227,6 +1239,8 @@ coh901318_terminate_all(struct dma_chan *chan) | |||
1227 | cohc->busy = 0; | 1239 | cohc->busy = 0; |
1228 | 1240 | ||
1229 | spin_unlock_irqrestore(&cohc->lock, flags); | 1241 | spin_unlock_irqrestore(&cohc->lock, flags); |
1242 | |||
1243 | return 0; | ||
1230 | } | 1244 | } |
1231 | void coh901318_base_init(struct dma_device *dma, const int *pick_chans, | 1245 | void coh901318_base_init(struct dma_device *dma, const int *pick_chans, |
1232 | struct coh901318_base *base) | 1246 | struct coh901318_base *base) |
@@ -1344,7 +1358,7 @@ static int __init coh901318_probe(struct platform_device *pdev) | |||
1344 | base->dma_slave.device_prep_slave_sg = coh901318_prep_slave_sg; | 1358 | base->dma_slave.device_prep_slave_sg = coh901318_prep_slave_sg; |
1345 | base->dma_slave.device_is_tx_complete = coh901318_is_tx_complete; | 1359 | base->dma_slave.device_is_tx_complete = coh901318_is_tx_complete; |
1346 | base->dma_slave.device_issue_pending = coh901318_issue_pending; | 1360 | base->dma_slave.device_issue_pending = coh901318_issue_pending; |
1347 | base->dma_slave.device_terminate_all = coh901318_terminate_all; | 1361 | base->dma_slave.device_control = coh901318_control; |
1348 | base->dma_slave.dev = &pdev->dev; | 1362 | base->dma_slave.dev = &pdev->dev; |
1349 | 1363 | ||
1350 | err = dma_async_device_register(&base->dma_slave); | 1364 | err = dma_async_device_register(&base->dma_slave); |
@@ -1364,7 +1378,7 @@ static int __init coh901318_probe(struct platform_device *pdev) | |||
1364 | base->dma_memcpy.device_prep_dma_memcpy = coh901318_prep_memcpy; | 1378 | base->dma_memcpy.device_prep_dma_memcpy = coh901318_prep_memcpy; |
1365 | base->dma_memcpy.device_is_tx_complete = coh901318_is_tx_complete; | 1379 | base->dma_memcpy.device_is_tx_complete = coh901318_is_tx_complete; |
1366 | base->dma_memcpy.device_issue_pending = coh901318_issue_pending; | 1380 | base->dma_memcpy.device_issue_pending = coh901318_issue_pending; |
1367 | base->dma_memcpy.device_terminate_all = coh901318_terminate_all; | 1381 | base->dma_memcpy.device_control = coh901318_control; |
1368 | base->dma_memcpy.dev = &pdev->dev; | 1382 | base->dma_memcpy.dev = &pdev->dev; |
1369 | /* | 1383 | /* |
1370 | * This controller can only access address at even 32bit boundaries, | 1384 | * This controller can only access address at even 32bit boundaries, |