From: "John W. Linville" tulip_stop_rxtx() doesn't wait for DMA to fully stop like the function call name implies. This was submitted through my employer -- I am not the original author of this patch. However, I passed it by Jeff Garizk and he expressed interest in having it upstream. Acked-by: Grant Grundler Acked-by: Charlie Brett Signed-off-by: John W. Linville Signed-off-by: Andrew Morton --- 25-akpm/drivers/net/tulip/21142.c | 2 +- 25-akpm/drivers/net/tulip/eeprom.c | 1 + 25-akpm/drivers/net/tulip/interrupt.c | 2 +- 25-akpm/drivers/net/tulip/media.c | 1 + 25-akpm/drivers/net/tulip/pnic.c | 1 + 25-akpm/drivers/net/tulip/pnic2.c | 2 +- 25-akpm/drivers/net/tulip/timer.c | 1 + 25-akpm/drivers/net/tulip/tulip.h | 15 ++++++++++++++- 25-akpm/drivers/net/tulip/tulip_core.c | 2 +- 9 files changed, 22 insertions(+), 5 deletions(-) diff -puN drivers/net/tulip/21142.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/21142.c --- 25/drivers/net/tulip/21142.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/21142.c Wed Nov 17 14:13:00 2004 @@ -14,9 +14,9 @@ */ -#include "tulip.h" #include #include +#include "tulip.h" static u16 t21142_csr13[] = { 0x0001, 0x0009, 0x0009, 0x0000, 0x0001, }; diff -puN drivers/net/tulip/eeprom.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/eeprom.c --- 25/drivers/net/tulip/eeprom.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/eeprom.c Wed Nov 17 14:13:00 2004 @@ -14,6 +14,7 @@ */ +#include #include "tulip.h" #include #include diff -puN drivers/net/tulip/interrupt.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/interrupt.c --- 25/drivers/net/tulip/interrupt.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/interrupt.c Wed Nov 17 14:13:00 2004 @@ -14,10 +14,10 @@ */ +#include #include "tulip.h" #include #include -#include int tulip_rx_copybreak; unsigned int tulip_max_interrupt_work; diff -puN drivers/net/tulip/media.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/media.c --- 25/drivers/net/tulip/media.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/media.c Wed Nov 17 14:13:00 2004 @@ -18,6 +18,7 @@ #include #include #include +#include #include "tulip.h" diff -puN drivers/net/tulip/pnic2.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/pnic2.c --- 25/drivers/net/tulip/pnic2.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/pnic2.c Wed Nov 17 14:13:00 2004 @@ -76,8 +76,8 @@ -#include "tulip.h" #include +#include "tulip.h" #include diff -puN drivers/net/tulip/pnic.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/pnic.c --- 25/drivers/net/tulip/pnic.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/pnic.c Wed Nov 17 14:13:00 2004 @@ -15,6 +15,7 @@ */ #include +#include #include "tulip.h" diff -puN drivers/net/tulip/timer.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/timer.c --- 25/drivers/net/tulip/timer.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/timer.c Wed Nov 17 14:13:00 2004 @@ -14,6 +14,7 @@ */ +#include #include "tulip.h" diff -puN drivers/net/tulip/tulip_core.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/tulip_core.c --- 25/drivers/net/tulip/tulip_core.c~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/tulip_core.c Wed Nov 17 14:13:00 2004 @@ -26,8 +26,8 @@ #include -#include "tulip.h" #include +#include "tulip.h" #include #include #include diff -puN drivers/net/tulip/tulip.h~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop drivers/net/tulip/tulip.h --- 25/drivers/net/tulip/tulip.h~tulip-make-tulip_stop_rxtx-wait-for-dma-to-fully-stop Wed Nov 17 14:13:00 2004 +++ 25-akpm/drivers/net/tulip/tulip.h Wed Nov 17 14:13:00 2004 @@ -149,6 +149,9 @@ enum status_bits { TxIntr = 0x01, }; +/* bit mask for CSR5 TX/RX process state */ +#define CSR5_TS 0x00700000 +#define CSR5_RS 0x000e0000 enum tulip_mode_bits { TxThreshold = (1 << 22), @@ -460,9 +463,19 @@ static inline void tulip_stop_rxtx(struc u32 csr6 = ioread32(ioaddr + CSR6); if (csr6 & RxTx) { + unsigned i=1300/10; iowrite32(csr6 & ~RxTx, ioaddr + CSR6); barrier(); - (void) ioread32(ioaddr + CSR6); /* mmio sync */ + /* wait until in-flight frame completes. + * Max time @ 10BT: 1500*8b/10Mbps == 1200us (+ 100us margin) + * Typically expect this loop to end in < 50 us on 100BT. + */ + while (--i && (ioread32(ioaddr + CSR5) & (CSR5_TS|CSR5_RS))) + udelay(10); + + if (!i) + printk(KERN_DEBUG "%s: tulip_stop_rxtx() failed\n", + tp->pdev->slot_name); } } _