diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2016-01-20 13:15:21 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2016-01-20 13:15:21 -0500 |
commit | 9fa686068a32ddf256df03982b3e3967c18654a8 (patch) | |
tree | 888adef0b50c93df4c40c4b5555a6c17460320c1 | |
parent | 90bf353bc1451b71067b12541db08ab7774723eb (diff) | |
parent | 6bf641039f96e0c0cf3b10ae9ba12039d1a01122 (diff) |
Merge tag 'dmaengine-fix-4.5-rc1' of git://git.infradead.org/users/vkoul/slave-dma
Pull dmaengine fixes from Vinod Koul:
"Here is my second pull request for this window:
A few driver fixes have piled up and one missed rcar bindings patch
which got somehow lost in for-linus branch so cherry-picked that one.
Fixes are for dw, at_hdmac, edma"
* tag 'dmaengine-fix-4.5-rc1' of git://git.infradead.org/users/vkoul/slave-dma:
dmaengine: rcar-dmac: Document SoC specific bindings
dmaengine: at_xdmac: fix resume for cyclic transfers
dmaengine: dw: fix cyclic transfer callbacks
dmaengine: dw: fix cyclic transfer setup
dmaengine: edma: Fix paRAM slot allocation for entry channel 0
-rw-r--r-- | Documentation/devicetree/bindings/dma/renesas,rcar-dmac.txt | 13 | ||||
-rw-r--r-- | drivers/dma/at_xdmac.c | 3 | ||||
-rw-r--r-- | drivers/dma/dw/core.c | 44 | ||||
-rw-r--r-- | drivers/dma/edma.c | 2 |
4 files changed, 30 insertions, 32 deletions
diff --git a/Documentation/devicetree/bindings/dma/renesas,rcar-dmac.txt b/Documentation/devicetree/bindings/dma/renesas,rcar-dmac.txt index 09daeef1ff22..5b902ac8d97e 100644 --- a/Documentation/devicetree/bindings/dma/renesas,rcar-dmac.txt +++ b/Documentation/devicetree/bindings/dma/renesas,rcar-dmac.txt | |||
@@ -14,7 +14,14 @@ not described in these device tree bindings. | |||
14 | 14 | ||
15 | Required Properties: | 15 | Required Properties: |
16 | 16 | ||
17 | - compatible: must contain "renesas,rcar-dmac" | 17 | - compatible: "renesas,dmac-<soctype>", "renesas,rcar-dmac" as fallback. |
18 | Examples with soctypes are: | ||
19 | - "renesas,dmac-r8a7790" (R-Car H2) | ||
20 | - "renesas,dmac-r8a7791" (R-Car M2-W) | ||
21 | - "renesas,dmac-r8a7792" (R-Car V2H) | ||
22 | - "renesas,dmac-r8a7793" (R-Car M2-N) | ||
23 | - "renesas,dmac-r8a7794" (R-Car E2) | ||
24 | - "renesas,dmac-r8a7795" (R-Car H3) | ||
18 | 25 | ||
19 | - reg: base address and length of the registers block for the DMAC | 26 | - reg: base address and length of the registers block for the DMAC |
20 | 27 | ||
@@ -35,7 +42,7 @@ Required Properties: | |||
35 | Example: R8A7790 (R-Car H2) SYS-DMACs | 42 | Example: R8A7790 (R-Car H2) SYS-DMACs |
36 | 43 | ||
37 | dmac0: dma-controller@e6700000 { | 44 | dmac0: dma-controller@e6700000 { |
38 | compatible = "renesas,rcar-dmac"; | 45 | compatible = "renesas,dmac-r8a7790", "renesas,rcar-dmac"; |
39 | reg = <0 0xe6700000 0 0x20000>; | 46 | reg = <0 0xe6700000 0 0x20000>; |
40 | interrupts = <0 197 IRQ_TYPE_LEVEL_HIGH | 47 | interrupts = <0 197 IRQ_TYPE_LEVEL_HIGH |
41 | 0 200 IRQ_TYPE_LEVEL_HIGH | 48 | 0 200 IRQ_TYPE_LEVEL_HIGH |
@@ -65,7 +72,7 @@ Example: R8A7790 (R-Car H2) SYS-DMACs | |||
65 | }; | 72 | }; |
66 | 73 | ||
67 | dmac1: dma-controller@e6720000 { | 74 | dmac1: dma-controller@e6720000 { |
68 | compatible = "renesas,rcar-dmac"; | 75 | compatible = "renesas,dmac-r8a7790", "renesas,rcar-dmac"; |
69 | reg = <0 0xe6720000 0 0x20000>; | 76 | reg = <0 0xe6720000 0 0x20000>; |
70 | interrupts = <0 220 IRQ_TYPE_LEVEL_HIGH | 77 | interrupts = <0 220 IRQ_TYPE_LEVEL_HIGH |
71 | 0 216 IRQ_TYPE_LEVEL_HIGH | 78 | 0 216 IRQ_TYPE_LEVEL_HIGH |
diff --git a/drivers/dma/at_xdmac.c b/drivers/dma/at_xdmac.c index 39f59666f93f..64f5d1bdbb48 100644 --- a/drivers/dma/at_xdmac.c +++ b/drivers/dma/at_xdmac.c | |||
@@ -1700,6 +1700,7 @@ static int at_xdmac_device_terminate_all(struct dma_chan *chan) | |||
1700 | list_for_each_entry_safe(desc, _desc, &atchan->xfers_list, xfer_node) | 1700 | list_for_each_entry_safe(desc, _desc, &atchan->xfers_list, xfer_node) |
1701 | at_xdmac_remove_xfer(atchan, desc); | 1701 | at_xdmac_remove_xfer(atchan, desc); |
1702 | 1702 | ||
1703 | clear_bit(AT_XDMAC_CHAN_IS_PAUSED, &atchan->status); | ||
1703 | clear_bit(AT_XDMAC_CHAN_IS_CYCLIC, &atchan->status); | 1704 | clear_bit(AT_XDMAC_CHAN_IS_CYCLIC, &atchan->status); |
1704 | spin_unlock_irqrestore(&atchan->lock, flags); | 1705 | spin_unlock_irqrestore(&atchan->lock, flags); |
1705 | 1706 | ||
@@ -1832,6 +1833,8 @@ static int atmel_xdmac_resume(struct device *dev) | |||
1832 | atchan = to_at_xdmac_chan(chan); | 1833 | atchan = to_at_xdmac_chan(chan); |
1833 | at_xdmac_chan_write(atchan, AT_XDMAC_CC, atchan->save_cc); | 1834 | at_xdmac_chan_write(atchan, AT_XDMAC_CC, atchan->save_cc); |
1834 | if (at_xdmac_chan_is_cyclic(atchan)) { | 1835 | if (at_xdmac_chan_is_cyclic(atchan)) { |
1836 | if (at_xdmac_chan_is_paused(atchan)) | ||
1837 | at_xdmac_device_resume(chan); | ||
1835 | at_xdmac_chan_write(atchan, AT_XDMAC_CNDA, atchan->save_cnda); | 1838 | at_xdmac_chan_write(atchan, AT_XDMAC_CNDA, atchan->save_cnda); |
1836 | at_xdmac_chan_write(atchan, AT_XDMAC_CNDC, atchan->save_cndc); | 1839 | at_xdmac_chan_write(atchan, AT_XDMAC_CNDC, atchan->save_cndc); |
1837 | at_xdmac_chan_write(atchan, AT_XDMAC_CIE, atchan->save_cim); | 1840 | at_xdmac_chan_write(atchan, AT_XDMAC_CIE, atchan->save_cim); |
diff --git a/drivers/dma/dw/core.c b/drivers/dma/dw/core.c index 8b20930ade98..e893318560db 100644 --- a/drivers/dma/dw/core.c +++ b/drivers/dma/dw/core.c | |||
@@ -156,6 +156,7 @@ static void dwc_initialize(struct dw_dma_chan *dwc) | |||
156 | 156 | ||
157 | /* Enable interrupts */ | 157 | /* Enable interrupts */ |
158 | channel_set_bit(dw, MASK.XFER, dwc->mask); | 158 | channel_set_bit(dw, MASK.XFER, dwc->mask); |
159 | channel_set_bit(dw, MASK.BLOCK, dwc->mask); | ||
159 | channel_set_bit(dw, MASK.ERROR, dwc->mask); | 160 | channel_set_bit(dw, MASK.ERROR, dwc->mask); |
160 | 161 | ||
161 | dwc->initialized = true; | 162 | dwc->initialized = true; |
@@ -536,16 +537,17 @@ EXPORT_SYMBOL(dw_dma_get_dst_addr); | |||
536 | 537 | ||
537 | /* Called with dwc->lock held and all DMAC interrupts disabled */ | 538 | /* Called with dwc->lock held and all DMAC interrupts disabled */ |
538 | static void dwc_handle_cyclic(struct dw_dma *dw, struct dw_dma_chan *dwc, | 539 | static void dwc_handle_cyclic(struct dw_dma *dw, struct dw_dma_chan *dwc, |
539 | u32 status_err, u32 status_xfer) | 540 | u32 status_block, u32 status_err, u32 status_xfer) |
540 | { | 541 | { |
541 | unsigned long flags; | 542 | unsigned long flags; |
542 | 543 | ||
543 | if (dwc->mask) { | 544 | if (status_block & dwc->mask) { |
544 | void (*callback)(void *param); | 545 | void (*callback)(void *param); |
545 | void *callback_param; | 546 | void *callback_param; |
546 | 547 | ||
547 | dev_vdbg(chan2dev(&dwc->chan), "new cyclic period llp 0x%08x\n", | 548 | dev_vdbg(chan2dev(&dwc->chan), "new cyclic period llp 0x%08x\n", |
548 | channel_readl(dwc, LLP)); | 549 | channel_readl(dwc, LLP)); |
550 | dma_writel(dw, CLEAR.BLOCK, dwc->mask); | ||
549 | 551 | ||
550 | callback = dwc->cdesc->period_callback; | 552 | callback = dwc->cdesc->period_callback; |
551 | callback_param = dwc->cdesc->period_callback_param; | 553 | callback_param = dwc->cdesc->period_callback_param; |
@@ -577,6 +579,7 @@ static void dwc_handle_cyclic(struct dw_dma *dw, struct dw_dma_chan *dwc, | |||
577 | channel_writel(dwc, CTL_LO, 0); | 579 | channel_writel(dwc, CTL_LO, 0); |
578 | channel_writel(dwc, CTL_HI, 0); | 580 | channel_writel(dwc, CTL_HI, 0); |
579 | 581 | ||
582 | dma_writel(dw, CLEAR.BLOCK, dwc->mask); | ||
580 | dma_writel(dw, CLEAR.ERROR, dwc->mask); | 583 | dma_writel(dw, CLEAR.ERROR, dwc->mask); |
581 | dma_writel(dw, CLEAR.XFER, dwc->mask); | 584 | dma_writel(dw, CLEAR.XFER, dwc->mask); |
582 | 585 | ||
@@ -593,10 +596,12 @@ static void dw_dma_tasklet(unsigned long data) | |||
593 | { | 596 | { |
594 | struct dw_dma *dw = (struct dw_dma *)data; | 597 | struct dw_dma *dw = (struct dw_dma *)data; |
595 | struct dw_dma_chan *dwc; | 598 | struct dw_dma_chan *dwc; |
599 | u32 status_block; | ||
596 | u32 status_xfer; | 600 | u32 status_xfer; |
597 | u32 status_err; | 601 | u32 status_err; |
598 | int i; | 602 | int i; |
599 | 603 | ||
604 | status_block = dma_readl(dw, RAW.BLOCK); | ||
600 | status_xfer = dma_readl(dw, RAW.XFER); | 605 | status_xfer = dma_readl(dw, RAW.XFER); |
601 | status_err = dma_readl(dw, RAW.ERROR); | 606 | status_err = dma_readl(dw, RAW.ERROR); |
602 | 607 | ||
@@ -605,7 +610,8 @@ static void dw_dma_tasklet(unsigned long data) | |||
605 | for (i = 0; i < dw->dma.chancnt; i++) { | 610 | for (i = 0; i < dw->dma.chancnt; i++) { |
606 | dwc = &dw->chan[i]; | 611 | dwc = &dw->chan[i]; |
607 | if (test_bit(DW_DMA_IS_CYCLIC, &dwc->flags)) | 612 | if (test_bit(DW_DMA_IS_CYCLIC, &dwc->flags)) |
608 | dwc_handle_cyclic(dw, dwc, status_err, status_xfer); | 613 | dwc_handle_cyclic(dw, dwc, status_block, status_err, |
614 | status_xfer); | ||
609 | else if (status_err & (1 << i)) | 615 | else if (status_err & (1 << i)) |
610 | dwc_handle_error(dw, dwc); | 616 | dwc_handle_error(dw, dwc); |
611 | else if (status_xfer & (1 << i)) | 617 | else if (status_xfer & (1 << i)) |
@@ -616,6 +622,7 @@ static void dw_dma_tasklet(unsigned long data) | |||
616 | * Re-enable interrupts. | 622 | * Re-enable interrupts. |
617 | */ | 623 | */ |
618 | channel_set_bit(dw, MASK.XFER, dw->all_chan_mask); | 624 | channel_set_bit(dw, MASK.XFER, dw->all_chan_mask); |
625 | channel_set_bit(dw, MASK.BLOCK, dw->all_chan_mask); | ||
619 | channel_set_bit(dw, MASK.ERROR, dw->all_chan_mask); | 626 | channel_set_bit(dw, MASK.ERROR, dw->all_chan_mask); |
620 | } | 627 | } |
621 | 628 | ||
@@ -640,6 +647,7 @@ static irqreturn_t dw_dma_interrupt(int irq, void *dev_id) | |||
640 | * softirq handler. | 647 | * softirq handler. |
641 | */ | 648 | */ |
642 | channel_clear_bit(dw, MASK.XFER, dw->all_chan_mask); | 649 | channel_clear_bit(dw, MASK.XFER, dw->all_chan_mask); |
650 | channel_clear_bit(dw, MASK.BLOCK, dw->all_chan_mask); | ||
643 | channel_clear_bit(dw, MASK.ERROR, dw->all_chan_mask); | 651 | channel_clear_bit(dw, MASK.ERROR, dw->all_chan_mask); |
644 | 652 | ||
645 | status = dma_readl(dw, STATUS_INT); | 653 | status = dma_readl(dw, STATUS_INT); |
@@ -650,6 +658,7 @@ static irqreturn_t dw_dma_interrupt(int irq, void *dev_id) | |||
650 | 658 | ||
651 | /* Try to recover */ | 659 | /* Try to recover */ |
652 | channel_clear_bit(dw, MASK.XFER, (1 << 8) - 1); | 660 | channel_clear_bit(dw, MASK.XFER, (1 << 8) - 1); |
661 | channel_clear_bit(dw, MASK.BLOCK, (1 << 8) - 1); | ||
653 | channel_clear_bit(dw, MASK.SRC_TRAN, (1 << 8) - 1); | 662 | channel_clear_bit(dw, MASK.SRC_TRAN, (1 << 8) - 1); |
654 | channel_clear_bit(dw, MASK.DST_TRAN, (1 << 8) - 1); | 663 | channel_clear_bit(dw, MASK.DST_TRAN, (1 << 8) - 1); |
655 | channel_clear_bit(dw, MASK.ERROR, (1 << 8) - 1); | 664 | channel_clear_bit(dw, MASK.ERROR, (1 << 8) - 1); |
@@ -1116,6 +1125,7 @@ static void dw_dma_off(struct dw_dma *dw) | |||
1116 | dma_writel(dw, CFG, 0); | 1125 | dma_writel(dw, CFG, 0); |
1117 | 1126 | ||
1118 | channel_clear_bit(dw, MASK.XFER, dw->all_chan_mask); | 1127 | channel_clear_bit(dw, MASK.XFER, dw->all_chan_mask); |
1128 | channel_clear_bit(dw, MASK.BLOCK, dw->all_chan_mask); | ||
1119 | channel_clear_bit(dw, MASK.SRC_TRAN, dw->all_chan_mask); | 1129 | channel_clear_bit(dw, MASK.SRC_TRAN, dw->all_chan_mask); |
1120 | channel_clear_bit(dw, MASK.DST_TRAN, dw->all_chan_mask); | 1130 | channel_clear_bit(dw, MASK.DST_TRAN, dw->all_chan_mask); |
1121 | channel_clear_bit(dw, MASK.ERROR, dw->all_chan_mask); | 1131 | channel_clear_bit(dw, MASK.ERROR, dw->all_chan_mask); |
@@ -1221,6 +1231,7 @@ static void dwc_free_chan_resources(struct dma_chan *chan) | |||
1221 | 1231 | ||
1222 | /* Disable interrupts */ | 1232 | /* Disable interrupts */ |
1223 | channel_clear_bit(dw, MASK.XFER, dwc->mask); | 1233 | channel_clear_bit(dw, MASK.XFER, dwc->mask); |
1234 | channel_clear_bit(dw, MASK.BLOCK, dwc->mask); | ||
1224 | channel_clear_bit(dw, MASK.ERROR, dwc->mask); | 1235 | channel_clear_bit(dw, MASK.ERROR, dwc->mask); |
1225 | 1236 | ||
1226 | spin_unlock_irqrestore(&dwc->lock, flags); | 1237 | spin_unlock_irqrestore(&dwc->lock, flags); |
@@ -1250,7 +1261,6 @@ static void dwc_free_chan_resources(struct dma_chan *chan) | |||
1250 | int dw_dma_cyclic_start(struct dma_chan *chan) | 1261 | int dw_dma_cyclic_start(struct dma_chan *chan) |
1251 | { | 1262 | { |
1252 | struct dw_dma_chan *dwc = to_dw_dma_chan(chan); | 1263 | struct dw_dma_chan *dwc = to_dw_dma_chan(chan); |
1253 | struct dw_dma *dw = to_dw_dma(dwc->chan.device); | ||
1254 | unsigned long flags; | 1264 | unsigned long flags; |
1255 | 1265 | ||
1256 | if (!test_bit(DW_DMA_IS_CYCLIC, &dwc->flags)) { | 1266 | if (!test_bit(DW_DMA_IS_CYCLIC, &dwc->flags)) { |
@@ -1259,27 +1269,7 @@ int dw_dma_cyclic_start(struct dma_chan *chan) | |||
1259 | } | 1269 | } |
1260 | 1270 | ||
1261 | spin_lock_irqsave(&dwc->lock, flags); | 1271 | spin_lock_irqsave(&dwc->lock, flags); |
1262 | 1272 | dwc_dostart(dwc, dwc->cdesc->desc[0]); | |
1263 | /* Assert channel is idle */ | ||
1264 | if (dma_readl(dw, CH_EN) & dwc->mask) { | ||
1265 | dev_err(chan2dev(&dwc->chan), | ||
1266 | "%s: BUG: Attempted to start non-idle channel\n", | ||
1267 | __func__); | ||
1268 | dwc_dump_chan_regs(dwc); | ||
1269 | spin_unlock_irqrestore(&dwc->lock, flags); | ||
1270 | return -EBUSY; | ||
1271 | } | ||
1272 | |||
1273 | dma_writel(dw, CLEAR.ERROR, dwc->mask); | ||
1274 | dma_writel(dw, CLEAR.XFER, dwc->mask); | ||
1275 | |||
1276 | /* Setup DMAC channel registers */ | ||
1277 | channel_writel(dwc, LLP, dwc->cdesc->desc[0]->txd.phys); | ||
1278 | channel_writel(dwc, CTL_LO, DWC_CTLL_LLP_D_EN | DWC_CTLL_LLP_S_EN); | ||
1279 | channel_writel(dwc, CTL_HI, 0); | ||
1280 | |||
1281 | channel_set_bit(dw, CH_EN, dwc->mask); | ||
1282 | |||
1283 | spin_unlock_irqrestore(&dwc->lock, flags); | 1273 | spin_unlock_irqrestore(&dwc->lock, flags); |
1284 | 1274 | ||
1285 | return 0; | 1275 | return 0; |
@@ -1484,6 +1474,7 @@ void dw_dma_cyclic_free(struct dma_chan *chan) | |||
1484 | 1474 | ||
1485 | dwc_chan_disable(dw, dwc); | 1475 | dwc_chan_disable(dw, dwc); |
1486 | 1476 | ||
1477 | dma_writel(dw, CLEAR.BLOCK, dwc->mask); | ||
1487 | dma_writel(dw, CLEAR.ERROR, dwc->mask); | 1478 | dma_writel(dw, CLEAR.ERROR, dwc->mask); |
1488 | dma_writel(dw, CLEAR.XFER, dwc->mask); | 1479 | dma_writel(dw, CLEAR.XFER, dwc->mask); |
1489 | 1480 | ||
@@ -1572,9 +1563,6 @@ int dw_dma_probe(struct dw_dma_chip *chip, struct dw_dma_platform_data *pdata) | |||
1572 | /* Force dma off, just in case */ | 1563 | /* Force dma off, just in case */ |
1573 | dw_dma_off(dw); | 1564 | dw_dma_off(dw); |
1574 | 1565 | ||
1575 | /* Disable BLOCK interrupts as well */ | ||
1576 | channel_clear_bit(dw, MASK.BLOCK, dw->all_chan_mask); | ||
1577 | |||
1578 | /* Create a pool of consistent memory blocks for hardware descriptors */ | 1566 | /* Create a pool of consistent memory blocks for hardware descriptors */ |
1579 | dw->desc_pool = dmam_pool_create("dw_dmac_desc_pool", chip->dev, | 1567 | dw->desc_pool = dmam_pool_create("dw_dmac_desc_pool", chip->dev, |
1580 | sizeof(struct dw_desc), 4, 0); | 1568 | sizeof(struct dw_desc), 4, 0); |
diff --git a/drivers/dma/edma.c b/drivers/dma/edma.c index 50584015e046..d92d65549406 100644 --- a/drivers/dma/edma.c +++ b/drivers/dma/edma.c | |||
@@ -484,7 +484,7 @@ static void edma_read_slot(struct edma_cc *ecc, unsigned slot, | |||
484 | */ | 484 | */ |
485 | static int edma_alloc_slot(struct edma_cc *ecc, int slot) | 485 | static int edma_alloc_slot(struct edma_cc *ecc, int slot) |
486 | { | 486 | { |
487 | if (slot > 0) { | 487 | if (slot >= 0) { |
488 | slot = EDMA_CHAN_SLOT(slot); | 488 | slot = EDMA_CHAN_SLOT(slot); |
489 | /* Requesting entry paRAM slot for a HW triggered channel. */ | 489 | /* Requesting entry paRAM slot for a HW triggered channel. */ |
490 | if (ecc->chmap_exist && slot < ecc->num_channels) | 490 | if (ecc->chmap_exist && slot < ecc->num_channels) |