mac80211: add HE 6 GHz Band Capability element
authorRajkumar Manoharan <rmanohar@codeaurora.org>
Thu, 28 May 2020 19:34:32 +0000 (21:34 +0200)
committerJohannes Berg <johannes.berg@intel.com>
Sun, 31 May 2020 09:26:39 +0000 (11:26 +0200)
Construct HE 6 GHz band capability element (IEEE 802.11ax/D6.0,
9.4.2.261) for association request and mesh beacon. The 6 GHz
capability information is passed by driver through iftypes caps.

Signed-off-by: Rajkumar Manoharan <rmanohar@codeaurora.org>
Link: https://lore.kernel.org/r/1589399105-25472-7-git-send-email-rmanohar@codeaurora.org
[handle SMPS, adjust for previous patches, reserve SKB space properly,
 change to handle SKB directly]
Link: https://lore.kernel.org/r/20200528213443.643aa8101111.I3f9747c1147480f65445f13eda5c4a5ed4e86757@changeid
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
net/mac80211/ieee80211_i.h
net/mac80211/mesh.c
net/mac80211/mesh.h
net/mac80211/mesh_plink.c
net/mac80211/mlme.c
net/mac80211/util.c

index dac016636d12ab346b50069d134becfa4a5508c6..344ea828e806302b7323e32a724f531a10c20237 100644 (file)
@@ -2177,6 +2177,8 @@ u8 ieee80211_ie_len_he_cap(struct ieee80211_sub_if_data *sdata, u8 iftype);
 u8 *ieee80211_ie_build_he_cap(u8 *pos,
                              const struct ieee80211_sta_he_cap *he_cap,
                              u8 *end);
+void ieee80211_ie_build_he_6ghz_cap(struct ieee80211_sub_if_data *sdata,
+                                   struct sk_buff *skb);
 u8 *ieee80211_ie_build_he_oper(u8 *pos);
 int ieee80211_parse_bitrates(struct cfg80211_chan_def *chandef,
                             const struct ieee80211_supported_band *sband,
index 5930d07b1e43d31533d515b06cecf413b151f90c..5e8d72bdbb98c3e6acb97d6ce577b39634538a47 100644 (file)
@@ -587,6 +587,13 @@ int mesh_add_he_oper_ie(struct ieee80211_sub_if_data *sdata,
        return 0;
 }
 
+int mesh_add_he_6ghz_cap_ie(struct ieee80211_sub_if_data *sdata,
+                           struct sk_buff *skb)
+{
+       ieee80211_ie_build_he_6ghz_cap(sdata, skb);
+       return 0;
+}
+
 static void ieee80211_mesh_path_timer(struct timer_list *t)
 {
        struct ieee80211_sub_if_data *sdata =
@@ -766,6 +773,7 @@ ieee80211_mesh_build_beacon(struct ieee80211_if_mesh *ifmsh)
                   2 + sizeof(struct ieee80211_vht_operation) +
                   ie_len_he_cap +
                   2 + 1 + sizeof(struct ieee80211_he_operation) +
+                  2 + 1 + sizeof(struct ieee80211_he_6ghz_capa) +
                   ifmsh->ie_len;
 
        bcn = kzalloc(sizeof(*bcn) + head_len + tail_len, GFP_KERNEL);
@@ -885,6 +893,7 @@ ieee80211_mesh_build_beacon(struct ieee80211_if_mesh *ifmsh)
            mesh_add_vht_oper_ie(sdata, skb) ||
            mesh_add_he_cap_ie(sdata, skb, ie_len_he_cap) ||
            mesh_add_he_oper_ie(sdata, skb) ||
+           mesh_add_he_6ghz_cap_ie(sdata, skb) ||
            mesh_add_vendor_ies(sdata, skb))
                goto out_free;
 
index 953f720754e82bfb16fdf0d535b753e211c88ad8..40492d1bd8fda6c4c3e19fc430d36c6a460a558f 100644 (file)
@@ -222,6 +222,8 @@ int mesh_add_he_cap_ie(struct ieee80211_sub_if_data *sdata,
                       struct sk_buff *skb, u8 ie_len);
 int mesh_add_he_oper_ie(struct ieee80211_sub_if_data *sdata,
                        struct sk_buff *skb);
+int mesh_add_he_6ghz_cap_ie(struct ieee80211_sub_if_data *sdata,
+                           struct sk_buff *skb);
 void mesh_rmc_free(struct ieee80211_sub_if_data *sdata);
 int mesh_rmc_init(struct ieee80211_sub_if_data *sdata);
 void ieee80211s_init(void);
index 737c5f4dbf5201b4c408059e1223be66880ac8f1..3aca89c97f36fd73f5fa524737f4efd6850dcc91 100644 (file)
@@ -238,6 +238,7 @@ static int mesh_plink_frame_tx(struct ieee80211_sub_if_data *sdata,
                            2 + sizeof(struct ieee80211_vht_operation) +
                            ie_len_he_cap +
                            2 + 1 + sizeof(struct ieee80211_he_operation) +
+                           2 + 1 + sizeof(struct ieee80211_he_6ghz_capa) +
                            2 + 8 + /* peering IE */
                            sdata->u.mesh.ie_len);
        if (!skb)
@@ -328,7 +329,8 @@ static int mesh_plink_frame_tx(struct ieee80211_sub_if_data *sdata,
                    mesh_add_vht_cap_ie(sdata, skb) ||
                    mesh_add_vht_oper_ie(sdata, skb) ||
                    mesh_add_he_cap_ie(sdata, skb, ie_len_he_cap) ||
-                   mesh_add_he_oper_ie(sdata, skb))
+                   mesh_add_he_oper_ie(sdata, skb) ||
+                   mesh_add_he_6ghz_cap_ie(sdata, skb))
                        goto free;
        }
 
