All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH] DMAEngine: sirf: add DMA pause/resume support
@ 2012-12-14  9:44 Barry Song
  2012-12-14  9:54 ` Russell King - ARM Linux
  0 siblings, 1 reply; 3+ messages in thread
From: Barry Song @ 2012-12-14  9:44 UTC (permalink / raw)
  To: linux-arm-kernel

From: Barry Song <Baohua.Song@csr.com>

pause/resume are important for users like sound drivers,
this patches make the sirf prima2/marco support

Signed-off-by: Barry Song <Baohua.Song@csr.com>
---
 drivers/dma/sirf-dma.c |   36 ++++++++++++++++++++++++++++++++++++
 1 files changed, 36 insertions(+), 0 deletions(-)

diff --git a/drivers/dma/sirf-dma.c b/drivers/dma/sirf-dma.c
index 212f69f..ffe05bb 100644
--- a/drivers/dma/sirf-dma.c
+++ b/drivers/dma/sirf-dma.c
@@ -313,6 +313,38 @@ static int sirfsoc_dma_terminate_all(struct sirfsoc_dma_chan *schan)
 	return 0;
 }
 
+static int sirfsoc_dma_pause_chan(struct sirfsoc_dma_chan *schan)
+{
+	struct sirfsoc_dma *sdma = dma_chan_to_sirfsoc_dma(&schan->chan);
+	int cid = schan->chan.chan_id;
+
+	if (!sdma->is_marco)
+		writel_relaxed(readl_relaxed(sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL)
+			& ~((1 << cid) | 1 << (cid + 16)),
+			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL);
+	else
+		writel_relaxed((1 << cid) | 1 << (cid + 16),
+			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL_CLR);
+
+	return 0;
+}
+
+static int sirfsoc_dma_resume_chan(struct sirfsoc_dma_chan *schan)
+{
+	struct sirfsoc_dma *sdma = dma_chan_to_sirfsoc_dma(&schan->chan);
+	int cid = schan->chan.chan_id;
+
+	if (!sdma->is_marco)
+		writel_relaxed(readl_relaxed(sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL)
+			| ((1 << cid) | 1 << (cid + 16)),
+			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL);
+	else
+		writel_relaxed((1 << cid) | 1 << (cid + 16),
+			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL);
+
+	return 0;
+}
+
 static int sirfsoc_dma_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd,
 	unsigned long arg)
 {
@@ -320,6 +352,10 @@ static int sirfsoc_dma_control(struct dma_chan *chan, enum dma_ctrl_cmd cmd,
 	struct sirfsoc_dma_chan *schan = dma_chan_to_sirfsoc_dma_chan(chan);
 
 	switch (cmd) {
+	case DMA_PAUSE:
+		return sirfsoc_dma_pause_chan(schan);
+	case DMA_RESUME:
+		return sirfsoc_dma_resume_chan(schan);
 	case DMA_TERMINATE_ALL:
 		return sirfsoc_dma_terminate_all(schan);
 	case DMA_SLAVE_CONFIG:
-- 
1.7.5.4



Member of the CSR plc group of companies. CSR plc registered in England and Wales, registered number 4187346, registered office Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, United Kingdom
More information can be found at www.csr.com. Follow CSR on Twitter at http://twitter.com/CSR_PLC and read our blog at www.csr.com/blog

^ permalink raw reply related	[flat|nested] 3+ messages in thread

* [PATCH] DMAEngine: sirf: add DMA pause/resume support
  2012-12-14  9:44 [PATCH] DMAEngine: sirf: add DMA pause/resume support Barry Song
@ 2012-12-14  9:54 ` Russell King - ARM Linux
  2012-12-14 10:47   ` Barry Song
  0 siblings, 1 reply; 3+ messages in thread
From: Russell King - ARM Linux @ 2012-12-14  9:54 UTC (permalink / raw)
  To: linux-arm-kernel

On Fri, Dec 14, 2012 at 05:44:42PM +0800, Barry Song wrote:
> diff --git a/drivers/dma/sirf-dma.c b/drivers/dma/sirf-dma.c
> index 212f69f..ffe05bb 100644
> --- a/drivers/dma/sirf-dma.c
> +++ b/drivers/dma/sirf-dma.c
> @@ -313,6 +313,38 @@ static int sirfsoc_dma_terminate_all(struct sirfsoc_dma_chan *schan)
>  	return 0;
>  }
>  
> +static int sirfsoc_dma_pause_chan(struct sirfsoc_dma_chan *schan)
> +{
> +	struct sirfsoc_dma *sdma = dma_chan_to_sirfsoc_dma(&schan->chan);
> +	int cid = schan->chan.chan_id;
> +
> +	if (!sdma->is_marco)
> +		writel_relaxed(readl_relaxed(sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL)
> +			& ~((1 << cid) | 1 << (cid + 16)),
> +			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL);
> +	else
> +		writel_relaxed((1 << cid) | 1 << (cid + 16),
> +			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL_CLR);

You're doing a read-modify-write to what looks like a shared register.
What makes this code safe in a preempt or SMP environment?

^ permalink raw reply	[flat|nested] 3+ messages in thread

* [PATCH] DMAEngine: sirf: add DMA pause/resume support
  2012-12-14  9:54 ` Russell King - ARM Linux
@ 2012-12-14 10:47   ` Barry Song
  0 siblings, 0 replies; 3+ messages in thread
From: Barry Song @ 2012-12-14 10:47 UTC (permalink / raw)
  To: linux-arm-kernel

2012/12/14, Russell King - ARM Linux <linux@arm.linux.org.uk>:
> On Fri, Dec 14, 2012 at 05:44:42PM +0800, Barry Song wrote:
>> diff --git a/drivers/dma/sirf-dma.c b/drivers/dma/sirf-dma.c
>> index 212f69f..ffe05bb 100644
>> --- a/drivers/dma/sirf-dma.c
>> +++ b/drivers/dma/sirf-dma.c
>> @@ -313,6 +313,38 @@ static int sirfsoc_dma_terminate_all(struct
>> sirfsoc_dma_chan *schan)
>>  	return 0;
>>  }
>>
>> +static int sirfsoc_dma_pause_chan(struct sirfsoc_dma_chan *schan)
>> +{
>> +	struct sirfsoc_dma *sdma = dma_chan_to_sirfsoc_dma(&schan->chan);
>> +	int cid = schan->chan.chan_id;
>> +
>> +	if (!sdma->is_marco)
>> +		writel_relaxed(readl_relaxed(sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL)
>> +			& ~((1 << cid) | 1 << (cid + 16)),
>> +			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL);
>> +	else
>> +		writel_relaxed((1 << cid) | 1 << (cid + 16),
>> +			sdma->base + SIRFSOC_DMA_CH_LOOP_CTRL_CLR);
>
> You're doing a read-modify-write to what looks like a shared register.
> What makes this code safe in a preempt or SMP environment?

the new registers of SiRFmarco split into SET and CLR instead of old
read-modify-write. anyway, here i missed holding a spinlock for
SiRFprimaII.

-barry

^ permalink raw reply	[flat|nested] 3+ messages in thread

end of thread, other threads:[~2012-12-14 10:47 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2012-12-14  9:44 [PATCH] DMAEngine: sirf: add DMA pause/resume support Barry Song
2012-12-14  9:54 ` Russell King - ARM Linux
2012-12-14 10:47   ` Barry Song

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.