Thread (1 message) 1 message, 1 author, 2011-07-12

Re: [ath9k-devel] [PATCH v2 07/46] net/wireless: ath9k: fix DMA API usage

From: Michał Mirosław <hidden>
Date: 2011-07-12 13:03:16
Also in: linux-mips, linux-wireless

On Tue, Jul 12, 2011 at 08:54:32PM +0800, Felix Fietkau wrote:
On 12.07.2011, at 17:55, Michał Mirosław [off-list ref] wrote:
quoted
On Tue, Jul 12, 2011 at 12:36:06PM +0800, Felix Fietkau wrote:
quoted
On 2011-07-11 8:52 AM, Michał Mirosław wrote:
quoted
Also constify buf_addr for ath9k_hw_process_rxdesc_edma() to verify
assumptions --- dma_sync_single_for_device() call can be removed.

Signed-off-by: Michał Mirosław<redacted>
---
drivers/net/wireless/ath/ath9k/ar9003_mac.c |    4 ++--
drivers/net/wireless/ath/ath9k/ar9003_mac.h |    2 +-
drivers/net/wireless/ath/ath9k/recv.c       |   10 +++-------
3 files changed, 6 insertions(+), 10 deletions(-)
diff --git a/drivers/net/wireless/ath/ath9k/recv.c b/drivers/net/wireless/ath/ath9k/recv.c
index 70dc8ec..c5f46d5 100644
--- a/drivers/net/wireless/ath/ath9k/recv.c
+++ b/drivers/net/wireless/ath/ath9k/recv.c
@@ -684,15 +684,11 @@ static bool ath_edma_get_buffers(struct ath_softc *sc,
   BUG_ON(!bf);

   dma_sync_single_for_cpu(sc->dev, bf->bf_buf_addr,
-                common->rx_bufsize, DMA_FROM_DEVICE);
+                common->rx_bufsize, DMA_BIDIRECTIONAL);

   ret = ath9k_hw_process_rxdesc_edma(ah, NULL, skb->data);
-    if (ret == -EINPROGRESS) {
-        /*let device gain the buffer again*/
-        dma_sync_single_for_device(sc->dev, bf->bf_buf_addr,
-                common->rx_bufsize, DMA_FROM_DEVICE);
+    if (ret == -EINPROGRESS)
       return false;
-    }

   __skb_unlink(skb,&rx_edma->rx_fifo);
   if (ret == -EINVAL) {
I have strong doubts about this change. On most MIPS devices,
dma_sync_single_for_cpu is a no-op, whereas
dma_sync_single_for_device flushes the cache range. With this
change, the CPU could cache the DMA status part behind skb->data and
that cache entry would not be flushed inbetween calls to this
functions on the same buffer, likely leading to rx stalls.
You're suggesting a platform implementation bug then. If the platform is not
cache-coherent, it should invalidate relevant CPU cache lines for sync_to_cpu
and unmap cases. Do other devices show such symptoms on MIPS systems?

I'm not familiar with the platform internals, so we should ask MIPS people.
I only mentioned MIPS to describe the potential side effect of this change. From my current understanding of the DMA API, it would be wrong on other platforms as well. I believe the _for_device function needs to be used to transfer ownership of the buffer back to the device, before calling _for_cpu again later for another read.
What you're saying reminds the wording in DMA-API-HOWTO.txt that I find
wrong (or at least misleading) compared to what DMA-API.txt describes.
DMA sync calls do not transfer the ownership of the buffer - they are
cache synchronization points, ownership passing is handled entirely by
the driver.
This is definitely required in this case, because when the return code is -EINPROGRESS, the driver waits for the hardware to complete this buffer, and the next call has to fetch the memory again after the device has updated it.
Correctness of this access should be provided by sync_to_cpu() call.

Best Regards,
Michał Mirosław
--
To unsubscribe from this list: send the line "unsubscribe linux-wireless" in
the body of a message to majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Keyboard shortcuts
hback out one level
jnext message in thread
kprevious message in thread
ldrill in
Escclose help / fold thread tree
?toggle this help