mt76: wait for pending tx to complete before switching channel
authorFelix Fietkau <nbd@nbd.name>
Sun, 20 May 2018 05:43:48 +0000 (07:43 +0200)
committerKalle Valo <kvalo@codeaurora.org>
Wed, 23 May 2018 07:58:14 +0000 (10:58 +0300)
Reduces interruption caused by scanning

Signed-off-by: Felix Fietkau <nbd@nbd.name>
Signed-off-by: Kalle Valo <kvalo@codeaurora.org>
drivers/net/wireless/mediatek/mt76/dma.c
drivers/net/wireless/mediatek/mt76/mac80211.c
drivers/net/wireless/mediatek/mt76/mt76.h

index 3518703..3dbedce 100644 (file)
@@ -178,6 +178,10 @@ mt76_dma_tx_cleanup(struct mt76_dev *dev, enum mt76_txq_id qid, bool flush)
                mt76_dma_sync_idx(dev, q);
 
        wake = wake && qid < IEEE80211_NUM_ACS && q->queued < q->ndesc - 8;
+
+       if (!q->queued)
+               wake_up(&dev->tx_wait);
+
        spin_unlock_bh(&q->lock);
 
        if (wake)
index d1044e5..fcd079a 100644 (file)
@@ -283,6 +283,7 @@ mt76_alloc_device(unsigned int size, const struct ieee80211_ops *ops)
        spin_lock_init(&dev->rx_lock);
        spin_lock_init(&dev->lock);
        spin_lock_init(&dev->cc_lock);
+       init_waitqueue_head(&dev->tx_wait);
 
        return dev;
 }
@@ -377,18 +378,33 @@ void mt76_rx(struct mt76_dev *dev, enum mt76_rxq_id q, struct sk_buff *skb)
 }
 EXPORT_SYMBOL_GPL(mt76_rx);
 
+static bool mt76_has_tx_pending(struct mt76_dev *dev)
+{
+       int i;
+
+       for (i = 0; i < ARRAY_SIZE(dev->q_tx); i++) {
+               if (dev->q_tx[i].queued)
+                       return true;
+       }
+
+       return false;
+}
+
 void mt76_set_channel(struct mt76_dev *dev)
 {
        struct ieee80211_hw *hw = dev->hw;
        struct cfg80211_chan_def *chandef = &hw->conf.chandef;
        struct mt76_channel_state *state;
        bool offchannel = hw->conf.flags & IEEE80211_CONF_OFFCHANNEL;
+       int timeout = HZ / 5;
 
        if (offchannel)
                set_bit(MT76_OFFCHANNEL, &dev->state);
        else
                clear_bit(MT76_OFFCHANNEL, &dev->state);
 
+       wait_event_timeout(dev->tx_wait, !mt76_has_tx_pending(dev), timeout);
+
        if (dev->drv->update_survey)
                dev->drv->update_survey(dev);
 
index bb158f8..d2166fb 100644 (file)
@@ -251,6 +251,8 @@ struct mt76_dev {
        struct mt76_queue q_rx[__MT_RXQ_MAX];
        const struct mt76_queue_ops *queue_ops;
 
+       wait_queue_head_t tx_wait;
+
        u8 macaddr[ETH_ALEN];
        u32 rev;
        unsigned long state;