summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorStefan Sperling <stsp@cvs.openbsd.org>2019-11-06 14:52:36 +0000
committerStefan Sperling <stsp@cvs.openbsd.org>2019-11-06 14:52:36 +0000
commitffee2699f85df2149fc15a9208c48733d63daf6b (patch)
tree5ef199f8c7ac0d1cfd0932a03d06f8155cb0f95e
parent4fcc9f259fd9b4b6e8623b59b08b24cb87c787a2 (diff)
Make iwn(4) flush remaining frames on the Tx aggregation queue when
Tx aggregation is stopped. Fixes a bug where outstanding frames on the aggregation queue interfere with roaming to another AP. net80211 will only roam once all outstanding frames destined for the old AP have been sent, i.e. once that AP node's Tx refcount goes to 0. Any outstanding frames sitting in the Tx aggregation queue, waiting to be ACKed, will keep this refcount above zero. To make roaming work reliably in combination with Tx aggregation, the driver must ensure that ieee80211_release_node() gets called for each frame on the queue when Tx aggregation is stopped. Problem observed by tobhe@ Fix tested + ok tobhe@ jca@
-rw-r--r--sys/dev/pci/if_iwn.c6
1 files changed, 5 insertions, 1 deletions
diff --git a/sys/dev/pci/if_iwn.c b/sys/dev/pci/if_iwn.c
index 8b405460ed1..72e19d7e356 100644
--- a/sys/dev/pci/if_iwn.c
+++ b/sys/dev/pci/if_iwn.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_iwn.c,v 1.220 2019/11/06 13:55:43 stsp Exp $ */
+/* $OpenBSD: if_iwn.c,v 1.221 2019/11/06 14:52:35 stsp Exp $ */
/*-
* Copyright (c) 2007-2010 Damien Bergamini <damien.bergamini@free.fr>
@@ -5689,6 +5689,10 @@ iwn_ampdu_tx_stop(struct ieee80211com *ic, struct ieee80211_node *ni,
struct iwn_node *wn = (void *)ni;
struct iwn_node_info node;
+ /* Discard all frames in the current window. */
+ iwn_ampdu_txq_advance(sc, &sc->txq[qid], qid,
+ IWN_AGG_SSN_TO_TXQ_IDX(ba->ba_winend));
+
if (iwn_nic_lock(sc) != 0)
return;
ops->ampdu_tx_stop(sc, tid, ba->ba_winstart);