]> www.infradead.org Git - nvme.git/commitdiff
wifi: mac80211: Support EHT 1024 aggregation size in TX
authorMeiChia Chiu <MeiChia.Chiu@mediatek.com>
Tue, 12 Nov 2024 08:38:46 +0000 (16:38 +0800)
committerJohannes Berg <johannes.berg@intel.com>
Tue, 12 Nov 2024 12:41:45 +0000 (13:41 +0100)
Support EHT 1024 aggregation size in TX

The 1024 agg size for RX is supported but not for TX.
This patch adds this support and refactors common parsing logics for
addbaext in both process_addba_resp and process_addba_req into a
function.

Reviewed-by: Shayne Chen <shayne.chen@mediatek.com>
Reviewed-by: Money Wang <money.wang@mediatek.com>
Co-developed-by: Peter Chiu <chui-hao.chiu@mediatek.com>
Signed-off-by: Peter Chiu <chui-hao.chiu@mediatek.com>
Signed-off-by: MeiChia Chiu <MeiChia.Chiu@mediatek.com>
Link: https://patch.msgid.link/20241112083846.32063-1-MeiChia.Chiu@mediatek.com
[pass elems/len instead of mgmt/len/is_req]
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
include/linux/ieee80211.h
net/mac80211/agg-rx.c
net/mac80211/agg-tx.c
net/mac80211/ht.c
net/mac80211/ieee80211_i.h

index 456bca45ff05286c0c29503839183b99f7b19891..05dedc45505ce191daa4083d61318fdaceb01aca 100644 (file)
@@ -1445,6 +1445,8 @@ struct ieee80211_mgmt {
                                        __le16 status;
                                        __le16 capab;
                                        __le16 timeout;
+                                       /* followed by BA Extension */
+                                       u8 variable[];
                                } __packed addba_resp;
                                struct{
                                        u8 action_code;
index fe7eab4b681b013187c76fdddc5ad6279cef5c00..f3fbe5a4395e5c69098c3d6dff1bf3f29e22644a 100644 (file)
@@ -170,28 +170,63 @@ static void sta_rx_agg_reorder_timer_expired(struct timer_list *t)
        rcu_read_unlock();
 }
 