index a259b4487b6001f02d6554a6b0ca4b8d176dbd51..f6ddce646f185d3053999528ded4baad493ee2f0 100644 (file)
@@ -658,6 +658,8 @@ static void ieee80211_add_he_ie(struct ieee80211_sub_if_data *sdata,
                                      he_cap->he_cap_elem.phy_cap_info);
        pos = skb_put(skb, he_cap_size);
        ieee80211_ie_build_he_cap(pos, he_cap, pos + he_cap_size);
+
+       ieee80211_ie_build_he_6ghz_cap(sdata, skb);
 }
 
 static void ieee80211_send_assoc(struct ieee80211_sub_if_data *sdata)
@@ -731,6 +733,7 @@ static void ieee80211_send_assoc(struct ieee80211_sub_if_data *sdata)
                        2 + 1 + sizeof(struct ieee80211_he_cap_elem) + /* HE */
                                sizeof(struct ieee80211_he_mcs_nss_supp) +
                                IEEE80211_HE_PPE_THRES_MAX_LEN +
+                       2 + 1 + sizeof(struct ieee80211_he_6ghz_capa) +
                        assoc_data->ie_len + /* extra IEs */
                        (assoc_data->fils_kek_len ? 16 /* AES-SIV */ : 0) +
                        9, /* WMM */
index 5d2c5ae8aadbf65cb384284c2d6e770b15f8a9bc..048b38546a568749c7325fdeaf8745c96f4eec65 100644 (file)
@@ -2839,6 +2839,52 @@ end:
        return pos;
 }
 
+void ieee80211_ie_build_he_6ghz_cap(struct ieee80211_sub_if_data *sdata,
+                                   struct sk_buff *skb)
+{
+       struct ieee80211_supported_band *sband;
+       const struct ieee80211_sband_iftype_data *iftd;
+       enum nl80211_iftype iftype = ieee80211_vif_type_p2p(&sdata->vif);
+       u8 *pos;
+       u16 cap;
+
+       sband = ieee80211_get_sband(sdata);
+       if (!sband)
+               return;
+
+       iftd = ieee80211_get_sband_iftype_data(sband, iftype);
+       if (WARN_ON(!iftd))
+               return;
+
+       cap = le16_to_cpu(iftd->he_6ghz_capa.capa);
+       cap &= ~IEEE80211_HE_6GHZ_CAP_SM_PS;
+
+       switch (sdata->smps_mode) {
+       case IEEE80211_SMPS_AUTOMATIC:
+       case IEEE80211_SMPS_NUM_MODES:
+               WARN_ON(1);
+               /* fall through */
+       case IEEE80211_SMPS_OFF:
+               cap |= u16_encode_bits(WLAN_HT_CAP_SM_PS_DISABLED,
+                                      IEEE80211_HE_6GHZ_CAP_SM_PS);
+               break;
+       case IEEE80211_SMPS_STATIC:
+               cap |= u16_encode_bits(WLAN_HT_CAP_SM_PS_STATIC,
+                                      IEEE80211_HE_6GHZ_CAP_SM_PS);
+               break;
+       case IEEE80211_SMPS_DYNAMIC:
+               cap |= u16_encode_bits(WLAN_HT_CAP_SM_PS_DYNAMIC,
+                                      IEEE80211_HE_6GHZ_CAP_SM_PS);
+               break;
+       }
+
+       pos = skb_put(skb, 2 + 1 + sizeof(cap));
+       *pos++ = WLAN_EID_EXTENSION;
+       *pos++ = 1 + sizeof(cap);
+       *pos++ = WLAN_EID_EXT_HE_6GHZ_CAPA;
+       put_unaligned_le16(cap, pos);
+}
+
 u8 *ieee80211_ie_build_ht_oper(u8 *pos, struct ieee80211_sta_ht_cap *ht_cap,
                               const struct cfg80211_chan_def *chandef,
                               u16 prot_mode, bool rifs_mode)