| From 04de28f9fec0bce1f37dd6bb3c2081d91bdc218e Mon Sep 17 00:00:00 2001 |
| From: Peter Chiu <chui-hao.chiu@mediatek.com> |
| Date: Mon, 23 Oct 2023 10:26:01 +0800 |
| Subject: [PATCH 5/6] Revert "wifi: mt76: fix race condition related to |
| checking tx queue fill status" |
| |
| This reverts commit f1e1e67d97d1e9a8bb01b59ab20c45ebc985a958. |
| --- |
| mac80211.c | 27 -------------- |
| mt76.h | 5 --- |
| tx.c | 108 ++++++++++------------------------------------------- |
| 3 files changed, 20 insertions(+), 120 deletions(-) |
| |
| diff --git a/mac80211.c b/mac80211.c |
| index e7c4dc4..4caa714 100644 |
| --- a/mac80211.c |
| +++ b/mac80211.c |
| @@ -427,9 +427,6 @@ mt76_phy_init(struct mt76_phy *phy, struct ieee80211_hw *hw) |
| struct mt76_dev *dev = phy->dev; |
| struct wiphy *wiphy = hw->wiphy; |
| |
| - INIT_LIST_HEAD(&phy->tx_list); |
| - spin_lock_init(&phy->tx_lock); |
| - |
| SET_IEEE80211_DEV(hw, dev->dev); |
| SET_IEEE80211_PERM_ADDR(hw, phy->macaddr); |
| |
| @@ -662,7 +659,6 @@ int mt76_register_device(struct mt76_dev *dev, bool vht, |
| int ret; |
| |
| dev_set_drvdata(dev->dev, dev); |
| - mt76_wcid_init(&dev->global_wcid); |
| ret = mt76_phy_init(phy, hw); |
| if (ret) |
| return ret; |
| @@ -718,7 +714,6 @@ void mt76_unregister_device(struct mt76_dev *dev) |
| if (IS_ENABLED(CONFIG_MT76_LEDS)) |
| mt76_led_cleanup(&dev->phy); |
| mt76_tx_status_check(dev, true); |
| - mt76_wcid_cleanup(dev, &dev->global_wcid); |
| ieee80211_unregister_hw(hw); |
| } |
| EXPORT_SYMBOL_GPL(mt76_unregister_device); |
| @@ -1465,9 +1460,6 @@ EXPORT_SYMBOL_GPL(mt76_sta_pre_rcu_remove); |
| |
| void mt76_wcid_init(struct mt76_wcid *wcid) |
| { |
| - INIT_LIST_HEAD(&wcid->tx_list); |
| - skb_queue_head_init(&wcid->tx_pending); |
| - |
| INIT_LIST_HEAD(&wcid->list); |
| idr_init(&wcid->pktid); |
| } |
| @@ -1475,32 +1467,13 @@ EXPORT_SYMBOL_GPL(mt76_wcid_init); |
| |
| void mt76_wcid_cleanup(struct mt76_dev *dev, struct mt76_wcid *wcid) |
| { |
| - struct mt76_phy *phy = dev->phys[wcid->phy_idx]; |
| - struct ieee80211_hw *hw; |
| struct sk_buff_head list; |
| - struct sk_buff *skb; |
| |
| mt76_tx_status_lock(dev, &list); |
| mt76_tx_status_skb_get(dev, wcid, -1, &list); |
| mt76_tx_status_unlock(dev, &list); |
| |
| idr_destroy(&wcid->pktid); |
| - |
| - spin_lock_bh(&phy->tx_lock); |
| - |
| - if (!list_empty(&wcid->tx_list)) |
| - list_del_init(&wcid->tx_list); |
| - |
| - spin_lock(&wcid->tx_pending.lock); |
| - skb_queue_splice_tail_init(&wcid->tx_pending, &list); |
| - spin_unlock(&wcid->tx_pending.lock); |
| - |
| - spin_unlock_bh(&phy->tx_lock); |
| - |
| - while ((skb = __skb_dequeue(&list)) != NULL) { |
| - hw = mt76_tx_status_get_hw(dev, skb); |
| - ieee80211_free_txskb(hw, skb); |
| - } |
| } |
| EXPORT_SYMBOL_GPL(mt76_wcid_cleanup); |
| |
| diff --git a/mt76.h b/mt76.h |
| index 3ff348f..5eb571e 100644 |
| --- a/mt76.h |
| +++ b/mt76.h |
| @@ -354,9 +354,6 @@ struct mt76_wcid { |
| u32 tx_info; |
| bool sw_iv; |
| |
| - struct list_head tx_list; |
| - struct sk_buff_head tx_pending; |
| - |
| struct list_head list; |
| struct idr pktid; |
| |
| @@ -808,8 +805,6 @@ struct mt76_phy { |
| unsigned long state; |
| u8 band_idx; |
| |
| - spinlock_t tx_lock; |
| - struct list_head tx_list; |
| struct mt76_queue *q_tx[__MT_TXQ_MAX]; |
| |
| struct cfg80211_chan_def chandef; |
| diff --git a/tx.c b/tx.c |
| index 96f9009..c88fb29 100644 |
| --- a/tx.c |
| +++ b/tx.c |
| @@ -328,32 +328,40 @@ void |
| mt76_tx(struct mt76_phy *phy, struct ieee80211_sta *sta, |
| struct mt76_wcid *wcid, struct sk_buff *skb) |
| { |
| + struct mt76_dev *dev = phy->dev; |
| struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); |
| + struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data; |
| + struct mt76_queue *q; |
| + int qid = skb_get_queue_mapping(skb); |
| |
| if (mt76_testmode_enabled(phy)) { |
| ieee80211_free_txskb(phy->hw, skb); |
| return; |
| } |
| |
| - if (WARN_ON(skb_get_queue_mapping(skb) >= MT_TXQ_PSD)) |
| - skb_set_queue_mapping(skb, MT_TXQ_BE); |
| + if (WARN_ON(qid >= MT_TXQ_PSD)) { |
| + qid = MT_TXQ_BE; |
| + skb_set_queue_mapping(skb, qid); |
| + } |
| + |
| + if ((dev->drv->drv_flags & MT_DRV_HW_MGMT_TXQ) && |
| + !(info->flags & IEEE80211_TX_CTL_HW_80211_ENCAP) && |
| + !ieee80211_is_data(hdr->frame_control) && |
| + !ieee80211_is_bufferable_mmpdu(skb)) { |
| + qid = MT_TXQ_PSD; |
| + } |
| |
| if (wcid && !(wcid->tx_info & MT_WCID_TX_INFO_SET)) |
| ieee80211_get_tx_rates(info->control.vif, sta, skb, |
| info->control.rates, 1); |
| |
| info->hw_queue |= FIELD_PREP(MT_TX_HW_QUEUE_PHY, phy->band_idx); |
| + q = phy->q_tx[qid]; |
| |
| - spin_lock_bh(&wcid->tx_pending.lock); |
| - __skb_queue_tail(&wcid->tx_pending, skb); |
| - spin_unlock_bh(&wcid->tx_pending.lock); |
| - |
| - spin_lock_bh(&phy->tx_lock); |
| - if (list_empty(&wcid->tx_list)) |
| - list_add_tail(&wcid->tx_list, &phy->tx_list); |
| - spin_unlock_bh(&phy->tx_lock); |
| - |
| - mt76_worker_schedule(&phy->dev->tx_worker); |
| + spin_lock_bh(&q->lock); |
| + __mt76_tx_queue_skb(phy, qid, skb, wcid, sta, NULL); |
| + dev->queue_ops->kick(dev, q); |
| + spin_unlock_bh(&q->lock); |
| } |
| EXPORT_SYMBOL_GPL(mt76_tx); |
| |
| @@ -584,86 +592,10 @@ void mt76_txq_schedule(struct mt76_phy *phy, enum mt76_txq_id qid) |
| } |
| EXPORT_SYMBOL_GPL(mt76_txq_schedule); |
| |
| -static int |
| -mt76_txq_schedule_pending_wcid(struct mt76_phy *phy, struct mt76_wcid *wcid) |
| -{ |
| - struct mt76_dev *dev = phy->dev; |
| - struct ieee80211_sta *sta; |
| - struct mt76_queue *q; |
| - struct sk_buff *skb; |
| - int ret = 0; |
| - |
| - spin_lock(&wcid->tx_pending.lock); |
| - while ((skb = skb_peek(&wcid->tx_pending)) != NULL) { |
| - struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data; |
| - struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); |
| - int qid = skb_get_queue_mapping(skb); |
| - |
| - if ((dev->drv->drv_flags & MT_DRV_HW_MGMT_TXQ) && |
| - !(info->flags & IEEE80211_TX_CTL_HW_80211_ENCAP) && |
| - !ieee80211_is_data(hdr->frame_control) && |
| - !ieee80211_is_bufferable_mmpdu(skb)) |
| - qid = MT_TXQ_PSD; |
| - |
| - q = phy->q_tx[qid]; |
| - if (mt76_txq_stopped(q)) { |
| - ret = -1; |
| - break; |
| - } |
| - |
| - __skb_unlink(skb, &wcid->tx_pending); |
| - spin_unlock(&wcid->tx_pending.lock); |
| - |
| - sta = wcid_to_sta(wcid); |
| - spin_lock(&q->lock); |
| - __mt76_tx_queue_skb(phy, qid, skb, wcid, sta, NULL); |
| - dev->queue_ops->kick(dev, q); |
| - spin_unlock(&q->lock); |
| - |
| - spin_lock(&wcid->tx_pending.lock); |
| - } |
| - spin_unlock(&wcid->tx_pending.lock); |
| - |
| - return ret; |
| -} |
| - |
| -static void mt76_txq_schedule_pending(struct mt76_phy *phy) |
| -{ |
| - if (list_empty(&phy->tx_list)) |
| - return; |
| - |
| - local_bh_disable(); |
| - rcu_read_lock(); |
| - |
| - spin_lock(&phy->tx_lock); |
| - while (!list_empty(&phy->tx_list)) { |
| - struct mt76_wcid *wcid = NULL; |
| - int ret; |
| - |
| - wcid = list_first_entry(&phy->tx_list, struct mt76_wcid, tx_list); |
| - list_del_init(&wcid->tx_list); |
| - |
| - spin_unlock(&phy->tx_lock); |
| - ret = mt76_txq_schedule_pending_wcid(phy, wcid); |
| - spin_lock(&phy->tx_lock); |
| - |
| - if (ret) { |
| - if (list_empty(&wcid->tx_list)) |
| - list_add_tail(&wcid->tx_list, &phy->tx_list); |
| - break; |
| - } |
| - } |
| - spin_unlock(&phy->tx_lock); |
| - |
| - rcu_read_unlock(); |
| - local_bh_enable(); |
| -} |
| - |
| void mt76_txq_schedule_all(struct mt76_phy *phy) |
| { |
| int i; |
| |
| - mt76_txq_schedule_pending(phy); |
| for (i = 0; i <= MT_TXQ_BK; i++) |
| mt76_txq_schedule(phy, i); |
| } |
| -- |
| 2.18.0 |
| |