Linux-Wireless Archive mirror
 help / color / mirror / Atom feed
* [PATCH] ath9k: Implement op_flush()
@ 2011-02-19  9:13 Vasanthakumar Thiagarajan
  2011-02-19  9:24 ` Johannes Berg
  0 siblings, 1 reply; 7+ messages in thread
From: Vasanthakumar Thiagarajan @ 2011-02-19  9:13 UTC (permalink / raw
  To: linville; +Cc: linux-wireless

When op_flush() is called with no drop (drop=false), the driver
tries to tx as many frames as possible in about 100ms on every
hw queue. During this time period frames from sw queue are also
scheduled on to respective hw queue.

Signed-off-by: Vasanthakumar Thiagarajan <vasanth@atheros.com>
---
 drivers/net/wireless/ath/ath9k/ath9k.h |    1 +
 drivers/net/wireless/ath/ath9k/main.c  |   71 ++++++++++++++++++++++++++++++++
 drivers/net/wireless/ath/ath9k/xmit.c  |   27 +++++++-----
 3 files changed, 88 insertions(+), 11 deletions(-)

diff --git a/drivers/net/wireless/ath/ath9k/ath9k.h b/drivers/net/wireless/ath/ath9k/ath9k.h
index a224c56..f9f0389 100644
--- a/drivers/net/wireless/ath/ath9k/ath9k.h
+++ b/drivers/net/wireless/ath/ath9k/ath9k.h
@@ -189,6 +189,7 @@ struct ath_txq {
 	u32 axq_ampdu_depth;
 	bool stopped;
 	bool axq_tx_inprogress;
+	bool txq_flush_inprogress;
 	struct list_head axq_acq;
 	struct list_head txq_fifo[ATH_TXFIFO_DEPTH];
 	struct list_head txq_fifo_pending;
diff --git a/drivers/net/wireless/ath/ath9k/main.c b/drivers/net/wireless/ath/ath9k/main.c
index 1d2c7c3..a715500 100644
--- a/drivers/net/wireless/ath/ath9k/main.c
+++ b/drivers/net/wireless/ath/ath9k/main.c
@@ -15,6 +15,7 @@
  */
 
 #include <linux/nl80211.h>
+#include <linux/delay.h>
 #include "ath9k.h"
 #include "btcoex.h"
 
@@ -53,6 +54,21 @@ static u8 parse_mpdudensity(u8 mpdudensity)
 	}
 }
 
+static bool ath9k_has_pending_frames(struct ath_softc *sc, struct ath_txq *txq)
+{
+	bool pending = false;
+
+	spin_lock_bh(&txq->axq_lock);
+
+	if (txq->axq_depth || !list_empty(&txq->axq_acq))
+		pending = true;
+	else if (sc->sc_ah->caps.hw_caps & ATH9K_HW_CAP_EDMA)
+		pending = !list_empty(&txq->txq_fifo_pending);
+
+	spin_unlock_bh(&txq->axq_lock);
+	return pending;
+}
+
 bool ath9k_setpower(struct ath_softc *sc, enum ath9k_power_mode mode)
 {
 	unsigned long flags;
@@ -2111,6 +2127,60 @@ static void ath9k_set_coverage_class(struct ieee80211_hw *hw, u8 coverage_class)
 	mutex_unlock(&sc->mutex);
 }
 
+static void ath9k_flush(struct ieee80211_hw *hw, bool drop)
+{
+#define ATH_FLUSH_TIMEOUT	60 /* ms */
+	struct ath_softc *sc = hw->priv;
+	struct ath_txq *txq;
+	struct ath_hw *ah = sc->sc_ah;
+	struct ath_common *common = ath9k_hw_common(ah);
+	int i, j, npend = 0;
+
+	mutex_lock(&sc->mutex);
+
+	cancel_delayed_work_sync(&sc->tx_complete_work);
+
+	for (i = 0; i < ATH9K_NUM_TX_QUEUES; i++) {
+		if (!ATH_TXQ_SETUP(sc, i))
+			continue;
+		txq = &sc->tx.txq[i];
+
+		if (!drop) {
+			for (j = 0; j < ATH_FLUSH_TIMEOUT; j++) {
+				if (!ath9k_has_pending_frames(sc, txq))
+					break;
+				usleep_range(1000, 2000);
+			}
+		}
+
+		if (drop || ath9k_has_pending_frames(sc, txq)) {
+			ath_dbg(common, ATH_DBG_QUEUE, "Drop frames from hw queue:%d\n",
+				txq->axq_qnum);
+			spin_lock_bh(&txq->axq_lock);
+			txq->txq_flush_inprogress = true;
+			spin_unlock_bh(&txq->axq_lock);
+
+			ath9k_ps_wakeup(sc);
+			ath9k_hw_stoptxdma(ah, txq->axq_qnum);
+			npend = ath9k_hw_numtxpending(ah, txq->axq_qnum);
+			ath9k_ps_restore(sc);
+			if (npend)
+				break;
+
+			ath_draintxq(sc, txq, false);
+			txq->txq_flush_inprogress = false;
+		}
+	}
+
+	if (npend) {
+		ath_reset(sc, false);
+		txq->txq_flush_inprogress = false;
+	}
+
+	ieee80211_queue_delayed_work(hw, &sc->tx_complete_work, 0);
+	mutex_unlock(&sc->mutex);
+}
+
 struct ieee80211_ops ath9k_ops = {
 	.tx 		    = ath9k_tx,
 	.start 		    = ath9k_start,
@@ -2132,4 +2202,5 @@ struct ieee80211_ops ath9k_ops = {
 	.get_survey	    = ath9k_get_survey,
 	.rfkill_poll        = ath9k_rfkill_poll_state,
 	.set_coverage_class = ath9k_set_coverage_class,
+	.flush		    = ath9k_flush,
 };
diff --git a/drivers/net/wireless/ath/ath9k/xmit.c b/drivers/net/wireless/ath/ath9k/xmit.c
index bc614ac..e16136d 100644
--- a/drivers/net/wireless/ath/ath9k/xmit.c
+++ b/drivers/net/wireless/ath/ath9k/xmit.c
@@ -2014,7 +2014,8 @@ static void ath_tx_processq(struct ath_softc *sc, struct ath_txq *txq)
 		spin_lock_bh(&txq->axq_lock);
 		if (list_empty(&txq->axq_q)) {
 			txq->axq_link = NULL;
-			if (sc->sc_flags & SC_OP_TXAGGR)
+			if (sc->sc_flags & SC_OP_TXAGGR &&
+			    !txq->txq_flush_inprogress)
 				ath_txq_schedule(sc, txq);
 			spin_unlock_bh(&txq->axq_lock);
 			break;
@@ -2071,6 +2072,7 @@ static void ath_tx_processq(struct ath_softc *sc, struct ath_txq *txq)
 
 		if (bf_is_ampdu_not_probing(bf))
 			txq->axq_ampdu_depth--;
+
 		spin_unlock_bh(&txq->axq_lock);
 
 		if (bf_held)
@@ -2094,7 +2096,7 @@ static void ath_tx_processq(struct ath_softc *sc, struct ath_txq *txq)
 
 		spin_lock_bh(&txq->axq_lock);
 
-		if (sc->sc_flags & SC_OP_TXAGGR)
+		if (sc->sc_flags & SC_OP_TXAGGR && !txq->txq_flush_inprogress)
 			ath_txq_schedule(sc, txq);
 		spin_unlock_bh(&txq->axq_lock);
 	}
@@ -2265,15 +2267,18 @@ void ath_tx_edma_tasklet(struct ath_softc *sc)
 
 		spin_lock_bh(&txq->axq_lock);
 
-		if (!list_empty(&txq->txq_fifo_pending)) {
-			INIT_LIST_HEAD(&bf_head);
-			bf = list_first_entry(&txq->txq_fifo_pending,
-				struct ath_buf, list);
-			list_cut_position(&bf_head, &txq->txq_fifo_pending,
-				&bf->bf_lastbf->list);
-			ath_tx_txqaddbuf(sc, txq, &bf_head);
-		} else if (sc->sc_flags & SC_OP_TXAGGR)
-			ath_txq_schedule(sc, txq);
+		if (!txq->txq_flush_inprogress) {
+			if (!list_empty(&txq->txq_fifo_pending)) {
+				INIT_LIST_HEAD(&bf_head);
+				bf = list_first_entry(&txq->txq_fifo_pending,
+						      struct ath_buf, list);
+				list_cut_position(&bf_head,
+						  &txq->txq_fifo_pending,
+						  &bf->bf_lastbf->list);
+				ath_tx_txqaddbuf(sc, txq, &bf_head);
+			} else if (sc->sc_flags & SC_OP_TXAGGR)
+				ath_txq_schedule(sc, txq);
+		}
 		spin_unlock_bh(&txq->axq_lock);
 	}
 }
-- 
1.7.0.4


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19  9:13 [PATCH] ath9k: Implement op_flush() Vasanthakumar Thiagarajan
@ 2011-02-19  9:24 ` Johannes Berg
  2011-02-19  9:49   ` Vasanthakumar Thiagarajan
  0 siblings, 1 reply; 7+ messages in thread
From: Johannes Berg @ 2011-02-19  9:24 UTC (permalink / raw
  To: Vasanthakumar Thiagarajan; +Cc: linville, linux-wireless

On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> When op_flush() is called with no drop (drop=false), the driver
> tries to tx as many frames as possible in about 100ms on every
> hw queue. During this time period frames from sw queue are also
> scheduled on to respective hw queue.

Given how long HW queues currently are, I wouldn't set the timeout to
100ms -- mac80211 has no expectation how long this will take, although
100ms seems pretty long I'm not sure it'll always be sufficient?

johannes


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19  9:24 ` Johannes Berg
@ 2011-02-19  9:49   ` Vasanthakumar Thiagarajan
  2011-02-19  9:56     ` Johannes Berg
  0 siblings, 1 reply; 7+ messages in thread
