diff options
author | Russell King - ARM Linux <linux@arm.linux.org.uk> | 2011-01-07 10:49:20 -0500 |
---|---|---|
committer | Grant Likely <grant.likely@secretlab.ca> | 2011-01-07 15:44:24 -0500 |
commit | 07fe0351702b6f0c9749e80cdbcb758686b0fe9b (patch) | |
tree | 49abac1907bae4be949696e806fbe0626e138d8f | |
parent | 735759389b7143f95ae7badc97511e7e4729e873 (diff) |
spi/omap: Fix DMA API usage in OMAP MCSPI driver
Running the latest kernel on the 4430SDP board with DMA API debugging
enabled results in this:
WARNING: at lib/dma-debug.c:803 check_unmap+0x19c/0x6f0()
NULL NULL: DMA-API: device driver tries to free DMA memory it has not allocated
[device address=0x000000008129901a] [size=260 bytes]
Modules linked in:
Backtrace:
[<c003cbe0>] (dump_backtrace+0x0/0x10c) from [<c0278da8>] (dump_stack+0x18/0x1c)
r7:c1839dc0 r6:c0198578 r5:c0304b17 r4:00000323
[<c0278d90>] (dump_stack+0x0/0x1c) from [<c005b158>] (warn_slowpath_common+0x58/0x70)
[<c005b100>] (warn_slowpath_common+0x0/0x70) from [<c005b214>] (warn_slowpath_fmt+0x38/0x40)
r8:c1839e40 r7:00000000 r6:00000104 r5:00000000 r4:8129901a
[<c005b1dc>] (warn_slowpath_fmt+0x0/0x40) from [<c0198578>] (check_unmap+0x19c/0x6f0)
r3:c03110de r2:c0304e6b
[<c01983dc>] (check_unmap+0x0/0x6f0) from [<c0198cd8>] (debug_dma_unmap_page+0x74/0x80)
[<c0198c64>] (debug_dma_unmap_page+0x0/0x80) from [<c01d5ad8>] (omap2_mcspi_work+0x514/0xbf0)
[<c01d55c4>] (omap2_mcspi_work+0x0/0xbf0) from [<c006dfb0>] (process_one_work+0x294/0x400)
[<c006dd1c>] (process_one_work+0x0/0x400) from [<c006e50c>] (worker_thread+0x220/0x3f8)
[<c006e2ec>] (worker_thread+0x0/0x3f8) from [<c00738d0>] (kthread+0x88/0x90)
[<c0073848>] (kthread+0x0/0x90) from [<c005e924>] (do_exit+0x0/0x5fc)
r7:00000013 r6:c005e924 r5:c0073848 r4:c1829ee0
---[ end trace 1b75b31a2719ed20 ]---
I've no idea why this driver uses NULL for dma_unmap_single instead of
the &spi->dev that is laying around just waiting to be used in that
function - but it's an easy fix.
Also replace this comment with a FIXME comment:
/* Do DMA mapping "early" for better error reporting and
* dcache use. Note that if dma_unmap_single() ever starts
* to do real work on ARM, we'd need to clean up mappings
* for previous transfers on *ALL* exits of this loop...
*/
as the comment is not true - we do work in dma_unmap() functions,
particularly on ARMv6 and above. I've corrected the existing unmap
functions but if any others are required they must be added ASAP.
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Acked-by: Tony Lindgren <tony@atomide.com>
Signed-off-by: Grant Likely <grant.likely@secretlab.ca>
-rw-r--r-- | drivers/spi/omap2_mcspi.c | 11 |
1 files changed, 3 insertions, 8 deletions
diff --git a/drivers/spi/omap2_mcspi.c b/drivers/spi/omap2_mcspi.c index 2a651e61bfbf..df85990bb335 100644 --- a/drivers/spi/omap2_mcspi.c +++ b/drivers/spi/omap2_mcspi.c | |||
@@ -397,7 +397,7 @@ omap2_mcspi_txrx_dma(struct spi_device *spi, struct spi_transfer *xfer) | |||
397 | 397 | ||
398 | if (tx != NULL) { | 398 | if (tx != NULL) { |
399 | wait_for_completion(&mcspi_dma->dma_tx_completion); | 399 | wait_for_completion(&mcspi_dma->dma_tx_completion); |
400 | dma_unmap_single(NULL, xfer->tx_dma, count, DMA_TO_DEVICE); | 400 | dma_unmap_single(&spi->dev, xfer->tx_dma, count, DMA_TO_DEVICE); |
401 | 401 | ||
402 | /* for TX_ONLY mode, be sure all words have shifted out */ | 402 | /* for TX_ONLY mode, be sure all words have shifted out */ |
403 | if (rx == NULL) { | 403 | if (rx == NULL) { |
@@ -412,7 +412,7 @@ omap2_mcspi_txrx_dma(struct spi_device *spi, struct spi_transfer *xfer) | |||
412 | 412 | ||
413 | if (rx != NULL) { | 413 | if (rx != NULL) { |
414 | wait_for_completion(&mcspi_dma->dma_rx_completion); | 414 | wait_for_completion(&mcspi_dma->dma_rx_completion); |
415 | dma_unmap_single(NULL, xfer->rx_dma, count, DMA_FROM_DEVICE); | 415 | dma_unmap_single(&spi->dev, xfer->rx_dma, count, DMA_FROM_DEVICE); |
416 | omap2_mcspi_set_enable(spi, 0); | 416 | omap2_mcspi_set_enable(spi, 0); |
417 | 417 | ||
418 | if (l & OMAP2_MCSPI_CHCONF_TURBO) { | 418 | if (l & OMAP2_MCSPI_CHCONF_TURBO) { |
@@ -1025,11 +1025,6 @@ static int omap2_mcspi_transfer(struct spi_device *spi, struct spi_message *m) | |||
1025 | if (m->is_dma_mapped || len < DMA_MIN_BYTES) | 1025 | if (m->is_dma_mapped || len < DMA_MIN_BYTES) |
1026 | continue; | 1026 | continue; |
1027 | 1027 | ||
1028 | /* Do DMA mapping "early" for better error reporting and | ||
1029 | * dcache use. Note that if dma_unmap_single() ever starts | ||
1030 | * to do real work on ARM, we'd need to clean up mappings | ||
1031 | * for previous transfers on *ALL* exits of this loop... | ||
1032 | */ | ||
1033 | if (tx_buf != NULL) { | 1028 | if (tx_buf != NULL) { |
1034 | t->tx_dma = dma_map_single(&spi->dev, (void *) tx_buf, | 1029 | t->tx_dma = dma_map_single(&spi->dev, (void *) tx_buf, |
1035 | len, DMA_TO_DEVICE); | 1030 | len, DMA_TO_DEVICE); |
@@ -1046,7 +1041,7 @@ static int omap2_mcspi_transfer(struct spi_device *spi, struct spi_message *m) | |||
1046 | dev_dbg(&spi->dev, "dma %cX %d bytes error\n", | 1041 | dev_dbg(&spi->dev, "dma %cX %d bytes error\n", |
1047 | 'R', len); | 1042 | 'R', len); |
1048 | if (tx_buf != NULL) | 1043 | if (tx_buf != NULL) |
1049 | dma_unmap_single(NULL, t->tx_dma, | 1044 | dma_unmap_single(&spi->dev, t->tx_dma, |
1050 | len, DMA_TO_DEVICE); | 1045 | len, DMA_TO_DEVICE); |
1051 | return -EINVAL; | 1046 | return -EINVAL; |
1052 | } | 1047 | } |