]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
wifi: mt76: mt7996: fix reading zeroed info->control.flags after mt76_tx_status_skb_add()
authorLorenzo Bianconi <lorenzo@kernel.org>
Sun, 31 May 2026 08:55:04 +0000 (10:55 +0200)
committerFelix Fietkau <nbd@nbd.name>
Tue, 9 Jun 2026 10:15:22 +0000 (10:15 +0000)
mt76_tx_status_skb_add() zeroes the mt76_tx_cb struct stored at
info->status.status_driver_data via memset(). Since info->control and
info->status are members of the same union in ieee80211_tx_info,
this overwrites info->control.flags.
In mt7996_tx_prepare_skb(), mt76_tx_status_skb_add() is called before
mt7996_mac_write_txwi(), which re-reads info->control.flags to extract
IEEE80211_TX_CTRL_MLO_LINK. Because the field has been zeroed, the
link_id always resolves to 0 for frames using global_wcid, leading to
incorrect TXWI configuration.
Fix this by passing link_id as an explicit parameter to
mt7996_mac_write_txwi(). In mt7996_tx_prepare_skb(), the link_id is
already extracted from info->control.flags before the destructive
mt76_tx_status_skb_add() call. For the beacon and inband discovery
callers in mcu.c, use link_conf->link_id directly.

Fixes: f0b0b239b8f36 ("wifi: mt76: mt7996: rework mt7996_mac_write_txwi() for MLO support")
Signed-off-by: Lorenzo Bianconi <lorenzo@kernel.org>
Link: https://patch.msgid.link/20260531-mt76_tx_status_skb_add-overwrite-fix-v2-1-b73c4b4a9798@kernel.org
Signed-off-by: Felix Fietkau <nbd@nbd.name>
drivers/net/wireless/mediatek/mt76/mt7996/mac.c
drivers/net/wireless/mediatek/mt76/mt7996/mcu.c
drivers/net/wireless/mediatek/mt76/mt7996/mt7996.h

index 2224fbe043916b3b7c1f08421994612a6e3c5c7d..44713a7a86f8cc2259a7f0b7e980786dc8e6c726 100644 (file)
@@ -857,7 +857,8 @@ mt7996_mac_write_txwi_80211(struct mt7996_dev *dev, __le32 *txwi,
 void mt7996_mac_write_txwi(struct mt7996_dev *dev, __le32 *txwi,
                           struct sk_buff *skb, struct mt76_wcid *wcid,
                           struct ieee80211_key_conf *key, int pid,
-                          enum mt76_txq_id qid, u32 changed)
+                          enum mt76_txq_id qid, u32 changed,
+                          unsigned int link_id)
 {
        struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)skb->data;
        struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb);
@@ -867,7 +868,6 @@ void mt7996_mac_write_txwi(struct mt7996_dev *dev, __le32 *txwi,
        bool is_8023 = info->flags & IEEE80211_TX_CTL_HW_80211_ENCAP;
        struct mt76_vif_link *mlink = NULL;
        struct mt7996_vif *mvif;
-       unsigned int link_id;
        u16 tx_count = 15;
        u32 val;
        bool inband_disc = !!(changed & (BSS_CHANGED_UNSOL_BCAST_PROBE_RESP |
@@ -875,17 +875,11 @@ void mt7996_mac_write_txwi(struct mt7996_dev *dev, __le32 *txwi,
        bool beacon = !!(changed & (BSS_CHANGED_BEACON |
                                    BSS_CHANGED_BEACON_ENABLED)) && (!inband_disc);
 
-       if (wcid != &dev->mt76.global_wcid)
-               link_id = wcid->link_id;
-       else
-               link_id = u32_get_bits(info->control.flags,
-                                      IEEE80211_TX_CTRL_MLO_LINK);
-
        mvif = vif ? (struct mt7996_vif *)vif->drv_priv : NULL;
        if (mvif) {
                if (wcid->offchannel)
                        mlink = rcu_dereference(mvif->mt76.offchannel_link);
-               if (!mlink)
+               if (!mlink && link_id != IEEE80211_LINK_UNSPECIFIED)
                        mlink = rcu_dereference(mvif->mt76.link[link_id]);
        }
 
@@ -1097,7 +1091,7 @@ int mt7996_tx_prepare_skb(struct mt76_dev *mdev, void *txwi_ptr,
        /* Transmit non qos data by 802.11 header and need to fill txd by host*/
        if (!is_8023 || pid >= MT_PACKET_ID_FIRST)
                mt7996_mac_write_txwi(dev, txwi_ptr, tx_info->skb, wcid, key,
-                                     pid, qid, 0);
+                                     pid, qid, 0, link_id);
 
        /* MT7996 and MT7992 require driver to provide the MAC TXP for AddBA
         * req
index 8be40d60ad293f1cd716cc9c9328cb38b6306b2e..a14c63438923d60a0df7c7c436fc69a292dc9de2 100644 (file)
@@ -3103,7 +3103,7 @@ mt7996_mcu_beacon_cont(struct mt7996_dev *dev,
 
        buf = (u8 *)bcn + sizeof(*bcn);
        mt7996_mac_write_txwi(dev, (__le32 *)buf, skb, wcid, NULL, 0, 0,
-                             BSS_CHANGED_BEACON);
+                             BSS_CHANGED_BEACON, link_conf->link_id);
 
        memcpy(buf + MT_TXD_SIZE, skb->data, skb->len);
 }
@@ -3249,7 +3249,8 @@ int mt7996_mcu_beacon_inband_discov(struct mt7996_dev *dev,
 
        buf = (u8 *)tlv + sizeof(*discov);
 
-       mt7996_mac_write_txwi(dev, (__le32 *)buf, skb, wcid, NULL, 0, 0, changed);
+       mt7996_mac_write_txwi(dev, (__le32 *)buf, skb, wcid, NULL, 0, 0,
+                             changed, link_conf->link_id);
 
        memcpy(buf + MT_TXD_SIZE, skb->data, skb->len);
 
index 0dc4198fcf8bbc75d6a2d96aaa052f39723fb9f1..0d6488522ba7115de68841f4735eeb270d2bd5ec 100644 (file)
@@ -874,7 +874,8 @@ void mt7996_mac_enable_nf(struct mt7996_dev *dev, u8 band);
 void mt7996_mac_write_txwi(struct mt7996_dev *dev, __le32 *txwi,
                           struct sk_buff *skb, struct mt76_wcid *wcid,
                           struct ieee80211_key_conf *key, int pid,
-                          enum mt76_txq_id qid, u32 changed);
+                          enum mt76_txq_id qid, u32 changed,
+                          unsigned int link_id);
 void mt7996_mac_update_beacons(struct mt7996_phy *phy);
 void mt7996_mac_set_coverage_class(struct mt7996_phy *phy);
 void mt7996_mac_work(struct work_struct *work);