-static void ieee80211_add_addbaext(struct ieee80211_sub_if_data *sdata,
-                                  struct sk_buff *skb,
-                                  const struct ieee80211_addba_ext_ie *req,
-                                  u16 buf_size)
+void ieee80211_add_addbaext(struct sk_buff *skb,
+                           const u8 req_addba_ext_data,
+                           u16 buf_size)
 {
-       struct ieee80211_addba_ext_ie *resp;
+       struct ieee80211_addba_ext_ie *addba_ext;
        u8 *pos;
 
        pos = skb_put_zero(skb, 2 + sizeof(struct ieee80211_addba_ext_ie));
        *pos++ = WLAN_EID_ADDBA_EXT;
        *pos++ = sizeof(struct ieee80211_addba_ext_ie);
-       resp = (struct ieee80211_addba_ext_ie *)pos;
-       resp->data = req->data & IEEE80211_ADDBA_EXT_NO_FRAG;
+       addba_ext = (struct ieee80211_addba_ext_ie *)pos;
 
-       resp->data |= u8_encode_bits(buf_size >> IEEE80211_ADDBA_EXT_BUF_SIZE_SHIFT,
-                                    IEEE80211_ADDBA_EXT_BUF_SIZE_MASK);
+       addba_ext->data = IEEE80211_ADDBA_EXT_NO_FRAG;
+       if (req_addba_ext_data)
+               addba_ext->data &= req_addba_ext_data;
+
+       addba_ext->data |=
+               u8_encode_bits(buf_size >> IEEE80211_ADDBA_EXT_BUF_SIZE_SHIFT,
+                              IEEE80211_ADDBA_EXT_BUF_SIZE_MASK);
+}
+
+u8 ieee80211_retrieve_addba_ext_data(struct sta_info *sta,
+                                    const void *elem_data, ssize_t elem_len,
+                                    u16 *buf_size)
+{
+       struct ieee802_11_elems *elems;
+       u8 buf_size_1k, data = 0;
+
+       if (!sta->sta.deflink.he_cap.has_he)
+               return 0;
+
+       if (elem_len <= 0)
+               return 0;
+
+       elems = ieee802_11_parse_elems(elem_data, elem_len, true, NULL);
+
+       if (elems && !elems->parse_error && elems->addba_ext_ie) {
+               data = elems->addba_ext_ie->data;
+
+               if (!sta->sta.deflink.eht_cap.has_eht || !buf_size)
+                       goto free;
+
+               buf_size_1k = u8_get_bits(elems->addba_ext_ie->data,
+                                         IEEE80211_ADDBA_EXT_BUF_SIZE_MASK);
+               *buf_size |= (u16)buf_size_1k <<
+                            IEEE80211_ADDBA_EXT_BUF_SIZE_SHIFT;
+       }
+free:
+       kfree(elems);
+
+       return data;
 }
 
 static void ieee80211_send_addba_resp(struct sta_info *sta, u8 *da, u16 tid,
                                      u8 dialog_token, u16 status, u16 policy,
                                      u16 buf_size, u16 timeout,
-                                     const struct ieee80211_addba_ext_ie *addbaext)
+                                     const u8 req_addba_ext_data)
 {
        struct ieee80211_sub_if_data *sdata = sta->sdata;
        struct ieee80211_local *local = sdata->local;
@@ -223,8 +258,8 @@ static void ieee80211_send_addba_resp(struct sta_info *sta, u8 *da, u16 tid,
        mgmt->u.action.u.addba_resp.timeout = cpu_to_le16(timeout);
        mgmt->u.action.u.addba_resp.status = cpu_to_le16(status);
 
-       if (sta->sta.deflink.he_cap.has_he && addbaext)
-               ieee80211_add_addbaext(sdata, skb, addbaext, buf_size);
+       if (sta->sta.deflink.he_cap.has_he)
+               ieee80211_add_addbaext(skb, req_addba_ext_data, buf_size);
 
        ieee80211_tx_skb(sdata, skb);
 }
@@ -233,7 +268,7 @@ void __ieee80211_start_rx_ba_session(struct sta_info *sta,
                                     u8 dialog_token, u16 timeout,
                                     u16 start_seq_num, u16 ba_policy, u16 tid,
                                     u16 buf_size, bool tx, bool auto_seq,
-                                    const struct ieee80211_addba_ext_ie *addbaext)
+                                    const u8 addba_ext_data)
 {
        struct ieee80211_local *local = sta->sdata->local;
        struct tid_ampdu_rx *tid_agg_rx;
@@ -419,7 +454,7 @@ end:
        if (tx)
                ieee80211_send_addba_resp(sta, sta->sta.addr, tid,
                                          dialog_token, status, 1, buf_size,
-                                         timeout, addbaext);
+                                         timeout, addba_ext_data);
 }
 
 void ieee80211_process_addba_request(struct ieee80211_local *local,
@@ -428,9 +463,7 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,
                                     size_t len)
 {
        u16 capab, tid, timeout, ba_policy, buf_size, start_seq_num;
-       struct ieee802_11_elems *elems = NULL;
-       u8 dialog_token;
-       int ies_len;
+       u8 dialog_token, addba_ext_data;
 
        /* extract session parameters from addba request frame */
        dialog_token = mgmt->u.action.u.addba_req.dialog_token;
@@ -443,28 +476,17 @@ void ieee80211_process_addba_request(struct ieee80211_local *local,
        tid = (capab & IEEE80211_ADDBA_PARAM_TID_MASK) >> 2;
        buf_size = (capab & IEEE80211_ADDBA_PARAM_BUF_SIZE_MASK) >> 6;
 
-       ies_len = len - offsetof(struct ieee80211_mgmt,
-                                u.action.u.addba_req.variable);
-       if (ies_len) {
-               elems = ieee802_11_parse_elems(mgmt->u.action.u.addba_req.variable,
-                                              ies_len, true, NULL);
-               if (!elems || elems->parse_error)
-                       goto free;
-       }
-
-       if (sta->sta.deflink.eht_cap.has_eht && elems && elems->addba_ext_ie) {
-               u8 buf_size_1k = u8_get_bits(elems->addba_ext_ie->data,
-                                            IEEE80211_ADDBA_EXT_BUF_SIZE_MASK);
-
-               buf_size |= buf_size_1k << IEEE80211_ADDBA_EXT_BUF_SIZE_SHIFT;
-       }
+       addba_ext_data =
+               ieee80211_retrieve_addba_ext_data(sta,
+                                                 mgmt->u.action.u.addba_req.variable,
+                                                 len -
+                                                 offsetof(typeof(*mgmt),
+                                                          u.action.u.addba_req.variable),
+                                                 &buf_size);
 
        __ieee80211_start_rx_ba_session(sta, dialog_token, timeout,
                                        start_seq_num, ba_policy, tid,
-                                       buf_size, true, false,
-                                       elems ? elems->addba_ext_ie : NULL);
-free:
-       kfree(elems);
+                                       buf_size, true, false, addba_ext_data);
 }
 
 void ieee80211_manage_rx_ba_offl(struct ieee80211_vif *vif,
index 04cb45cfb3107c498c2c81334e5119aa32e8afd9..61f2cac37728ce2bb48a881be33265a55ca71eeb 100644 (file)
  * complete.
  */
 
-static void ieee80211_send_addba_request(struct ieee80211_sub_if_data *sdata,
-                                        const u8 *da, u16 tid,
+static void ieee80211_send_addba_request(struct sta_info *sta, u16 tid,
                                         u8 dialog_token, u16 start_seq_num,
                                         u16 agg_size, u16 timeout)
 {
+       struct ieee80211_sub_if_data *sdata = sta->sdata;
        struct ieee80211_local *local = sdata->local;
        struct sk_buff *skb;
        struct ieee80211_mgmt *mgmt;
        u16 capab;
 
-       skb = dev_alloc_skb(sizeof(*mgmt) + local->hw.extra_tx_headroom);
-
+       skb = dev_alloc_skb(sizeof(*mgmt) +
+                           2 + sizeof(struct ieee80211_addba_ext_ie) +
+                           local->hw.extra_tx_headroom);
        if (!skb)
                return;
 
        skb_reserve(skb, local->hw.extra_tx_headroom);
-       mgmt = ieee80211_mgmt_ba(skb, da, sdata);
+       mgmt = ieee80211_mgmt_ba(skb, sta->sta.addr, sdata);
 
        skb_put(skb, 1 + sizeof(mgmt->u.action.u.addba_req));
 
@@ -93,6 +94,9 @@ static void ieee80211_send_addba_request(struct ieee80211_sub_if_data *sdata,
        mgmt->u.action.u.addba_req.start_seq_num =
                                        cpu_to_le16(start_seq_num << 4);
 
+       if (sta->sta.deflink.he_cap.has_he)
+               ieee80211_add_addbaext(skb, 0, agg_size);
+
        ieee80211_tx_skb_tid(sdata, skb, tid, -1);
 }
 
@@ -460,8 +464,11 @@ static void ieee80211_send_addba_with_timeout(struct sta_info *sta,
        sta->ampdu_mlme.addba_req_num[tid]++;
        spin_unlock_bh(&sta->lock);
 
-       if (sta->sta.deflink.he_cap.has_he) {
+       if (sta->sta.deflink.eht_cap.has_eht) {
                buf_size = local->hw.max_tx_aggregation_subframes;
+       } else if (sta->sta.deflink.he_cap.has_he) {
+               buf_size = min_t(u16, local->hw.max_tx_aggregation_subframes,
+                                IEEE80211_MAX_AMPDU_BUF_HE);
        } else {
                /*
                 * We really should use what the driver told us it will
@@ -473,9 +480,8 @@ static void ieee80211_send_addba_with_timeout(struct sta_info *sta,
        }
 
        /* send AddBA request */
-       ieee80211_send_addba_request(sdata, sta->sta.addr, tid,
-                                    tid_tx->dialog_token, tid_tx->ssn,
-                                    buf_size, tid_tx->timeout);
+       ieee80211_send_addba_request(sta, tid, tid_tx->dialog_token,
+                                    tid_tx->ssn, buf_size, tid_tx->timeout);
 
        WARN_ON(test_and_set_bit(HT_AGG_STATE_SENT_ADDBA, &tid_tx->state));
 }
@@ -970,6 +976,13 @@ void ieee80211_process_addba_resp(struct ieee80211_local *local,
        amsdu = capab & IEEE80211_ADDBA_PARAM_AMSDU_MASK;
        tid = u16_get_bits(capab, IEEE80211_ADDBA_PARAM_TID_MASK);
        buf_size = u16_get_bits(capab, IEEE80211_ADDBA_PARAM_BUF_SIZE_MASK);
+
+       ieee80211_retrieve_addba_ext_data(sta,
+                                         mgmt->u.action.u.addba_resp.variable,
+                                         len - offsetof(typeof(*mgmt),
+                                                        u.action.u.addba_resp.variable),
+                                         &buf_size);
+
        buf_size = min(buf_size, local->hw.max_tx_aggregation_subframes);
 
        txq = sta->sta.txq[tid];
index 1c2b7dd8976aa13d12bf12a0670713962f164453..32390d8a9d753e50b60873ef8da6a71ea957a137 100644 (file)
@@ -379,7 +379,7 @@ void ieee80211_ba_session_work(struct wiphy *wiphy, struct wiphy_work *work)
                                       sta->ampdu_mlme.tid_rx_manage_offl))
                        __ieee80211_start_rx_ba_session(sta, 0, 0, 0, 1, tid,
                                                        IEEE80211_MAX_AMPDU_BUF_HT,
-                                                       false, true, NULL);
+                                                       false, true, 0);
 
                if (test_and_clear_bit(tid + IEEE80211_NUM_TIDS,
                                       sta->ampdu_mlme.tid_rx_manage_offl))
index 7dcb46120abc4dffa0786c314cfb67598edd9076..752297bcde76c94391886517e253b87cacd66261 100644 (file)
@@ -2111,14 +2111,19 @@ int ieee80211_send_smps_action(struct ieee80211_sub_if_data *sdata,
                               const u8 *bssid, int link_id);
 bool ieee80211_smps_is_restrictive(enum ieee80211_smps_mode smps_mode_old,
                                   enum ieee80211_smps_mode smps_mode_new);
-
+void ieee80211_add_addbaext(struct sk_buff *skb,
+                           const u8 req_addba_ext_data,
+                           u16 buf_size);
+u8 ieee80211_retrieve_addba_ext_data(struct sta_info *sta,
+                                    const void *elem_data, ssize_t elem_len,
+                                    u16 *buf_size);
 void __ieee80211_stop_rx_ba_session(struct sta_info *sta, u16 tid,
                                    u16 initiator, u16 reason, bool stop);
 void __ieee80211_start_rx_ba_session(struct sta_info *sta,
                                     u8 dialog_token, u16 timeout,
                                     u16 start_seq_num, u16 ba_policy, u16 tid,
                                     u16 buf_size, bool tx, bool auto_seq,
-                                    const struct ieee80211_addba_ext_ie *addbaext);
+                                    const u8 addba_ext_data);
 void ieee80211_sta_tear_down_BA_sessions(struct sta_info *sta,
                                         enum ieee80211_agg_stop_reason reason);
 void ieee80211_process_delba(struct ieee80211_sub_if_data *sdata,