From: Vasanthakumar Thiagarajan @ 2011-02-19  9:49 UTC (permalink / raw
  To: Johannes Berg
  Cc: Vasanth Thiagarajan, linville@tuxdriver.com,
	linux-wireless@vger.kernel.org

On Sat, Feb 19, 2011 at 02:54:00PM +0530, Johannes Berg wrote:
> On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> > When op_flush() is called with no drop (drop=false), the driver
> > tries to tx as many frames as possible in about 100ms on every
> > hw queue. During this time period frames from sw queue are also
> > scheduled on to respective hw queue.
> 
> Given how long HW queues currently are, I wouldn't set the timeout to
> 100ms -- mac80211 has no expectation how long this will take, although
> 100ms seems pretty long I'm not sure it'll always be sufficient?

It is not that we wait for 100ms always, we return as soon as
possible if there are no pending frames in sw/hw queues. I never
hit this timeout though. In the worst case there can be 128 (4 aggr)
frames pending in sw queue and 2 in hw queue. If we assume each one of
these aggregates has 4ms duration, we at least need 24ms air time at
the lowest rate. If they are not part of AMPDU, it would take a
little bit more I think. So, probably we can give ~40ms timeout,
is that reasonable?. If i'm reading the code correctly, iwlwifi
seems to use 2000ms.

vasanth

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19  9:49   ` Vasanthakumar Thiagarajan
@ 2011-02-19  9:56     ` Johannes Berg
  2011-02-19 10:00       ` Vasanthakumar Thiagarajan
  2011-02-19 10:09       ` Vasanthakumar Thiagarajan
  0 siblings, 2 replies; 7+ messages in thread
From: Johannes Berg @ 2011-02-19  9:56 UTC (permalink / raw
  To: Vasanthakumar Thiagarajan
  Cc: Vasanth Thiagarajan, linville@tuxdriver.com,
	linux-wireless@vger.kernel.org

On Sat, 2011-02-19 at 15:19 +0530, Vasanthakumar Thiagarajan wrote:
> On Sat, Feb 19, 2011 at 02:54:00PM +0530, Johannes Berg wrote:
> > On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> > > When op_flush() is called with no drop (drop=false), the driver
> > > tries to tx as many frames as possible in about 100ms on every
> > > hw queue. During this time period frames from sw queue are also
> > > scheduled on to respective hw queue.
> > 
> > Given how long HW queues currently are, I wouldn't set the timeout to
> > 100ms -- mac80211 has no expectation how long this will take, although
> > 100ms seems pretty long I'm not sure it'll always be sufficient?
> 
> It is not that we wait for 100ms always, we return as soon as
> possible if there are no pending frames in sw/hw queues. I never
> hit this timeout though. In the worst case there can be 128 (4 aggr)
> frames pending in sw queue and 2 in hw queue. If we assume each one of
> these aggregates has 4ms duration, we at least need 24ms air time at
> the lowest rate. If they are not part of AMPDU, it would take a
> little bit more I think. So, probably we can give ~40ms timeout,
> is that reasonable?. If i'm reading the code correctly, iwlwifi
> seems to use 2000ms.

Ok. I wasn't concerned about it being too high, I was thinking 100ms
might not be enough, but since you say it will be I guess it's fine.

johannes


^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19  9:56     ` Johannes Berg
@ 2011-02-19 10:00       ` Vasanthakumar Thiagarajan
  2011-02-19 10:09       ` Vasanthakumar Thiagarajan
  1 sibling, 0 replies; 7+ messages in thread
From: Vasanthakumar Thiagarajan @ 2011-02-19 10:00 UTC (permalink / raw
  To: Johannes Berg
  Cc: Vasanth Thiagarajan, linville@tuxdriver.com,
	linux-wireless@vger.kernel.org

On Sat, Feb 19, 2011 at 03:26:30PM +0530, Johannes Berg wrote:
> On Sat, 2011-02-19 at 15:19 +0530, Vasanthakumar Thiagarajan wrote:
> > On Sat, Feb 19, 2011 at 02:54:00PM +0530, Johannes Berg wrote:
> > > On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> > > > When op_flush() is called with no drop (drop=false), the driver
> > > > tries to tx as many frames as possible in about 100ms on every
> > > > hw queue. During this time period frames from sw queue are also
> > > > scheduled on to respective hw queue.
> > > 
> > > Given how long HW queues currently are, I wouldn't set the timeout to
> > > 100ms -- mac80211 has no expectation how long this will take, although
> > > 100ms seems pretty long I'm not sure it'll always be sufficient?
> > 
> > It is not that we wait for 100ms always, we return as soon as
> > possible if there are no pending frames in sw/hw queues. I never
> > hit this timeout though. In the worst case there can be 128 (4 aggr)
> > frames pending in sw queue and 2 in hw queue. If we assume each one of
> > these aggregates has 4ms duration, we at least need 24ms air time at
> > the lowest rate. If they are not part of AMPDU, it would take a
> > little bit more I think. So, probably we can give ~40ms timeout,
> > is that reasonable?. If i'm reading the code correctly, iwlwifi
> > seems to use 2000ms.
> 
> Ok. I wasn't concerned about it being too high, I was thinking 100ms
> might not be enough, but since you say it will be I guess it's fine.

Yeah, I misread it. This 100ms timeout is on every ac queue and is pretty
much good enough from my testing.

Vasanth

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19  9:56     ` Johannes Berg
  2011-02-19 10:00       ` Vasanthakumar Thiagarajan
@ 2011-02-19 10:09       ` Vasanthakumar Thiagarajan
  2011-02-19 10:18         ` Johannes Berg
  1 sibling, 1 reply; 7+ messages in thread
From: Vasanthakumar Thiagarajan @ 2011-02-19 10:09 UTC (permalink / raw
  To: Johannes Berg
  Cc: Vasanth Thiagarajan, linville@tuxdriver.com,
	linux-wireless@vger.kernel.org

On Sat, Feb 19, 2011 at 03:26:30PM +0530, Johannes Berg wrote:
> On Sat, 2011-02-19 at 15:19 +0530, Vasanthakumar Thiagarajan wrote:
> > On Sat, Feb 19, 2011 at 02:54:00PM +0530, Johannes Berg wrote:
> > > On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> > > > When op_flush() is called with no drop (drop=false), the driver
> > > > tries to tx as many frames as possible in about 100ms on every
> > > > hw queue. During this time period frames from sw queue are also
> > > > scheduled on to respective hw queue.
> > > 
> > > Given how long HW queues currently are, I wouldn't set the timeout to
> > > 100ms -- mac80211 has no expectation how long this will take, although
> > > 100ms seems pretty long I'm not sure it'll always be sufficient?

Would not having a long timeout for flush affect bg scanning by
applications (scan time out)?.

Vasanth

^ permalink raw reply	[flat|nested] 7+ messages in thread

* Re: [PATCH] ath9k: Implement op_flush()
  2011-02-19 10:09       ` Vasanthakumar Thiagarajan
@ 2011-02-19 10:18         ` Johannes Berg
  0 siblings, 0 replies; 7+ messages in thread
From: Johannes Berg @ 2011-02-19 10:18 UTC (permalink / raw
  To: Vasanthakumar Thiagarajan
  Cc: Vasanth Thiagarajan, linville@tuxdriver.com,
	linux-wireless@vger.kernel.org

On Sat, 2011-02-19 at 15:39 +0530, Vasanthakumar Thiagarajan wrote:
> On Sat, Feb 19, 2011 at 03:26:30PM +0530, Johannes Berg wrote:
> > On Sat, 2011-02-19 at 15:19 +0530, Vasanthakumar Thiagarajan wrote:
> > > On Sat, Feb 19, 2011 at 02:54:00PM +0530, Johannes Berg wrote:
> > > > On Sat, 2011-02-19 at 01:13 -0800, Vasanthakumar Thiagarajan wrote:
> > > > > When op_flush() is called with no drop (drop=false), the driver
> > > > > tries to tx as many frames as possible in about 100ms on every
> > > > > hw queue. During this time period frames from sw queue are also
> > > > > scheduled on to respective hw queue.
> > > > 
> > > > Given how long HW queues currently are, I wouldn't set the timeout to
> > > > 100ms -- mac80211 has no expectation how long this will take, although
> > > > 100ms seems pretty long I'm not sure it'll always be sufficient?
> 
> Would not having a long timeout for flush affect bg scanning by
> applications (scan time out)?.

In theory, yes. In practise, if the timeout is so low, your application
is not going to work right anyway :)

johannes


^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2011-02-19 10:18 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2011-02-19  9:13 [PATCH] ath9k: Implement op_flush() Vasanthakumar Thiagarajan
2011-02-19  9:24 ` Johannes Berg
2011-02-19  9:49   ` Vasanthakumar Thiagarajan
2011-02-19  9:56     ` Johannes Berg
2011-02-19 10:00       ` Vasanthakumar Thiagarajan
2011-02-19 10:09       ` Vasanthakumar Thiagarajan
2011-02-19 10:18         ` Johannes Berg

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).