summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authoradrian <adrian@FreeBSD.org>2013-05-10 10:06:45 +0000
committeradrian <adrian@FreeBSD.org>2013-05-10 10:06:45 +0000
commit4d334b94a501aa2decee54a72c4cb04f2f8c0a2b (patch)
tree2c968a1e648997097ec0de74ead473137884e107
parent080f111dcbace394f7e22ba2d07553092dab6bad (diff)
downloadFreeBSD-src-4d334b94a501aa2decee54a72c4cb04f2f8c0a2b.zip
FreeBSD-src-4d334b94a501aa2decee54a72c4cb04f2f8c0a2b.tar.gz
Make sure the holding descriptor and link pointer are both freed during
a non-loss reset. When the drain functions are called, the holding descriptor and link pointers are NULLed out. But when the processq function is called during a non-loss reset, this doesn't occur. So the next time a DMA occurs, it's chained to a descriptor that no longer exists and the hardware gets angry. Tested: * AR5416, STA mode; use sysctl dev.ath.X.forcebstuck=1 to force a non-loss reset. TODO: * Further AR9380 testing just to check that the behaviour for the EDMA chips is sane. PR: kern/178477
-rw-r--r--sys/dev/ath/if_ath.c16
-rw-r--r--sys/dev/ath/if_ath_tx_edma.c16
2 files changed, 30 insertions, 2 deletions
diff --git a/sys/dev/ath/if_ath.c b/sys/dev/ath/if_ath.c
index 6f5fb45..5331fd0 100644
--- a/sys/dev/ath/if_ath.c
+++ b/sys/dev/ath/if_ath.c
@@ -4668,9 +4668,21 @@ ath_legacy_tx_drain(struct ath_softc *sc, ATH_RESET_TYPE reset_type)
if (sc->sc_debug & ATH_DEBUG_RESET)
ath_tx_dump(sc, &sc->sc_txq[i]);
#endif /* ATH_DEBUG */
- if (reset_type == ATH_RESET_NOLOSS)
+ if (reset_type == ATH_RESET_NOLOSS) {
ath_tx_processq(sc, &sc->sc_txq[i], 0);
- else
+ ATH_TXQ_LOCK(&sc->sc_txq[i]);
+ /*
+ * Free the holding buffer; DMA is now
+ * stopped.
+ */
+ ath_txq_freeholdingbuf(sc, &sc->sc_txq[i]);
+ /*
+ * Reset the link pointer to NULL; there's
+ * no frames to chain DMA to.
+ */
+ sc->sc_txq[i].axq_link = NULL;
+ ATH_TXQ_UNLOCK(&sc->sc_txq[i]);
+ } else
ath_tx_draintxq(sc, &sc->sc_txq[i]);
}
}
diff --git a/sys/dev/ath/if_ath_tx_edma.c b/sys/dev/ath/if_ath_tx_edma.c
index b86352a..5498dd5 100644
--- a/sys/dev/ath/if_ath_tx_edma.c
+++ b/sys/dev/ath/if_ath_tx_edma.c
@@ -551,6 +551,22 @@ ath_edma_tx_drain(struct ath_softc *sc, ATH_RESET_TYPE reset_type)
*/
if (reset_type == ATH_RESET_NOLOSS) {
ath_edma_tx_processq(sc, 0);
+ for (i = 0; i < HAL_NUM_TX_QUEUES; i++) {
+ if (ATH_TXQ_SETUP(sc, i)) {
+ ATH_TXQ_LOCK(&sc->sc_txq[i]);
+ /*
+ * Free the holding buffer; DMA is now
+ * stopped.
+ */
+ ath_txq_freeholdingbuf(sc, &sc->sc_txq[i]);
+ /*
+ * Reset the link pointer to NULL; there's
+ * no frames to chain DMA to.
+ */
+ sc->sc_txq[i].axq_link = NULL;
+ ATH_TXQ_UNLOCK(&sc->sc_txq[i]);
+ }
+ }
} else {
for (i = 0; i < HAL_NUM_TX_QUEUES; i++) {
if (ATH_TXQ_SETUP(sc, i))
OpenPOWER on IntegriCloud