ath9k: fix retry counting / BAR handling during queue flush
authorFelix Fietkau <nbd@openwrt.org>
Fri, 7 Oct 2011 00:28:15 +0000 (02:28 +0200)
committerJohn W. Linville <linville@tuxdriver.com>
Tue, 11 Oct 2011 20:41:15 +0000 (16:41 -0400)
When tx is suspended temporarily and the queue is flushed, do not increase
the retry count or attempt to send out BAR frames. Instead simply retry
the affected subframes normally after the reset.

Signed-off-by: Felix Fietkau <nbd@openwrt.org>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
drivers/net/wireless/ath/ath9k/mac.h
drivers/net/wireless/ath/ath9k/xmit.c

index 91c96546c0cde73c395646819e552a65272bcab6..5eb4ee454325af84e42d32c260cfb4976d773780 100644 (file)
 #define ATH9K_TXERR_XTXOP          0x08
 #define ATH9K_TXERR_TIMER_EXPIRED  0x10
 #define ATH9K_TX_ACKED            0x20
+#define ATH9K_TX_FLUSH            0x40
 #define ATH9K_TXERR_MASK                                               \
        (ATH9K_TXERR_XRETRY | ATH9K_TXERR_FILT | ATH9K_TXERR_FIFO |     \
-        ATH9K_TXERR_XTXOP | ATH9K_TXERR_TIMER_EXPIRED)
+        ATH9K_TXERR_XTXOP | ATH9K_TXERR_TIMER_EXPIRED | ATH9K_TX_FLUSH)
 
 #define ATH9K_TX_BA                0x01
 #define ATH9K_TX_PWRMGMT           0x02
index 9903fc3af723ecd976f4adce6c2dddbd46ec4f49..03b0a651a591c39e79b331ca3c303cc0f08e6d93 100644 (file)
@@ -373,6 +373,7 @@ static void ath_tx_complete_aggr(struct ath_softc *sc, struct ath_txq *txq,
        struct ath_frame_info *fi;
        int nframes;
        u8 tidno;
+       bool flush = !!(ts->ts_status & ATH9K_TX_FLUSH);
 
        skb = bf->bf_mpdu;
        hdr = (struct ieee80211_hdr *)skb->data;
@@ -461,6 +462,8 @@ static void ath_tx_complete_aggr(struct ath_softc *sc, struct ath_txq *txq,
                                 * the un-acked sub-frames
                                 */
                                txfail = 1;
+                       } else if (flush) {
+                               txpending = 1;
                        } else if (fi->retries < ATH_MAX_SW_RETRIES) {
                                if (txok || !an->sleeping)
                                        ath_tx_set_retry(sc, txq, bf->bf_mpdu);
@@ -518,7 +521,8 @@ static void ath_tx_complete_aggr(struct ath_softc *sc, struct ath_txq *txq,
 
                                                ath_tx_complete_buf(sc, bf, txq,
                                                                    &bf_head,
-                                                                   ts, 0, 1);
+                                                                   ts, 0,
+                                                                   !flush);
                                                break;
                                        }
 
@@ -1401,6 +1405,7 @@ static void ath_drain_txq_list(struct ath_softc *sc, struct ath_txq *txq,
        struct ath_tx_status ts;
 
        memset(&ts, 0, sizeof(ts));
+       ts.ts_status = ATH9K_TX_FLUSH;
        INIT_LIST_HEAD(&bf_head);
 
        while (!list_empty(list)) {