aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/dma/at_hdmac.c
diff options
context:
space:
mode:
authorNicolas Ferre <nicolas.ferre@atmel.com>2011-07-27 08:21:29 -0400
committerVinod Koul <vinod.koul@intel.com>2011-08-19 09:09:54 -0400
commitc0ba5947370a0900b1823922fc4faf41515bc901 (patch)
treef3c32b76f48fc7f78ebd32a6d33f5e6b99e1328a /drivers/dma/at_hdmac.c
parentd8cb04b070c2a55f7201714d231cff4f8f9fbd16 (diff)
dmaengine: at_hdmac: improve power management routines
Save/restore dma controller state across a suspend-resume sequence. The prepare() function will wait for the non-cyclic channels to become idle. It also deals with cyclic operations with the start at next period while resuming. Signed-off-by: Nicolas Ferre <nicolas.ferre@atmel.com> Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de> Signed-off-by: Vinod Koul <vinod.koul@intel.com>
Diffstat (limited to 'drivers/dma/at_hdmac.c')
-rw-r--r--drivers/dma/at_hdmac.c88
1 files changed, 87 insertions, 1 deletions
diff --git a/drivers/dma/at_hdmac.c b/drivers/dma/at_hdmac.c
index fd87b9690e1b..0ead008e3bdf 100644
--- a/drivers/dma/at_hdmac.c
+++ b/drivers/dma/at_hdmac.c
@@ -1385,27 +1385,113 @@ static void at_dma_shutdown(struct platform_device *pdev)
1385 clk_disable(atdma->clk); 1385 clk_disable(atdma->clk);
1386} 1386}
1387 1387
1388static int at_dma_prepare(struct device *dev)
1389{
1390 struct platform_device *pdev = to_platform_device(dev);
1391 struct at_dma *atdma = platform_get_drvdata(pdev);
1392 struct dma_chan *chan, *_chan;
1393
1394 list_for_each_entry_safe(chan, _chan, &atdma->dma_common.channels,
1395 device_node) {
1396 struct at_dma_chan *atchan = to_at_dma_chan(chan);
1397 /* wait for transaction completion (except in cyclic case) */
1398 if (atc_chan_is_enabled(atchan) &&
1399 !test_bit(ATC_IS_CYCLIC, &atchan->status))
1400 return -EAGAIN;
1401 }
1402 return 0;
1403}
1404
1405static void atc_suspend_cyclic(struct at_dma_chan *atchan)
1406{
1407 struct dma_chan *chan = &atchan->chan_common;
1408
1409 /* Channel should be paused by user
1410 * do it anyway even if it is not done already */
1411 if (!test_bit(ATC_IS_PAUSED, &atchan->status)) {
1412 dev_warn(chan2dev(chan),
1413 "cyclic channel not paused, should be done by channel user\n");
1414 atc_control(chan, DMA_PAUSE, 0);
1415 }
1416
1417 /* now preserve additional data for cyclic operations */
1418 /* next descriptor address in the cyclic list */
1419 atchan->save_dscr = channel_readl(atchan, DSCR);
1420
1421 vdbg_dump_regs(atchan);
1422}
1423
1388static int at_dma_suspend_noirq(struct device *dev) 1424static int at_dma_suspend_noirq(struct device *dev)
1389{ 1425{
1390 struct platform_device *pdev = to_platform_device(dev); 1426 struct platform_device *pdev = to_platform_device(dev);
1391 struct at_dma *atdma = platform_get_drvdata(pdev); 1427 struct at_dma *atdma = platform_get_drvdata(pdev);
1428 struct dma_chan *chan, *_chan;
1392 1429
1393 at_dma_off(platform_get_drvdata(pdev)); 1430 /* preserve data */
1431 list_for_each_entry_safe(chan, _chan, &atdma->dma_common.channels,
1432 device_node) {
1433 struct at_dma_chan *atchan = to_at_dma_chan(chan);
1434
1435 if (test_bit(ATC_IS_CYCLIC, &atchan->status))
1436 atc_suspend_cyclic(atchan);
1437 atchan->save_cfg = channel_readl(atchan, CFG);
1438 }
1439 atdma->save_imr = dma_readl(atdma, EBCIMR);
1440
1441 /* disable DMA controller */
1442 at_dma_off(atdma);
1394 clk_disable(atdma->clk); 1443 clk_disable(atdma->clk);
1395 return 0; 1444 return 0;
1396} 1445}
1397 1446
1447static void atc_resume_cyclic(struct at_dma_chan *atchan)
1448{
1449 struct at_dma *atdma = to_at_dma(atchan->chan_common.device);
1450
1451 /* restore channel status for cyclic descriptors list:
1452 * next descriptor in the cyclic list at the time of suspend */
1453 channel_writel(atchan, SADDR, 0);
1454 channel_writel(atchan, DADDR, 0);
1455 channel_writel(atchan, CTRLA, 0);
1456 channel_writel(atchan, CTRLB, 0);
1457 channel_writel(atchan, DSCR, atchan->save_dscr);
1458 dma_writel(atdma, CHER, atchan->mask);
1459
1460 /* channel pause status should be removed by channel user
1461 * We cannot take the initiative to do it here */
1462
1463 vdbg_dump_regs(atchan);
1464}
1465
1398static int at_dma_resume_noirq(struct device *dev) 1466static int at_dma_resume_noirq(struct device *dev)
1399{ 1467{
1400 struct platform_device *pdev = to_platform_device(dev); 1468 struct platform_device *pdev = to_platform_device(dev);
1401 struct at_dma *atdma = platform_get_drvdata(pdev); 1469 struct at_dma *atdma = platform_get_drvdata(pdev);
1470 struct dma_chan *chan, *_chan;
1402 1471
1472 /* bring back DMA controller */
1403 clk_enable(atdma->clk); 1473 clk_enable(atdma->clk);
1404 dma_writel(atdma, EN, AT_DMA_ENABLE); 1474 dma_writel(atdma, EN, AT_DMA_ENABLE);
1475
1476 /* clear any pending interrupt */
1477 while (dma_readl(atdma, EBCISR))
1478 cpu_relax();
1479
1480 /* restore saved data */
1481 dma_writel(atdma, EBCIER, atdma->save_imr);
1482 list_for_each_entry_safe(chan, _chan, &atdma->dma_common.channels,
1483 device_node) {
1484 struct at_dma_chan *atchan = to_at_dma_chan(chan);
1485
1486 channel_writel(atchan, CFG, atchan->save_cfg);
1487 if (test_bit(ATC_IS_CYCLIC, &atchan->status))
1488 atc_resume_cyclic(atchan);
1489 }
1405 return 0; 1490 return 0;
1406} 1491}
1407 1492
1408static const struct dev_pm_ops at_dma_dev_pm_ops = { 1493static const struct dev_pm_ops at_dma_dev_pm_ops = {
1494 .prepare = at_dma_prepare,
1409 .suspend_noirq = at_dma_suspend_noirq, 1495 .suspend_noirq = at_dma_suspend_noirq,
1410 .resume_noirq = at_dma_resume_noirq, 1496 .resume_noirq = at_dma_resume_noirq,
1411}; 1497};