diff options
author | jhb <jhb@FreeBSD.org> | 2006-04-28 20:08:16 +0000 |
---|---|---|
committer | jhb <jhb@FreeBSD.org> | 2006-04-28 20:08:16 +0000 |
commit | ca65c8d400f4a855b84394629b8695274cf7bfb4 (patch) | |
tree | d47bd827c11d10f92040d69f9f335f6cae2c372b /sys | |
parent | 4bbacc2510826865e6a74758b30aa79da6cc9f92 (diff) | |
download | FreeBSD-src-ca65c8d400f4a855b84394629b8695274cf7bfb4.zip FreeBSD-src-ca65c8d400f4a855b84394629b8695274cf7bfb4.tar.gz |
The nvidia binary blob sometimes defers tx completion notification to the
OS dependent layer. Thus, the watchdog timer can go off when the tx
engine is working fine but the OS dependent layer just hasn't been called
to cleanup finished tx transactions. To workaround this, when the watchdog
fires, poke the binary blob to force it to flush any pending tx
completions. If this drops the pending tx count to zero then just return
without logging a message or resetting the chip.
This reportedly fixes the 'device timeout()' errors with at least several
NF4 nve(4) parts.
Submitted by: Nathan Alexander Whitehorn <nathanw@uchicago.edu> (code)
Submitted by: dg (inspiration for comment and explanation)
MFC after: 1 week
Diffstat (limited to 'sys')
-rw-r--r-- | sys/dev/nve/if_nve.c | 20 |
1 files changed, 19 insertions, 1 deletions
diff --git a/sys/dev/nve/if_nve.c b/sys/dev/nve/if_nve.c index 5fd8e8c..db31d1b 100644 --- a/sys/dev/nve/if_nve.c +++ b/sys/dev/nve/if_nve.c @@ -1277,9 +1277,27 @@ nve_watchdog(struct ifnet *ifp) { struct nve_softc *sc = ifp->if_softc; + NVE_LOCK(sc); + + /* + * The nvidia driver blob defers tx completion notifications. + * Thus, sometimes the watchdog timer will go off when the + * tx engine is fine, but the tx completions are just deferred. + * Try kicking the driver blob to clear out any pending tx + * completions. If that clears up all the pending tx + * operations, then just return without printing the warning + * message or resetting the adapter. + */ + sc->hwapi->pfnDisableInterrupts(sc->hwapi->pADCX); + sc->hwapi->pfnHandleInterrupt(sc->hwapi->pADCX); + sc->hwapi->pfnEnableInterrupts(sc->hwapi->pADCX); + if (sc->pending_txs == 0) { + NVE_UNLOCK(sc); + return; + } + device_printf(sc->dev, "device timeout (%d)\n", sc->pending_txs); - NVE_LOCK(sc); sc->tx_errors++; nve_stop(sc); |