aboutsummaryrefslogtreecommitdiffstats
path: root/arch
diff options
context:
space:
mode:
authorJarkko Nikula <jhnikula@gmail.com>2010-10-11 17:18:45 -0400
committerTony Lindgren <tony@atomide.com>2010-10-11 17:18:45 -0400
commit3e57f1626b5febe5cc99aa6870377deef3ae03cc (patch)
tree19a93b368344ebf64fd7d57b01133e9f28dd54b2 /arch
parenta1b04cc197e17c47b043051becb548dc8c60c886 (diff)
omap: dma: Fix buffering disable bit setting for omap24xx
An errata workaround for omap24xx is not setting the buffering disable bit 25 what is the purpose but channel enable bit 7 instead. Background for this fix is the DMA stalling issue with ASoC omap-mcbsp driver. Peter Ujfalusi <peter.ujfalusi@nokia.com> has found an issue in recording that the DMA stall could happen if there were a buffer overrun detected by ALSA and the DMA was stopped and restarted due that. This problem is known to occur on both OMAP2420 and OMAP3. It can recover on OMAP3 after dma free, dma request and reconfiguration cycle. However, on OMAP2420 it seems that only way to recover is a reset. Problem was not visible before the commit c12abc0. That commit changed that the McBSP transmitter/receiver is released from reset only when needed. That is, only enabled McBSP transmitter without transmission was able to prevent this DMA stall problem in receiving side and underlying problem did not show up until now. McBSP transmitter itself seems to no be reason since DMA stall does not recover by enabling the transmission after stall. Debugging showed that there were a DMA write active during DMA stop time and it never completed even when restarting the DMA. Experimenting showed that the DMA buffering disable bit could be used to avoid stalling when using source synchronized transfers. However that could have performance hit and OMAP3 TRM states that buffering disable is not allowed for destination synchronized transfers so subsequent patch will implement a method to complete DMA writes when stopping. This patch is based on assumtion that complete lock-up on OMAP2420 is different but related problem. I don't have access to OMAP2420 errata but I believe this old workaround here is put for a reason but unfortunately a wrong bit was typed and problem showed up only now. Signed-off-by: Jarkko Nikula <jhnikula@gmail.com> Signed-off-by: Peter Ujfalusi <peter.ujfalusi@nokia.com> Acked-by: Manjunath Kondaiah G <manjugk@ti.com> Signed-off-by: Tony Lindgren <tony@atomide.com>
Diffstat (limited to 'arch')
-rw-r--r--arch/arm/plat-omap/dma.c14
-rw-r--r--arch/arm/plat-omap/include/plat/dma.h1
2 files changed, 11 insertions, 4 deletions
diff --git a/arch/arm/plat-omap/dma.c b/arch/arm/plat-omap/dma.c
index ec7eddf9e525..420cef370b33 100644
--- a/arch/arm/plat-omap/dma.c
+++ b/arch/arm/plat-omap/dma.c
@@ -996,11 +996,17 @@ void omap_start_dma(int lch)
996 l = dma_read(CCR(lch)); 996 l = dma_read(CCR(lch));
997 997
998 /* 998 /*
999 * Errata: On ES2.0 BUFFERING disable must be set. 999 * Errata: Inter Frame DMA buffering issue (All OMAP2420 and
1000 * This will always fail on ES1.0 1000 * OMAP2430ES1.0): DMA will wrongly buffer elements if packing and
1001 * bursting is enabled. This might result in data gets stalled in
1002 * FIFO at the end of the block.
1003 * Workaround: DMA channels must have BUFFERING_DISABLED bit set to
1004 * guarantee no data will stay in the DMA FIFO in case inter frame
1005 * buffering occurs.
1001 */ 1006 */
1002 if (cpu_is_omap24xx()) 1007 if (cpu_is_omap2420() ||
1003 l |= OMAP_DMA_CCR_EN; 1008 (cpu_is_omap2430() && (omap_type() == OMAP2430_REV_ES1_0)))
1009 l |= OMAP_DMA_CCR_BUFFERING_DISABLE;
1004 1010
1005 l |= OMAP_DMA_CCR_EN; 1011 l |= OMAP_DMA_CCR_EN;
1006 dma_write(l, CCR(lch)); 1012 dma_write(l, CCR(lch));
diff --git a/arch/arm/plat-omap/include/plat/dma.h b/arch/arm/plat-omap/include/plat/dma.h
index 098f154f5d41..6f70f7cfe91d 100644
--- a/arch/arm/plat-omap/include/plat/dma.h
+++ b/arch/arm/plat-omap/include/plat/dma.h
@@ -337,6 +337,7 @@
337#define OMAP2_DMA_MISALIGNED_ERR_IRQ (1 << 11) 337#define OMAP2_DMA_MISALIGNED_ERR_IRQ (1 << 11)
338 338
339#define OMAP_DMA_CCR_EN (1 << 7) 339#define OMAP_DMA_CCR_EN (1 << 7)
340#define OMAP_DMA_CCR_BUFFERING_DISABLE (1 << 25)
340 341
341#define OMAP_DMA_DATA_TYPE_S8 0x00 342#define OMAP_DMA_DATA_TYPE_S8 0x00
342#define OMAP_DMA_DATA_TYPE_S16 0x01 343#define OMAP_DMA_DATA_TYPE_S16 0x01