Commit 6239da18 authored by Johannes Berg's avatar Johannes Berg

wifi: mac80211: adjust EHT capa when lowering bandwidth

If intending to associate with a lower bandwidth, remove capabilities
related to 320 MHz from the EHT capabilities element. Also change the
EHT MCS-NSS set accordingly: if just reducing 320->160 or similar the
format doesn't change, just cut off the last bytes. If changing from
higher bandwidth to 20 MHz only EHT STA, adjust the format.

Note that this also requires adjusting the caller in mlme.c since the
data written can now be shorter than it determined. We need to clean
all that up. Since the other callers pass NULL for the conn limit, we
don't need to change things there.

Link: https://msgid.link/20240129202041.b5f6df108c77.I0d8ea04079c61cb3744cc88625eeaf0d4776dc2b@changeidSigned-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent 90233160
...@@ -3060,6 +3060,9 @@ ieee80211_he_spr_size(const u8 *he_spr_ie) ...@@ -3060,6 +3060,9 @@ ieee80211_he_spr_size(const u8 *he_spr_ie)
#define IEEE80211_EHT_PHY_CAP5_SUPP_EXTRA_EHT_LTF 0x40 #define IEEE80211_EHT_PHY_CAP5_SUPP_EXTRA_EHT_LTF 0x40
#define IEEE80211_EHT_PHY_CAP6_MAX_NUM_SUPP_EHT_LTF_MASK 0x07 #define IEEE80211_EHT_PHY_CAP6_MAX_NUM_SUPP_EHT_LTF_MASK 0x07
#define IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_80MHZ 0x08
#define IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_160MHZ 0x30
#define IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_320MHZ 0x40
#define IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_MASK 0x78 #define IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_MASK 0x78
#define IEEE80211_EHT_PHY_CAP6_EHT_DUP_6GHZ_SUPP 0x80 #define IEEE80211_EHT_PHY_CAP6_EHT_DUP_6GHZ_SUPP 0x80
......
...@@ -2652,7 +2652,8 @@ void ieee80211_init_frag_cache(struct ieee80211_fragment_cache *cache); ...@@ -2652,7 +2652,8 @@ void ieee80211_init_frag_cache(struct ieee80211_fragment_cache *cache);
void ieee80211_destroy_frag_cache(struct ieee80211_fragment_cache *cache); void ieee80211_destroy_frag_cache(struct ieee80211_fragment_cache *cache);
u8 ieee80211_ie_len_eht_cap(struct ieee80211_sub_if_data *sdata); u8 ieee80211_ie_len_eht_cap(struct ieee80211_sub_if_data *sdata);
u8 *ieee80211_ie_build_eht_cap(u8 *pos, u8 *ieee80211_ie_build_eht_cap(const struct ieee80211_conn_settings *conn,
u8 *pos,
const struct ieee80211_sta_he_cap *he_cap, const struct ieee80211_sta_he_cap *he_cap,
const struct ieee80211_sta_eht_cap *eht_cap, const struct ieee80211_sta_eht_cap *eht_cap,
u8 *end, u8 *end,
......
...@@ -668,7 +668,8 @@ int mesh_add_eht_cap_ie(struct ieee80211_sub_if_data *sdata, ...@@ -668,7 +668,8 @@ int mesh_add_eht_cap_ie(struct ieee80211_sub_if_data *sdata,
return -ENOMEM; return -ENOMEM;
pos = skb_put(skb, ie_len); pos = skb_put(skb, ie_len);
ieee80211_ie_build_eht_cap(pos, he_cap, eht_cap, pos + ie_len, false); ieee80211_ie_build_eht_cap(NULL, pos, he_cap, eht_cap, pos + ie_len,
false);
return 0; return 0;
} }
......
...@@ -1072,9 +1072,10 @@ static void ieee80211_add_he_ie(struct ieee80211_sub_if_data *sdata, ...@@ -1072,9 +1072,10 @@ static void ieee80211_add_he_ie(struct ieee80211_sub_if_data *sdata,
static void ieee80211_add_eht_ie(struct ieee80211_sub_if_data *sdata, static void ieee80211_add_eht_ie(struct ieee80211_sub_if_data *sdata,
struct sk_buff *skb, struct sk_buff *skb,
struct ieee80211_supported_band *sband) struct ieee80211_supported_band *sband,
const struct ieee80211_conn_settings *conn)
{ {
u8 *pos; u8 *pos, *pre_eht_pos;
const struct ieee80211_sta_he_cap *he_cap; const struct ieee80211_sta_he_cap *he_cap;
const struct ieee80211_sta_eht_cap *eht_cap; const struct ieee80211_sta_eht_cap *eht_cap;
u8 eht_cap_size; u8 eht_cap_size;
...@@ -1097,8 +1098,11 @@ static void ieee80211_add_eht_ie(struct ieee80211_sub_if_data *sdata, ...@@ -1097,8 +1098,11 @@ static void ieee80211_add_eht_ie(struct ieee80211_sub_if_data *sdata,
ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0], ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0],
eht_cap->eht_cap_elem.phy_cap_info); eht_cap->eht_cap_elem.phy_cap_info);
pos = skb_put(skb, eht_cap_size); pos = skb_put(skb, eht_cap_size);
ieee80211_ie_build_eht_cap(pos, he_cap, eht_cap, pos + eht_cap_size, pre_eht_pos = pos;
false); pos = ieee80211_ie_build_eht_cap(conn, pos, he_cap, eht_cap,
pos + eht_cap_size, false);
/* trim excess if any */
skb_trim(skb, skb->len - (pre_eht_pos + eht_cap_size - pos));
} }
static void ieee80211_assoc_add_rates(struct sk_buff *skb, static void ieee80211_assoc_add_rates(struct sk_buff *skb,
...@@ -1453,7 +1457,8 @@ static size_t ieee80211_assoc_link_elems(struct ieee80211_sub_if_data *sdata, ...@@ -1453,7 +1457,8 @@ static size_t ieee80211_assoc_link_elems(struct ieee80211_sub_if_data *sdata,
present_elems = NULL; present_elems = NULL;
if (assoc_data->link[link_id].conn.mode >= IEEE80211_CONN_MODE_EHT) if (assoc_data->link[link_id].conn.mode >= IEEE80211_CONN_MODE_EHT)
ieee80211_add_eht_ie(sdata, skb, sband); ieee80211_add_eht_ie(sdata, skb, sband,
&assoc_data->link[link_id].conn);
if (sband->band == NL80211_BAND_S1GHZ) { if (sband->band == NL80211_BAND_S1GHZ) {
ieee80211_add_aid_request_ie(sdata, skb); ieee80211_add_aid_request_ie(sdata, skb);
......
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* Copyright 2014, Intel Corporation * Copyright 2014, Intel Corporation
* Copyright 2014 Intel Mobile Communications GmbH * Copyright 2014 Intel Mobile Communications GmbH
* Copyright 2015 - 2016 Intel Deutschland GmbH * Copyright 2015 - 2016 Intel Deutschland GmbH
* Copyright (C) 2019, 2021-2023 Intel Corporation * Copyright (C) 2019, 2021-2024 Intel Corporation
*/ */
#include <linux/ieee80211.h> #include <linux/ieee80211.h>
...@@ -604,7 +604,8 @@ ieee80211_tdls_add_setup_start_ies(struct ieee80211_link_data *link, ...@@ -604,7 +604,8 @@ ieee80211_tdls_add_setup_start_ies(struct ieee80211_link_data *link,
ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0], ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0],
eht_cap->eht_cap_elem.phy_cap_info); eht_cap->eht_cap_elem.phy_cap_info);
pos = skb_put(skb, cap_size); pos = skb_put(skb, cap_size);
ieee80211_ie_build_eht_cap(pos, he_cap, eht_cap, pos + cap_size, false); ieee80211_ie_build_eht_cap(NULL, pos, he_cap, eht_cap,
pos + cap_size, false);
} }
/* add any remaining IEs */ /* add any remaining IEs */
......
...@@ -2250,7 +2250,8 @@ static int ieee80211_build_preq_ies_band(struct ieee80211_sub_if_data *sdata, ...@@ -2250,7 +2250,8 @@ static int ieee80211_build_preq_ies_band(struct ieee80211_sub_if_data *sdata,
cfg80211_any_usable_channels(local->hw.wiphy, BIT(sband->band), cfg80211_any_usable_channels(local->hw.wiphy, BIT(sband->band),
IEEE80211_CHAN_NO_HE | IEEE80211_CHAN_NO_HE |
IEEE80211_CHAN_NO_EHT)) { IEEE80211_CHAN_NO_EHT)) {
pos = ieee80211_ie_build_eht_cap(pos, he_cap, eht_cap, end, pos = ieee80211_ie_build_eht_cap(NULL, pos, he_cap, eht_cap,
end,
sdata->vif.type == NL80211_IFTYPE_AP); sdata->vif.type == NL80211_IFTYPE_AP);
if (!pos) if (!pos)
goto out_err; goto out_err;
...@@ -3294,6 +3295,24 @@ u8 ieee80211_ie_len_he_cap(struct ieee80211_sub_if_data *sdata) ...@@ -3294,6 +3295,24 @@ u8 ieee80211_ie_len_he_cap(struct ieee80211_sub_if_data *sdata)
he_cap->he_cap_elem.phy_cap_info); he_cap->he_cap_elem.phy_cap_info);
} }
static void
ieee80211_get_adjusted_he_cap(const struct ieee80211_conn_settings *conn,
const struct ieee80211_sta_he_cap *he_cap,
struct ieee80211_he_cap_elem *elem)
{
*elem = he_cap->he_cap_elem;
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_40)
elem->phy_cap_info[0] &=
~(IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_80MHZ_IN_5G |
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_IN_2G);
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_160)
elem->phy_cap_info[0] &=
~(IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G |
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G);
}
u8 *ieee80211_ie_build_he_cap(const struct ieee80211_conn_settings *conn, u8 *ieee80211_ie_build_he_cap(const struct ieee80211_conn_settings *conn,
const struct ieee80211_sta_he_cap *he_cap, const struct ieee80211_sta_he_cap *he_cap,
u8 *pos, u8 *end) u8 *pos, u8 *end)
...@@ -3307,25 +3326,11 @@ u8 *ieee80211_ie_build_he_cap(const struct ieee80211_conn_settings *conn, ...@@ -3307,25 +3326,11 @@ u8 *ieee80211_ie_build_he_cap(const struct ieee80211_conn_settings *conn,
conn = &ieee80211_conn_settings_unlimited; conn = &ieee80211_conn_settings_unlimited;
/* Make sure we have place for the IE */ /* Make sure we have place for the IE */
/*
* TODO: the 1 added is because this temporarily is under the EXTENSION
* IE. Get rid of it when it moves.
*/
if (!he_cap) if (!he_cap)
return orig_pos; return orig_pos;
/* modify on stack first to calculate 'n' and 'ie_len' correctly */ /* modify on stack first to calculate 'n' and 'ie_len' correctly */
elem = he_cap->he_cap_elem; ieee80211_get_adjusted_he_cap(conn, he_cap, &elem);
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_40)
elem.phy_cap_info[0] &=
~(IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_80MHZ_IN_5G |
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_IN_2G);
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_160)
elem.phy_cap_info[0] &=
~(IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G |
IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G);
n = ieee80211_he_mcs_nss_size(&elem); n = ieee80211_he_mcs_nss_size(&elem);
ie_len = 2 + 1 + ie_len = 2 + 1 +
...@@ -5096,25 +5101,65 @@ u8 ieee80211_ie_len_eht_cap(struct ieee80211_sub_if_data *sdata) ...@@ -5096,25 +5101,65 @@ u8 ieee80211_ie_len_eht_cap(struct ieee80211_sub_if_data *sdata)
return 0; return 0;
} }
u8 *ieee80211_ie_build_eht_cap(u8 *pos, u8 *ieee80211_ie_build_eht_cap(const struct ieee80211_conn_settings *conn,
u8 *pos,
const struct ieee80211_sta_he_cap *he_cap, const struct ieee80211_sta_he_cap *he_cap,
const struct ieee80211_sta_eht_cap *eht_cap, const struct ieee80211_sta_eht_cap *eht_cap,
u8 *end, u8 *end, bool for_ap)
bool for_ap)
{ {
struct ieee80211_eht_cap_elem_fixed fixed, *out;
struct ieee80211_he_cap_elem he;
u8 mcs_nss_len, ppet_len; u8 mcs_nss_len, ppet_len;
u8 orig_mcs_nss_len;
u8 ie_len; u8 ie_len;
u8 *orig_pos = pos; u8 *orig_pos = pos;
if (!conn)
conn = &ieee80211_conn_settings_unlimited;
/* Make sure we have place for the IE */ /* Make sure we have place for the IE */
if (!he_cap || !eht_cap) if (!he_cap || !eht_cap)
return orig_pos; return orig_pos;
mcs_nss_len = ieee80211_eht_mcs_nss_size(&he_cap->he_cap_elem, orig_mcs_nss_len = ieee80211_eht_mcs_nss_size(&he_cap->he_cap_elem,
&eht_cap->eht_cap_elem, &eht_cap->eht_cap_elem,
for_ap); for_ap);
ieee80211_get_adjusted_he_cap(conn, he_cap, &he);
fixed = eht_cap->eht_cap_elem;
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_80)
fixed.phy_cap_info[6] &=
~IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_80MHZ;
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_160) {
fixed.phy_cap_info[1] &=
~IEEE80211_EHT_PHY_CAP1_BEAMFORMEE_SS_160MHZ_MASK;
fixed.phy_cap_info[2] &=
~IEEE80211_EHT_PHY_CAP2_SOUNDING_DIM_160MHZ_MASK;
fixed.phy_cap_info[6] &=
~IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_160MHZ;
}
if (conn->bw_limit < IEEE80211_CONN_BW_LIMIT_320) {
fixed.phy_cap_info[0] &=
~IEEE80211_EHT_PHY_CAP0_320MHZ_IN_6GHZ;
fixed.phy_cap_info[1] &=
~IEEE80211_EHT_PHY_CAP1_BEAMFORMEE_SS_320MHZ_MASK;
fixed.phy_cap_info[2] &=
~IEEE80211_EHT_PHY_CAP2_SOUNDING_DIM_320MHZ_MASK;
fixed.phy_cap_info[6] &=
~IEEE80211_EHT_PHY_CAP6_MCS15_SUPP_320MHZ;
}
if (conn->bw_limit == IEEE80211_CONN_BW_LIMIT_20)
fixed.phy_cap_info[0] &=
~IEEE80211_EHT_PHY_CAP0_242_TONE_RU_GT20MHZ;
mcs_nss_len = ieee80211_eht_mcs_nss_size(&he, &fixed, for_ap);
ppet_len = ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0], ppet_len = ieee80211_eht_ppe_size(eht_cap->eht_ppe_thres[0],
eht_cap->eht_cap_elem.phy_cap_info); fixed.phy_cap_info);
ie_len = 2 + 1 + sizeof(eht_cap->eht_cap_elem) + mcs_nss_len + ppet_len; ie_len = 2 + 1 + sizeof(eht_cap->eht_cap_elem) + mcs_nss_len + ppet_len;
if ((end - pos) < ie_len) if ((end - pos) < ie_len)
...@@ -5124,12 +5169,25 @@ u8 *ieee80211_ie_build_eht_cap(u8 *pos, ...@@ -5124,12 +5169,25 @@ u8 *ieee80211_ie_build_eht_cap(u8 *pos,
*pos++ = ie_len - 2; *pos++ = ie_len - 2;
*pos++ = WLAN_EID_EXT_EHT_CAPABILITY; *pos++ = WLAN_EID_EXT_EHT_CAPABILITY;
/* Fixed data */ out = (void *)pos;
memcpy(pos, &eht_cap->eht_cap_elem, sizeof(eht_cap->eht_cap_elem)); *out = fixed;
pos += sizeof(eht_cap->eht_cap_elem); pos += sizeof(*out);
if (mcs_nss_len == 4 && orig_mcs_nss_len != 4) {
/*
* If the (non-AP) STA became 20 MHz only, then convert from
* <=80 to 20-MHz-only format, where MCSes are indicated in
* the groups 0-7, 8-9, 10-11, 12-13 rather than just 0-9,
* 10-11, 12-13. Thus, use 0-9 for 0-7 and 8-9.
*/
*pos++ = eht_cap->eht_mcs_nss_supp.bw._80.rx_tx_mcs9_max_nss;
*pos++ = eht_cap->eht_mcs_nss_supp.bw._80.rx_tx_mcs9_max_nss;
*pos++ = eht_cap->eht_mcs_nss_supp.bw._80.rx_tx_mcs11_max_nss;
*pos++ = eht_cap->eht_mcs_nss_supp.bw._80.rx_tx_mcs13_max_nss;
} else {
memcpy(pos, &eht_cap->eht_mcs_nss_supp, mcs_nss_len); memcpy(pos, &eht_cap->eht_mcs_nss_supp, mcs_nss_len);
pos += mcs_nss_len; pos += mcs_nss_len;
}
if (ppet_len) { if (ppet_len) {
memcpy(pos, &eht_cap->eht_ppe_thres, ppet_len); memcpy(pos, &eht_cap->eht_ppe_thres, ppet_len);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment