Commit 81796616 authored by Russell King - ARM Linux's avatar Russell King - ARM Linux Committed by Dan Williams

DMA: PL08x: fix channel pausing to timeout rather than lockup

If a transfer is initiated from memory to a peripheral, then data is
fetched and the channel is marked busy.  This busy status persists until
the HALT bit is set and the queued data has been transfered to the
peripheral.  Waiting indefinitely after setting the HALT bit results in
system lockups.  Timeout this operation, and print an error when this
happens.
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
Acked-by: default avatarLinus Walleij <linus.walleij@stericsson.com>
Signed-off-by: default avatarDan Williams <dan.j.williams@intel.com>
parent fb526210
...@@ -79,6 +79,7 @@ ...@@ -79,6 +79,7 @@
#include <linux/module.h> #include <linux/module.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/delay.h>
#include <linux/dmapool.h> #include <linux/dmapool.h>
#include <linux/dmaengine.h> #include <linux/dmaengine.h>
#include <linux/amba/bus.h> #include <linux/amba/bus.h>
...@@ -235,16 +236,19 @@ static void pl08x_start_txd(struct pl08x_dma_chan *plchan, ...@@ -235,16 +236,19 @@ static void pl08x_start_txd(struct pl08x_dma_chan *plchan,
} }
/* /*
* Overall DMAC remains enabled always. * Pause the channel by setting the HALT bit.
* *
* Disabling individual channels could lose data. * For M->P transfers, pause the DMAC first and then stop the peripheral -
* the FIFO can only drain if the peripheral is still requesting data.
* (note: this can still timeout if the DMAC FIFO never drains of data.)
* *
* Disable the peripheral DMA after disabling the DMAC in order to allow * For P->M transfers, disable the peripheral first to stop it filling
* the DMAC FIFO to drain, and hence allow the channel to show inactive * the DMAC FIFO, and then pause the DMAC.
*/ */
static void pl08x_pause_phy_chan(struct pl08x_phy_chan *ch) static void pl08x_pause_phy_chan(struct pl08x_phy_chan *ch)
{ {
u32 val; u32 val;
int timeout;
/* Set the HALT bit and wait for the FIFO to drain */ /* Set the HALT bit and wait for the FIFO to drain */
val = readl(ch->base + PL080_CH_CONFIG); val = readl(ch->base + PL080_CH_CONFIG);
...@@ -252,8 +256,13 @@ static void pl08x_pause_phy_chan(struct pl08x_phy_chan *ch) ...@@ -252,8 +256,13 @@ static void pl08x_pause_phy_chan(struct pl08x_phy_chan *ch)
writel(val, ch->base + PL080_CH_CONFIG); writel(val, ch->base + PL080_CH_CONFIG);
/* Wait for channel inactive */ /* Wait for channel inactive */
while (pl08x_phy_channel_busy(ch)) for (timeout = 1000; timeout; timeout--) {
cpu_relax(); if (!pl08x_phy_channel_busy(ch))
break;
udelay(1);
}
if (pl08x_phy_channel_busy(ch))
pr_err("pl08x: channel%u timeout waiting for pause\n", ch->id);
} }
static void pl08x_resume_phy_chan(struct pl08x_phy_chan *ch) static void pl08x_resume_phy_chan(struct pl08x_phy_chan *ch)
......
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