wifi: mac80211: move element parsing to a new file
authorJohannes Berg <johannes.berg@intel.com>
Mon, 29 Jan 2024 19:19:34 +0000 (20:19 +0100)
committerJohannes Berg <johannes.berg@intel.com>
Thu, 8 Feb 2024 14:00:43 +0000 (15:00 +0100)
This code got really big, move it to a new file.
Pure code move.

Link: https://msgid.link/20240129202041.7f27f7c895e4.I0adfc28bd656a4d44c2bf47966277eecf56cbaa0@changeid
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
net/mac80211/Makefile
net/mac80211/parse.c [new file with mode: 0644]
net/mac80211/util.c

index 4406b4f8f3b974c73b8064654916fbc181ef8754..a33884967f212b2365cd94db967350878d651848 100644 (file)
@@ -29,7 +29,7 @@ mac80211-y := \
        spectmgmt.o \
        tx.o \
        key.o \
-       util.o \
+       util.o parse.o \
        wme.o \
        chan.o \
        trace.o mlme.o \
diff --git a/net/mac80211/parse.c b/net/mac80211/parse.c
new file mode 100644 (file)
index 0000000..196a882
--- /dev/null
@@ -0,0 +1,926 @@
+// SPDX-License-Identifier: GPL-2.0-only
+/*
+ * Copyright 2002-2005, Instant802 Networks, Inc.
+ * Copyright 2005-2006, Devicescape Software, Inc.
+ * Copyright 2006-2007 Jiri Benc <jbenc@suse.cz>
+ * Copyright 2007      Johannes Berg <johannes@sipsolutions.net>
+ * Copyright 2013-2014  Intel Mobile Communications GmbH
+ * Copyright (C) 2015-2017     Intel Deutschland GmbH
+ * Copyright (C) 2018-2024 Intel Corporation
+ *
+ * element parsing for mac80211
+ */
+
+#include <net/mac80211.h>
+#include <linux/netdevice.h>
+#include <linux/export.h>
+#include <linux/types.h>
+#include <linux/slab.h>
+#include <linux/skbuff.h>
+#include <linux/etherdevice.h>
+#include <linux/if_arp.h>
+#include <linux/bitmap.h>
+#include <linux/crc32.h>
+#include <net/net_namespace.h>
+#include <net/cfg80211.h>
+#include <net/rtnetlink.h>
+#include <kunit/visibility.h>
+
+#include "ieee80211_i.h"
+#include "driver-ops.h"
+#include "rate.h"
+#include "mesh.h"
+#include "wme.h"
+#include "led.h"
+#include "wep.h"
+
+static void
+ieee80211_parse_extension_element(u32 *crc,
+                                 const struct element *elem,
+                                 struct ieee802_11_elems *elems,
+                                 struct ieee80211_elems_parse_params *params)
+{
+       const void *data = elem->data + 1;
+       bool calc_crc = false;
+       u8 len;
+
+       if (!elem->datalen)
+               return;
+
+       len = elem->datalen - 1;
+
+       switch (elem->data[0]) {
+       case WLAN_EID_EXT_HE_MU_EDCA:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               calc_crc = true;
+               if (len >= sizeof(*elems->mu_edca_param_set))
+                       elems->mu_edca_param_set = data;
+               break;
+       case WLAN_EID_EXT_HE_CAPABILITY:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               if (ieee80211_he_capa_size_ok(data, len)) {
+                       elems->he_cap = data;
+                       elems->he_cap_len = len;
+               }
+               break;
+       case WLAN_EID_EXT_HE_OPERATION:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               calc_crc = true;
+               if (len >= sizeof(*elems->he_operation) &&
+                   len >= ieee80211_he_oper_size(data) - 1)
+                       elems->he_operation = data;
+               break;
+       case WLAN_EID_EXT_UORA:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               if (len >= 1)
+                       elems->uora_element = data;
+               break;
+       case WLAN_EID_EXT_MAX_CHANNEL_SWITCH_TIME:
+               if (len == 3)
+                       elems->max_channel_switch_time = data;
+               break;
+       case WLAN_EID_EXT_MULTIPLE_BSSID_CONFIGURATION:
+               if (len >= sizeof(*elems->mbssid_config_ie))
+                       elems->mbssid_config_ie = data;
+               break;
+       case WLAN_EID_EXT_HE_SPR:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               if (len >= sizeof(*elems->he_spr) &&
+                   len >= ieee80211_he_spr_size(data))
+                       elems->he_spr = data;
+               break;
+       case WLAN_EID_EXT_HE_6GHZ_CAPA:
+               if (params->mode < IEEE80211_CONN_MODE_HE)
+                       break;
+               if (len >= sizeof(*elems->he_6ghz_capa))
+                       elems->he_6ghz_capa = data;
+               break;
+       case WLAN_EID_EXT_EHT_CAPABILITY:
+               if (params->mode < IEEE80211_CONN_MODE_EHT)
+                       break;
+               if (ieee80211_eht_capa_size_ok(elems->he_cap,
+                                              data, len,
+                                              params->from_ap)) {
+                       elems->eht_cap = data;
+                       elems->eht_cap_len = len;
+               }
+               break;
+       case WLAN_EID_EXT_EHT_OPERATION:
+               if (params->mode < IEEE80211_CONN_MODE_EHT)
+                       break;
+               if (ieee80211_eht_oper_size_ok(data, len))
+                       elems->eht_operation = data;
+               calc_crc = true;
+               break;
+       case WLAN_EID_EXT_EHT_MULTI_LINK:
+               if (params->mode < IEEE80211_CONN_MODE_EHT)
+                       break;
+               calc_crc = true;
+
+               if (ieee80211_mle_size_ok(data, len)) {
+                       const struct ieee80211_multi_link_elem *mle =
+                               (void *)data;
+
+                       switch (le16_get_bits(mle->control,
+                                             IEEE80211_ML_CONTROL_TYPE)) {
+                       case IEEE80211_ML_CONTROL_TYPE_BASIC:
+                               if (elems->ml_basic) {
+                                       elems->parse_error |=
+                                               IEEE80211_PARSE_ERR_DUP_NEST_ML_BASIC;
+                                       break;
+                               }
+                               elems->ml_basic_elem = (void *)elem;
+                               elems->ml_basic = data;
+                               elems->ml_basic_len = len;
+                               break;
+                       case IEEE80211_ML_CONTROL_TYPE_RECONF:
+                               elems->ml_reconf_elem = (void *)elem;
+                               elems->ml_reconf = data;
+                               elems->ml_reconf_len = len;
+                               break;
+                       default:
+                               break;
+                       }
+               }
+               break;
+       case WLAN_EID_EXT_BANDWIDTH_INDICATION:
+               if (params->mode < IEEE80211_CONN_MODE_EHT)
+                       break;
+               if (ieee80211_bandwidth_indication_size_ok(data, len))
+                       elems->bandwidth_indication = data;
+               calc_crc = true;
+               break;
+       case WLAN_EID_EXT_TID_TO_LINK_MAPPING:
+               if (params->mode < IEEE80211_CONN_MODE_EHT)
+                       break;
+               calc_crc = true;
+               if (ieee80211_tid_to_link_map_size_ok(data, len) &&
+                   elems->ttlm_num < ARRAY_SIZE(elems->ttlm)) {
+                       elems->ttlm[elems->ttlm_num] = (void *)data;
+                       elems->ttlm_num++;
+               }
+               break;
+       }
+
+       if (crc && calc_crc)
+               *crc = crc32_be(*crc, (void *)elem, elem->datalen + 2);
+}
+
+static u32
+_ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params,
+                            struct ieee802_11_elems *elems,
+                            const struct element *check_inherit)
+{
+       const struct element *elem;
+       bool calc_crc = params->filter != 0;
+       DECLARE_BITMAP(seen_elems, 256);
+       u32 crc = params->crc;
+
+       bitmap_zero(seen_elems, 256);
+
+       for_each_element(elem, params->start, params->len) {
+               const struct element *subelem;
+               u8 elem_parse_failed;
+               u8 id = elem->id;
+               u8 elen = elem->datalen;
+               const u8 *pos = elem->data;
+
+               if (check_inherit &&
+                   !cfg80211_is_element_inherited(elem,
+                                                  check_inherit))
+                       continue;
+
+               switch (id) {
+               case WLAN_EID_SSID:
+               case WLAN_EID_SUPP_RATES:
+               case WLAN_EID_FH_PARAMS:
+               case WLAN_EID_DS_PARAMS:
+               case WLAN_EID_CF_PARAMS:
+               case WLAN_EID_TIM:
+               case WLAN_EID_IBSS_PARAMS:
+               case WLAN_EID_CHALLENGE:
+               case WLAN_EID_RSN:
+               case WLAN_EID_ERP_INFO:
+               case WLAN_EID_EXT_SUPP_RATES:
+               case WLAN_EID_HT_CAPABILITY:
+               case WLAN_EID_HT_OPERATION:
+               case WLAN_EID_VHT_CAPABILITY:
+               case WLAN_EID_VHT_OPERATION:
+               case WLAN_EID_MESH_ID:
+               case WLAN_EID_MESH_CONFIG:
+               case WLAN_EID_PEER_MGMT:
+               case WLAN_EID_PREQ:
+               case WLAN_EID_PREP:
+               case WLAN_EID_PERR:
+               case WLAN_EID_RANN:
+               case WLAN_EID_CHANNEL_SWITCH:
+               case WLAN_EID_EXT_CHANSWITCH_ANN:
+               case WLAN_EID_COUNTRY:
+               case WLAN_EID_PWR_CONSTRAINT:
+               case WLAN_EID_TIMEOUT_INTERVAL:
+               case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
+               case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
+               case WLAN_EID_CHAN_SWITCH_PARAM:
+               case WLAN_EID_EXT_CAPABILITY:
+               case WLAN_EID_CHAN_SWITCH_TIMING:
+               case WLAN_EID_LINK_ID:
+               case WLAN_EID_BSS_MAX_IDLE_PERIOD:
+               case WLAN_EID_RSNX:
+               case WLAN_EID_S1G_BCN_COMPAT:
+               case WLAN_EID_S1G_CAPABILITIES:
+               case WLAN_EID_S1G_OPERATION:
+               case WLAN_EID_AID_RESPONSE:
+               case WLAN_EID_S1G_SHORT_BCN_INTERVAL:
+               /*
+                * not listing WLAN_EID_CHANNEL_SWITCH_WRAPPER -- it seems possible
+                * that if the content gets bigger it might be needed more than once
+                */
+                       if (test_bit(id, seen_elems)) {
+                               elems->parse_error |=
+                                       IEEE80211_PARSE_ERR_DUP_ELEM;
+                               continue;
+                       }
+                       break;
+               }
+
+               if (calc_crc && id < 64 && (params->filter & (1ULL << id)))
+                       crc = crc32_be(crc, pos - 2, elen + 2);
+
+               elem_parse_failed = 0;
+
+               switch (id) {
+               case WLAN_EID_LINK_ID:
+                       if (elen + 2 < sizeof(struct ieee80211_tdls_lnkie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->lnk_id = (void *)(pos - 2);
+                       break;
+               case WLAN_EID_CHAN_SWITCH_TIMING:
+                       if (elen < sizeof(struct ieee80211_ch_switch_timing)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->ch_sw_timing = (void *)pos;
+                       break;
+               case WLAN_EID_EXT_CAPABILITY:
+                       elems->ext_capab = pos;
+                       elems->ext_capab_len = elen;
+                       break;
+               case WLAN_EID_SSID:
+                       elems->ssid = pos;
+                       elems->ssid_len = elen;
+                       break;
+               case WLAN_EID_SUPP_RATES:
+                       elems->supp_rates = pos;
+                       elems->supp_rates_len = elen;
+                       break;
+               case WLAN_EID_DS_PARAMS:
+                       if (elen >= 1)
+                               elems->ds_params = pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_TIM:
+                       if (elen >= sizeof(struct ieee80211_tim_ie)) {
+                               elems->tim = (void *)pos;
+                               elems->tim_len = elen;
+                       } else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_VENDOR_SPECIFIC:
+                       if (elen >= 4 && pos[0] == 0x00 && pos[1] == 0x50 &&
+                           pos[2] == 0xf2) {
+                               /* Microsoft OUI (00:50:F2) */
+
+                               if (calc_crc)
+                                       crc = crc32_be(crc, pos - 2, elen + 2);
+
+                               if (elen >= 5 && pos[3] == 2) {
+                                       /* OUI Type 2 - WMM IE */
+                                       if (pos[4] == 0) {
+                                               elems->wmm_info = pos;
+                                               elems->wmm_info_len = elen;
+                                       } else if (pos[4] == 1) {
+                                               elems->wmm_param = pos;
+                                               elems->wmm_param_len = elen;
+                                       }
+                               }
+                       }
+                       break;
+               case WLAN_EID_RSN:
+                       elems->rsn = pos;
+                       elems->rsn_len = elen;
+                       break;
+               case WLAN_EID_ERP_INFO:
+                       if (elen >= 1)
+                               elems->erp_info = pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_EXT_SUPP_RATES:
+                       elems->ext_supp_rates = pos;
+                       elems->ext_supp_rates_len = elen;
+                       break;
+               case WLAN_EID_HT_CAPABILITY:
+                       if (params->mode < IEEE80211_CONN_MODE_HT)
+                               break;
+                       if (elen >= sizeof(struct ieee80211_ht_cap))
+                               elems->ht_cap_elem = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_HT_OPERATION:
+                       if (params->mode < IEEE80211_CONN_MODE_HT)
+                               break;
+                       if (elen >= sizeof(struct ieee80211_ht_operation))
+                               elems->ht_operation = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_VHT_CAPABILITY:
+                       if (params->mode < IEEE80211_CONN_MODE_VHT)
+                               break;
+                       if (elen >= sizeof(struct ieee80211_vht_cap))
+                               elems->vht_cap_elem = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_VHT_OPERATION:
+                       if (params->mode < IEEE80211_CONN_MODE_VHT)
+                               break;
+                       if (elen >= sizeof(struct ieee80211_vht_operation)) {
+                               elems->vht_operation = (void *)pos;
+                               if (calc_crc)
+                                       crc = crc32_be(crc, pos - 2, elen + 2);
+                               break;
+                       }
+                       elem_parse_failed =
+                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_OPMODE_NOTIF:
+                       if (params->mode < IEEE80211_CONN_MODE_VHT)
+                               break;
+                       if (elen > 0) {
+                               elems->opmode_notif = pos;
+                               if (calc_crc)
+                                       crc = crc32_be(crc, pos - 2, elen + 2);
+                               break;
+                       }
+                       elem_parse_failed =
+                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_MESH_ID:
+                       elems->mesh_id = pos;
+                       elems->mesh_id_len = elen;
+                       break;
+               case WLAN_EID_MESH_CONFIG:
+                       if (elen >= sizeof(struct ieee80211_meshconf_ie))
+                               elems->mesh_config = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_PEER_MGMT:
+                       elems->peering = pos;
+                       elems->peering_len = elen;
+                       break;
+               case WLAN_EID_MESH_AWAKE_WINDOW:
+                       if (elen >= 2)
+                               elems->awake_window = (void *)pos;
+                       break;
+               case WLAN_EID_PREQ:
+                       elems->preq = pos;
+                       elems->preq_len = elen;
+                       break;
+               case WLAN_EID_PREP:
+                       elems->prep = pos;
+                       elems->prep_len = elen;
+                       break;
+               case WLAN_EID_PERR:
+                       elems->perr = pos;
+                       elems->perr_len = elen;
+                       break;
+               case WLAN_EID_RANN:
+                       if (elen >= sizeof(struct ieee80211_rann_ie))
+                               elems->rann = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_CHANNEL_SWITCH:
+                       if (elen != sizeof(struct ieee80211_channel_sw_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->ch_switch_ie = (void *)pos;
+                       break;
+               case WLAN_EID_EXT_CHANSWITCH_ANN:
+                       if (elen != sizeof(struct ieee80211_ext_chansw_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->ext_chansw_ie = (void *)pos;
+                       break;
+               case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
+                       if (params->mode < IEEE80211_CONN_MODE_HT)
+                               break;
+                       if (elen != sizeof(struct ieee80211_sec_chan_offs_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->sec_chan_offs = (void *)pos;
+                       break;
+               case WLAN_EID_CHAN_SWITCH_PARAM:
+                       if (elen <
+                           sizeof(*elems->mesh_chansw_params_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->mesh_chansw_params_ie = (void *)pos;
+                       break;
+               case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
+                       if (params->mode < IEEE80211_CONN_MODE_VHT)
+                               break;
+
+                       if (!params->action) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
+                               break;
+                       }
+
+                       if (elen < sizeof(*elems->wide_bw_chansw_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->wide_bw_chansw_ie = (void *)pos;
+                       break;
+               case WLAN_EID_CHANNEL_SWITCH_WRAPPER:
+                       if (params->mode < IEEE80211_CONN_MODE_VHT)
+                               break;
+                       if (params->action) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
+                               break;
+                       }
+                       /*
+                        * This is a bit tricky, but as we only care about
+                        * a few elements, parse them out manually.
+                        */
+                       subelem = cfg80211_find_elem(WLAN_EID_WIDE_BW_CHANNEL_SWITCH,
+                                                    pos, elen);
+                       if (subelem) {
+                               if (subelem->datalen >= sizeof(*elems->wide_bw_chansw_ie))
+                                       elems->wide_bw_chansw_ie =
+                                               (void *)subelem->data;
+                               else
+                                       elem_parse_failed =
+                                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       }
+
+                       if (params->mode < IEEE80211_CONN_MODE_EHT)
+                               break;
+
+                       subelem = cfg80211_find_ext_elem(WLAN_EID_EXT_BANDWIDTH_INDICATION,
+                                                        pos, elen);
+                       if (subelem) {
+                               const void *edata = subelem->data + 1;
+                               u8 edatalen = subelem->datalen - 1;
+
+                               if (ieee80211_bandwidth_indication_size_ok(edata,
+                                                                          edatalen))
+                                       elems->bandwidth_indication = edata;
+                               else
+                                       elem_parse_failed =
+                                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       }
+                       break;
+               case WLAN_EID_COUNTRY:
+                       elems->country_elem = pos;
+                       elems->country_elem_len = elen;
+                       break;
+               case WLAN_EID_PWR_CONSTRAINT:
+                       if (elen != 1) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->pwr_constr_elem = pos;
+                       break;
+               case WLAN_EID_CISCO_VENDOR_SPECIFIC:
+                       /* Lots of different options exist, but we only care
+                        * about the Dynamic Transmit Power Control element.
+                        * First check for the Cisco OUI, then for the DTPC
+                        * tag (0x00).
+                        */
+                       if (elen < 4) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+
+                       if (pos[0] != 0x00 || pos[1] != 0x40 ||
+                           pos[2] != 0x96 || pos[3] != 0x00)
+                               break;
+
+                       if (elen != 6) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+
+                       if (calc_crc)
+                               crc = crc32_be(crc, pos - 2, elen + 2);
+
+                       elems->cisco_dtpc_elem = pos;
+                       break;
+               case WLAN_EID_ADDBA_EXT:
+                       if (elen < sizeof(struct ieee80211_addba_ext_ie)) {
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                               break;
+                       }
+                       elems->addba_ext_ie = (void *)pos;
+                       break;
+               case WLAN_EID_TIMEOUT_INTERVAL:
+                       if (elen >= sizeof(struct ieee80211_timeout_interval_ie))
+                               elems->timeout_int = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_BSS_MAX_IDLE_PERIOD:
+                       if (elen >= sizeof(*elems->max_idle_period_ie))
+                               elems->max_idle_period_ie = (void *)pos;
+                       break;
+               case WLAN_EID_RSNX:
+                       elems->rsnx = pos;
+                       elems->rsnx_len = elen;
+                       break;
+               case WLAN_EID_TX_POWER_ENVELOPE:
+                       if (elen < 1 ||
+                           elen > sizeof(struct ieee80211_tx_pwr_env))
+                               break;
+
+                       if (elems->tx_pwr_env_num >= ARRAY_SIZE(elems->tx_pwr_env))
+                               break;
+
+                       elems->tx_pwr_env[elems->tx_pwr_env_num] = (void *)pos;
+                       elems->tx_pwr_env_len[elems->tx_pwr_env_num] = elen;
+                       elems->tx_pwr_env_num++;
+                       break;
+               case WLAN_EID_EXTENSION:
+                       ieee80211_parse_extension_element(calc_crc ?
+                                                               &crc : NULL,
+                                                         elem, elems, params);
+                       break;
+               case WLAN_EID_S1G_CAPABILITIES:
+                       if (params->mode != IEEE80211_CONN_MODE_S1G)
+                               break;
+                       if (elen >= sizeof(*elems->s1g_capab))
+                               elems->s1g_capab = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_S1G_OPERATION:
+                       if (params->mode != IEEE80211_CONN_MODE_S1G)
+                               break;
+                       if (elen == sizeof(*elems->s1g_oper))
+                               elems->s1g_oper = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_S1G_BCN_COMPAT:
+                       if (params->mode != IEEE80211_CONN_MODE_S1G)
+                               break;
+                       if (elen == sizeof(*elems->s1g_bcn_compat))
+                               elems->s1g_bcn_compat = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               case WLAN_EID_AID_RESPONSE:
+                       if (params->mode != IEEE80211_CONN_MODE_S1G)
+                               break;
+                       if (elen == sizeof(struct ieee80211_aid_response_ie))
+                               elems->aid_resp = (void *)pos;
+                       else
+                               elem_parse_failed =
+                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
+                       break;
+               default:
+                       break;
+               }
+
+               if (elem_parse_failed)
+                       elems->parse_error |= elem_parse_failed;
+               else
+                       __set_bit(id, seen_elems);
+       }
+
+       if (!for_each_element_completed(elem, params->start, params->len))
+               elems->parse_error |= IEEE80211_PARSE_ERR_INVALID_END;
+
+       return crc;
+}
+
+static size_t ieee802_11_find_bssid_profile(const u8 *start, size_t len,
+                                           struct ieee802_11_elems *elems,
+                                           struct cfg80211_bss *bss,
+                                           u8 *nontransmitted_profile)
+{
+       const struct element *elem, *sub;
+       size_t profile_len = 0;
+       bool found = false;
+
+       if (!bss || !bss->transmitted_bss)
+               return profile_len;
+
+       for_each_element_id(elem, WLAN_EID_MULTIPLE_BSSID, start, len) {
+               if (elem->datalen < 2)
+                       continue;
+               if (elem->data[0] < 1 || elem->data[0] > 8)
+                       continue;
+
+               for_each_element(sub, elem->data + 1, elem->datalen - 1) {
+                       u8 new_bssid[ETH_ALEN];
+                       const u8 *index;
+
+                       if (sub->id != 0 || sub->datalen < 4) {
+                               /* not a valid BSS profile */
+                               continue;
+                       }
+
+                       if (sub->data[0] != WLAN_EID_NON_TX_BSSID_CAP ||
+                           sub->data[1] != 2) {
+                               /* The first element of the
+                                * Nontransmitted BSSID Profile is not
+                                * the Nontransmitted BSSID Capability
+                                * element.
+                                */
+                               continue;
+                       }
+
+                       memset(nontransmitted_profile, 0, len);
+                       profile_len = cfg80211_merge_profile(start, len,
+                                                            elem,
+                                                            sub,
+                                                            nontransmitted_profile,
+                                                            len);
+
+                       /* found a Nontransmitted BSSID Profile */
+                       index = cfg80211_find_ie(WLAN_EID_MULTI_BSSID_IDX,
+                                                nontransmitted_profile,
+                                                profile_len);
+                       if (!index || index[1] < 1 || index[2] == 0) {
+                               /* Invalid MBSSID Index element */
+                               continue;
+                       }
+
+                       cfg80211_gen_new_bssid(bss->transmitted_bss->bssid,
+                                              elem->data[0],
+                                              index[2],
+                                              new_bssid);
+                       if (ether_addr_equal(new_bssid, bss->bssid)) {
+                               found = true;
+                               elems->bssid_index_len = index[1];
+                               elems->bssid_index = (void *)&index[2];
+                               break;
+                       }
+               }
+       }
+
+       return found ? profile_len : 0;
+}
+
+static void ieee80211_mle_get_sta_prof(struct ieee802_11_elems *elems,
+                                      u8 link_id)
+{
+       const struct ieee80211_multi_link_elem *ml = elems->ml_basic;
+       ssize_t ml_len = elems->ml_basic_len;
+       const struct element *sub;
+
+       if (!ml || !ml_len)
+               return;
+
+       if (le16_get_bits(ml->control, IEEE80211_ML_CONTROL_TYPE) !=
+           IEEE80211_ML_CONTROL_TYPE_BASIC)
+               return;
+
+       for_each_mle_subelement(sub, (u8 *)ml, ml_len) {
+               struct ieee80211_mle_per_sta_profile *prof = (void *)sub->data;
+               ssize_t sta_prof_len;
+               u16 control;
+
+               if (sub->id != IEEE80211_MLE_SUBELEM_PER_STA_PROFILE)
+                       continue;
+
+               if (!ieee80211_mle_basic_sta_prof_size_ok(sub->data,
+                                                         sub->datalen))
+                       return;
+
+               control = le16_to_cpu(prof->control);
+
+               if (link_id != u16_get_bits(control,
+                                           IEEE80211_MLE_STA_CONTROL_LINK_ID))
+                       continue;
+
+               if (!(control & IEEE80211_MLE_STA_CONTROL_COMPLETE_PROFILE))
+                       return;
+
+               /* the sub element can be fragmented */
+               sta_prof_len =
+                       cfg80211_defragment_element(sub,
+                                                   (u8 *)ml, ml_len,
+                                                   elems->scratch_pos,
+                                                   elems->scratch +
+                                                       elems->scratch_len -
+                                                       elems->scratch_pos,
+                                                   IEEE80211_MLE_SUBELEM_FRAGMENT);
+
+               if (sta_prof_len < 0)
+                       return;
+
+               elems->prof = (void *)elems->scratch_pos;
+               elems->sta_prof_len = sta_prof_len;
+               elems->scratch_pos += sta_prof_len;
+
+               return;
+       }
+}
+
+static void ieee80211_mle_parse_link(struct ieee802_11_elems *elems,
+                                    struct ieee80211_elems_parse_params *params)
+{
+       struct ieee80211_mle_per_sta_profile *prof;
+       struct ieee80211_elems_parse_params sub = {
+               .mode = params->mode,
+               .action = params->action,
+               .from_ap = params->from_ap,
+               .link_id = -1,
+       };
+       ssize_t ml_len = elems->ml_basic_len;
+       const struct element *non_inherit = NULL;
+       const u8 *end;
+
+       if (params->link_id == -1)
+               return;
+
+       ml_len = cfg80211_defragment_element(elems->ml_basic_elem,
+                                            elems->ie_start,
+                                            elems->total_len,
+                                            elems->scratch_pos,
+                                            elems->scratch +
+                                               elems->scratch_len -
+                                               elems->scratch_pos,
+                                            WLAN_EID_FRAGMENT);
+
+       if (ml_len < 0)
+               return;
+
+       elems->ml_basic = (const void *)elems->scratch_pos;
+       elems->ml_basic_len = ml_len;
+
+       ieee80211_mle_get_sta_prof(elems, params->link_id);
+       prof = elems->prof;
+
+       if (!prof)
+               return;
+
+       /* check if we have the 4 bytes for the fixed part in assoc response */
+       if (elems->sta_prof_len < sizeof(*prof) + prof->sta_info_len - 1 + 4) {
+               elems->prof = NULL;
+               elems->sta_prof_len = 0;
+               return;
+       }
+
+       /*
+        * Skip the capability information and the status code that are expected
+        * as part of the station profile in association response frames. Note
+        * the -1 is because the 'sta_info_len' is accounted to as part of the
+        * per-STA profile, but not part of the 'u8 variable[]' portion.
+        */
+       sub.start = prof->variable + prof->sta_info_len - 1 + 4;
+       end = (const u8 *)prof + elems->sta_prof_len;
+       sub.len = end - sub.start;
+
+       non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
+                                            sub.start, sub.len);
+       _ieee802_11_parse_elems_full(&sub, elems, non_inherit);
+}
+
+struct ieee802_11_elems *
+ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params)
+{
+       struct ieee802_11_elems *elems;
+       const struct element *non_inherit = NULL;
+       u8 *nontransmitted_profile;
+       int nontransmitted_profile_len = 0;
+       size_t scratch_len = 3 * params->len;
+
+       elems = kzalloc(struct_size(elems, scratch, scratch_len), GFP_ATOMIC);
+       if (!elems)
+               return NULL;
+       elems->ie_start = params->start;
+       elems->total_len = params->len;
+       elems->scratch_len = scratch_len;
+       elems->scratch_pos = elems->scratch;
+
+       nontransmitted_profile = elems->scratch_pos;
+       nontransmitted_profile_len =
+               ieee802_11_find_bssid_profile(params->start, params->len,
+                                             elems, params->bss,
+                                             nontransmitted_profile);
+       elems->scratch_pos += nontransmitted_profile_len;
+       elems->scratch_len -= nontransmitted_profile_len;
+       non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
+                                            nontransmitted_profile,
+                                            nontransmitted_profile_len);
+
+       elems->crc = _ieee802_11_parse_elems_full(params, elems, non_inherit);
+
+       /* Override with nontransmitted profile, if found */
+       if (nontransmitted_profile_len) {
+               struct ieee80211_elems_parse_params sub = {
+                       .mode = params->mode,
+                       .start = nontransmitted_profile,
+                       .len = nontransmitted_profile_len,
+                       .action = params->action,
+                       .link_id = params->link_id,
+               };
+
+               _ieee802_11_parse_elems_full(&sub, elems, NULL);
+       }
+
+       ieee80211_mle_parse_link(elems, params);
+
+       if (elems->tim && !elems->parse_error) {
+               const struct ieee80211_tim_ie *tim_ie = elems->tim;
+
+               elems->dtim_period = tim_ie->dtim_period;
+               elems->dtim_count = tim_ie->dtim_count;
+       }
+
+       /* Override DTIM period and count if needed */
+       if (elems->bssid_index &&
+           elems->bssid_index_len >=
+           offsetofend(struct ieee80211_bssid_index, dtim_period))
+               elems->dtim_period = elems->bssid_index->dtim_period;
+
+       if (elems->bssid_index &&
+           elems->bssid_index_len >=
+           offsetofend(struct ieee80211_bssid_index, dtim_count))
+               elems->dtim_count = elems->bssid_index->dtim_count;
+
+       return elems;
+}
+EXPORT_SYMBOL_IF_KUNIT(ieee802_11_parse_elems_full);
+
+int ieee80211_parse_bitrates(enum nl80211_chan_width width,
+                            const struct ieee80211_supported_band *sband,
+                            const u8 *srates, int srates_len, u32 *rates)
+{
+       u32 rate_flags = ieee80211_chanwidth_rate_flags(width);
+       struct ieee80211_rate *br;
+       int brate, rate, i, j, count = 0;
+
+       *rates = 0;
+
+       for (i = 0; i < srates_len; i++) {
+               rate = srates[i] & 0x7f;
+
+               for (j = 0; j < sband->n_bitrates; j++) {
+                       br = &sband->bitrates[j];
+                       if ((rate_flags & br->flags) != rate_flags)
+                               continue;
+
+                       brate = DIV_ROUND_UP(br->bitrate, 5);
+                       if (brate == rate) {
+                               *rates |= BIT(j);
+                               count++;
+                               break;
+                       }
+               }
+       }
+       return count;
+}
index ea863d78061e64964335ba2ca46a5a01a38d13a2..c675ac59d2bb7d976fee7cb82fa32c7f76eaf660 100644 (file)
@@ -917,868 +917,6 @@ void ieee80211_queue_delayed_work(struct ieee80211_hw *hw,
 }
 EXPORT_SYMBOL(ieee80211_queue_delayed_work);
 
-static void
-ieee80211_parse_extension_element(u32 *crc,
-                                 const struct element *elem,
-                                 struct ieee802_11_elems *elems,
-                                 struct ieee80211_elems_parse_params *params)
-{
-       const void *data = elem->data + 1;
-       bool calc_crc = false;
-       u8 len;
-
-       if (!elem->datalen)
-               return;
-
-       len = elem->datalen - 1;
-
-       switch (elem->data[0]) {
-       case WLAN_EID_EXT_HE_MU_EDCA:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               calc_crc = true;
-               if (len >= sizeof(*elems->mu_edca_param_set))
-                       elems->mu_edca_param_set = data;
-               break;
-       case WLAN_EID_EXT_HE_CAPABILITY:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               if (ieee80211_he_capa_size_ok(data, len)) {
-                       elems->he_cap = data;
-                       elems->he_cap_len = len;
-               }
-               break;
-       case WLAN_EID_EXT_HE_OPERATION:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               calc_crc = true;
-               if (len >= sizeof(*elems->he_operation) &&
-                   len >= ieee80211_he_oper_size(data) - 1)
-                       elems->he_operation = data;
-               break;
-       case WLAN_EID_EXT_UORA:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               if (len >= 1)
-                       elems->uora_element = data;
-               break;
-       case WLAN_EID_EXT_MAX_CHANNEL_SWITCH_TIME:
-               if (len == 3)
-                       elems->max_channel_switch_time = data;
-               break;
-       case WLAN_EID_EXT_MULTIPLE_BSSID_CONFIGURATION:
-               if (len >= sizeof(*elems->mbssid_config_ie))
-                       elems->mbssid_config_ie = data;
-               break;
-       case WLAN_EID_EXT_HE_SPR:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               if (len >= sizeof(*elems->he_spr) &&
-                   len >= ieee80211_he_spr_size(data))
-                       elems->he_spr = data;
-               break;
-       case WLAN_EID_EXT_HE_6GHZ_CAPA:
-               if (params->mode < IEEE80211_CONN_MODE_HE)
-                       break;
-               if (len >= sizeof(*elems->he_6ghz_capa))
-                       elems->he_6ghz_capa = data;
-               break;
-       case WLAN_EID_EXT_EHT_CAPABILITY:
-               if (params->mode < IEEE80211_CONN_MODE_EHT)
-                       break;
-               if (ieee80211_eht_capa_size_ok(elems->he_cap,
-                                              data, len,
-                                              params->from_ap)) {
-                       elems->eht_cap = data;
-                       elems->eht_cap_len = len;
-               }
-               break;
-       case WLAN_EID_EXT_EHT_OPERATION:
-               if (params->mode < IEEE80211_CONN_MODE_EHT)
-                       break;
-               if (ieee80211_eht_oper_size_ok(data, len))
-                       elems->eht_operation = data;
-               calc_crc = true;
-               break;
-       case WLAN_EID_EXT_EHT_MULTI_LINK:
-               if (params->mode < IEEE80211_CONN_MODE_EHT)
-                       break;
-               calc_crc = true;
-
-               if (ieee80211_mle_size_ok(data, len)) {
-                       const struct ieee80211_multi_link_elem *mle =
-                               (void *)data;
-
-                       switch (le16_get_bits(mle->control,
-                                             IEEE80211_ML_CONTROL_TYPE)) {
-                       case IEEE80211_ML_CONTROL_TYPE_BASIC:
-                               if (elems->ml_basic) {
-                                       elems->parse_error |=
-                                               IEEE80211_PARSE_ERR_DUP_NEST_ML_BASIC;
-                                       break;
-                               }
-                               elems->ml_basic_elem = (void *)elem;
-                               elems->ml_basic = data;
-                               elems->ml_basic_len = len;
-                               break;
-                       case IEEE80211_ML_CONTROL_TYPE_RECONF:
-                               elems->ml_reconf_elem = (void *)elem;
-                               elems->ml_reconf = data;
-                               elems->ml_reconf_len = len;
-                               break;
-                       default:
-                               break;
-                       }
-               }
-               break;
-       case WLAN_EID_EXT_BANDWIDTH_INDICATION:
-               if (params->mode < IEEE80211_CONN_MODE_EHT)
-                       break;
-               if (ieee80211_bandwidth_indication_size_ok(data, len))
-                       elems->bandwidth_indication = data;
-               calc_crc = true;
-               break;
-       case WLAN_EID_EXT_TID_TO_LINK_MAPPING:
-               if (params->mode < IEEE80211_CONN_MODE_EHT)
-                       break;
-               calc_crc = true;
-               if (ieee80211_tid_to_link_map_size_ok(data, len) &&
-                   elems->ttlm_num < ARRAY_SIZE(elems->ttlm)) {
-                       elems->ttlm[elems->ttlm_num] = (void *)data;
-                       elems->ttlm_num++;
-               }
-               break;
-       }
-
-       if (crc && calc_crc)
-               *crc = crc32_be(*crc, (void *)elem, elem->datalen + 2);
-}
-
-static u32
-_ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params,
-                            struct ieee802_11_elems *elems,
-                            const struct element *check_inherit)
-{
-       const struct element *elem;
-       bool calc_crc = params->filter != 0;
-       DECLARE_BITMAP(seen_elems, 256);
-       u32 crc = params->crc;
-
-       bitmap_zero(seen_elems, 256);
-
-       for_each_element(elem, params->start, params->len) {
-               const struct element *subelem;
-               u8 elem_parse_failed;
-               u8 id = elem->id;
-               u8 elen = elem->datalen;
-               const u8 *pos = elem->data;
-
-               if (check_inherit &&
-                   !cfg80211_is_element_inherited(elem,
-                                                  check_inherit))
-                       continue;
-
-               switch (id) {
-               case WLAN_EID_SSID:
-               case WLAN_EID_SUPP_RATES:
-               case WLAN_EID_FH_PARAMS:
-               case WLAN_EID_DS_PARAMS:
-               case WLAN_EID_CF_PARAMS:
-               case WLAN_EID_TIM:
-               case WLAN_EID_IBSS_PARAMS:
-               case WLAN_EID_CHALLENGE:
-               case WLAN_EID_RSN:
-               case WLAN_EID_ERP_INFO:
-               case WLAN_EID_EXT_SUPP_RATES:
-               case WLAN_EID_HT_CAPABILITY:
-               case WLAN_EID_HT_OPERATION:
-               case WLAN_EID_VHT_CAPABILITY:
-               case WLAN_EID_VHT_OPERATION:
-               case WLAN_EID_MESH_ID:
-               case WLAN_EID_MESH_CONFIG:
-               case WLAN_EID_PEER_MGMT:
-               case WLAN_EID_PREQ:
-               case WLAN_EID_PREP:
-               case WLAN_EID_PERR:
-               case WLAN_EID_RANN:
-               case WLAN_EID_CHANNEL_SWITCH:
-               case WLAN_EID_EXT_CHANSWITCH_ANN:
-               case WLAN_EID_COUNTRY:
-               case WLAN_EID_PWR_CONSTRAINT:
-               case WLAN_EID_TIMEOUT_INTERVAL:
-               case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
-               case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
-               case WLAN_EID_CHAN_SWITCH_PARAM:
-               case WLAN_EID_EXT_CAPABILITY:
-               case WLAN_EID_CHAN_SWITCH_TIMING:
-               case WLAN_EID_LINK_ID:
-               case WLAN_EID_BSS_MAX_IDLE_PERIOD:
-               case WLAN_EID_RSNX:
-               case WLAN_EID_S1G_BCN_COMPAT:
-               case WLAN_EID_S1G_CAPABILITIES:
-               case WLAN_EID_S1G_OPERATION:
-               case WLAN_EID_AID_RESPONSE:
-               case WLAN_EID_S1G_SHORT_BCN_INTERVAL:
-               /*
-                * not listing WLAN_EID_CHANNEL_SWITCH_WRAPPER -- it seems possible
-                * that if the content gets bigger it might be needed more than once
-                */
-                       if (test_bit(id, seen_elems)) {
-                               elems->parse_error |=
-                                       IEEE80211_PARSE_ERR_DUP_ELEM;
-                               continue;
-                       }
-                       break;
-               }
-
-               if (calc_crc && id < 64 && (params->filter & (1ULL << id)))
-                       crc = crc32_be(crc, pos - 2, elen + 2);
-
-               elem_parse_failed = 0;
-
-               switch (id) {
-               case WLAN_EID_LINK_ID:
-                       if (elen + 2 < sizeof(struct ieee80211_tdls_lnkie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->lnk_id = (void *)(pos - 2);
-                       break;
-               case WLAN_EID_CHAN_SWITCH_TIMING:
-                       if (elen < sizeof(struct ieee80211_ch_switch_timing)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->ch_sw_timing = (void *)pos;
-                       break;
-               case WLAN_EID_EXT_CAPABILITY:
-                       elems->ext_capab = pos;
-                       elems->ext_capab_len = elen;
-                       break;
-               case WLAN_EID_SSID:
-                       elems->ssid = pos;
-                       elems->ssid_len = elen;
-                       break;
-               case WLAN_EID_SUPP_RATES:
-                       elems->supp_rates = pos;
-                       elems->supp_rates_len = elen;
-                       break;
-               case WLAN_EID_DS_PARAMS:
-                       if (elen >= 1)
-                               elems->ds_params = pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_TIM:
-                       if (elen >= sizeof(struct ieee80211_tim_ie)) {
-                               elems->tim = (void *)pos;
-                               elems->tim_len = elen;
-                       } else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_VENDOR_SPECIFIC:
-                       if (elen >= 4 && pos[0] == 0x00 && pos[1] == 0x50 &&
-                           pos[2] == 0xf2) {
-                               /* Microsoft OUI (00:50:F2) */
-
-                               if (calc_crc)
-                                       crc = crc32_be(crc, pos - 2, elen + 2);
-
-                               if (elen >= 5 && pos[3] == 2) {
-                                       /* OUI Type 2 - WMM IE */
-                                       if (pos[4] == 0) {
-                                               elems->wmm_info = pos;
-                                               elems->wmm_info_len = elen;
-                                       } else if (pos[4] == 1) {
-                                               elems->wmm_param = pos;
-                                               elems->wmm_param_len = elen;
-                                       }
-                               }
-                       }
-                       break;
-               case WLAN_EID_RSN:
-                       elems->rsn = pos;
-                       elems->rsn_len = elen;
-                       break;
-               case WLAN_EID_ERP_INFO:
-                       if (elen >= 1)
-                               elems->erp_info = pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_EXT_SUPP_RATES:
-                       elems->ext_supp_rates = pos;
-                       elems->ext_supp_rates_len = elen;
-                       break;
-               case WLAN_EID_HT_CAPABILITY:
-                       if (params->mode < IEEE80211_CONN_MODE_HT)
-                               break;
-                       if (elen >= sizeof(struct ieee80211_ht_cap))
-                               elems->ht_cap_elem = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_HT_OPERATION:
-                       if (params->mode < IEEE80211_CONN_MODE_HT)
-                               break;
-                       if (elen >= sizeof(struct ieee80211_ht_operation))
-                               elems->ht_operation = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_VHT_CAPABILITY:
-                       if (params->mode < IEEE80211_CONN_MODE_VHT)
-                               break;
-                       if (elen >= sizeof(struct ieee80211_vht_cap))
-                               elems->vht_cap_elem = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_VHT_OPERATION:
-                       if (params->mode < IEEE80211_CONN_MODE_VHT)
-                               break;
-                       if (elen >= sizeof(struct ieee80211_vht_operation)) {
-                               elems->vht_operation = (void *)pos;
-                               if (calc_crc)
-                                       crc = crc32_be(crc, pos - 2, elen + 2);
-                               break;
-                       }
-                       elem_parse_failed =
-                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_OPMODE_NOTIF:
-                       if (params->mode < IEEE80211_CONN_MODE_VHT)
-                               break;
-                       if (elen > 0) {
-                               elems->opmode_notif = pos;
-                               if (calc_crc)
-                                       crc = crc32_be(crc, pos - 2, elen + 2);
-                               break;
-                       }
-                       elem_parse_failed =
-                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_MESH_ID:
-                       elems->mesh_id = pos;
-                       elems->mesh_id_len = elen;
-                       break;
-               case WLAN_EID_MESH_CONFIG:
-                       if (elen >= sizeof(struct ieee80211_meshconf_ie))
-                               elems->mesh_config = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_PEER_MGMT:
-                       elems->peering = pos;
-                       elems->peering_len = elen;
-                       break;
-               case WLAN_EID_MESH_AWAKE_WINDOW:
-                       if (elen >= 2)
-                               elems->awake_window = (void *)pos;
-                       break;
-               case WLAN_EID_PREQ:
-                       elems->preq = pos;
-                       elems->preq_len = elen;
-                       break;
-               case WLAN_EID_PREP:
-                       elems->prep = pos;
-                       elems->prep_len = elen;
-                       break;
-               case WLAN_EID_PERR:
-                       elems->perr = pos;
-                       elems->perr_len = elen;
-                       break;
-               case WLAN_EID_RANN:
-                       if (elen >= sizeof(struct ieee80211_rann_ie))
-                               elems->rann = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_CHANNEL_SWITCH:
-                       if (elen != sizeof(struct ieee80211_channel_sw_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->ch_switch_ie = (void *)pos;
-                       break;
-               case WLAN_EID_EXT_CHANSWITCH_ANN:
-                       if (elen != sizeof(struct ieee80211_ext_chansw_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->ext_chansw_ie = (void *)pos;
-                       break;
-               case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
-                       if (params->mode < IEEE80211_CONN_MODE_HT)
-                               break;
-                       if (elen != sizeof(struct ieee80211_sec_chan_offs_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->sec_chan_offs = (void *)pos;
-                       break;
-               case WLAN_EID_CHAN_SWITCH_PARAM:
-                       if (elen <
-                           sizeof(*elems->mesh_chansw_params_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->mesh_chansw_params_ie = (void *)pos;
-                       break;
-               case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
-                       if (params->mode < IEEE80211_CONN_MODE_VHT)
-                               break;
-
-                       if (!params->action) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
-                               break;
-                       }
-
-                       if (elen < sizeof(*elems->wide_bw_chansw_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->wide_bw_chansw_ie = (void *)pos;
-                       break;
-               case WLAN_EID_CHANNEL_SWITCH_WRAPPER:
-                       if (params->mode < IEEE80211_CONN_MODE_VHT)
-                               break;
-                       if (params->action) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
-                               break;
-                       }
-                       /*
-                        * This is a bit tricky, but as we only care about
-                        * a few elements, parse them out manually.
-                        */
-                       subelem = cfg80211_find_elem(WLAN_EID_WIDE_BW_CHANNEL_SWITCH,
-                                                    pos, elen);
-                       if (subelem) {
-                               if (subelem->datalen >= sizeof(*elems->wide_bw_chansw_ie))
-                                       elems->wide_bw_chansw_ie =
-                                               (void *)subelem->data;
-                               else
-                                       elem_parse_failed =
-                                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       }
-
-                       if (params->mode < IEEE80211_CONN_MODE_EHT)
-                               break;
-
-                       subelem = cfg80211_find_ext_elem(WLAN_EID_EXT_BANDWIDTH_INDICATION,
-                                                        pos, elen);
-                       if (subelem) {
-                               const void *edata = subelem->data + 1;
-                               u8 edatalen = subelem->datalen - 1;
-
-                               if (ieee80211_bandwidth_indication_size_ok(edata,
-                                                                          edatalen))
-                                       elems->bandwidth_indication = edata;
-                               else
-                                       elem_parse_failed =
-                                               IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       }
-                       break;
-               case WLAN_EID_COUNTRY:
-                       elems->country_elem = pos;
-                       elems->country_elem_len = elen;
-                       break;
-               case WLAN_EID_PWR_CONSTRAINT:
-                       if (elen != 1) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->pwr_constr_elem = pos;
-                       break;
-               case WLAN_EID_CISCO_VENDOR_SPECIFIC:
-                       /* Lots of different options exist, but we only care
-                        * about the Dynamic Transmit Power Control element.
-                        * First check for the Cisco OUI, then for the DTPC
-                        * tag (0x00).
-                        */
-                       if (elen < 4) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-
-                       if (pos[0] != 0x00 || pos[1] != 0x40 ||
-                           pos[2] != 0x96 || pos[3] != 0x00)
-                               break;
-
-                       if (elen != 6) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-
-                       if (calc_crc)
-                               crc = crc32_be(crc, pos - 2, elen + 2);
-
-                       elems->cisco_dtpc_elem = pos;
-                       break;
-               case WLAN_EID_ADDBA_EXT:
-                       if (elen < sizeof(struct ieee80211_addba_ext_ie)) {
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                               break;
-                       }
-                       elems->addba_ext_ie = (void *)pos;
-                       break;
-               case WLAN_EID_TIMEOUT_INTERVAL:
-                       if (elen >= sizeof(struct ieee80211_timeout_interval_ie))
-                               elems->timeout_int = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_BSS_MAX_IDLE_PERIOD:
-                       if (elen >= sizeof(*elems->max_idle_period_ie))
-                               elems->max_idle_period_ie = (void *)pos;
-                       break;
-               case WLAN_EID_RSNX:
-                       elems->rsnx = pos;
-                       elems->rsnx_len = elen;
-                       break;
-               case WLAN_EID_TX_POWER_ENVELOPE:
-                       if (elen < 1 ||
-                           elen > sizeof(struct ieee80211_tx_pwr_env))
-                               break;
-
-                       if (elems->tx_pwr_env_num >= ARRAY_SIZE(elems->tx_pwr_env))
-                               break;
-
-                       elems->tx_pwr_env[elems->tx_pwr_env_num] = (void *)pos;
-                       elems->tx_pwr_env_len[elems->tx_pwr_env_num] = elen;
-                       elems->tx_pwr_env_num++;
-                       break;
-               case WLAN_EID_EXTENSION:
-                       ieee80211_parse_extension_element(calc_crc ?
-                                                               &crc : NULL,
-                                                         elem, elems, params);
-                       break;
-               case WLAN_EID_S1G_CAPABILITIES:
-                       if (params->mode != IEEE80211_CONN_MODE_S1G)
-                               break;
-                       if (elen >= sizeof(*elems->s1g_capab))
-                               elems->s1g_capab = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_S1G_OPERATION:
-                       if (params->mode != IEEE80211_CONN_MODE_S1G)
-                               break;
-                       if (elen == sizeof(*elems->s1g_oper))
-                               elems->s1g_oper = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_S1G_BCN_COMPAT:
-                       if (params->mode != IEEE80211_CONN_MODE_S1G)
-                               break;
-                       if (elen == sizeof(*elems->s1g_bcn_compat))
-                               elems->s1g_bcn_compat = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               case WLAN_EID_AID_RESPONSE:
-                       if (params->mode != IEEE80211_CONN_MODE_S1G)
-                               break;
-                       if (elen == sizeof(struct ieee80211_aid_response_ie))
-                               elems->aid_resp = (void *)pos;
-                       else
-                               elem_parse_failed =
-                                       IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
-                       break;
-               default:
-                       break;
-               }
-
-               if (elem_parse_failed)
-                       elems->parse_error |= elem_parse_failed;
-               else
-                       __set_bit(id, seen_elems);
-       }
-
-       if (!for_each_element_completed(elem, params->start, params->len))
-               elems->parse_error |= IEEE80211_PARSE_ERR_INVALID_END;
-
-       return crc;
-}
-
-static size_t ieee802_11_find_bssid_profile(const u8 *start, size_t len,
-                                           struct ieee802_11_elems *elems,
-                                           struct cfg80211_bss *bss,
-                                           u8 *nontransmitted_profile)
-{
-       const struct element *elem, *sub;
-       size_t profile_len = 0;
-       bool found = false;
-
-       if (!bss || !bss->transmitted_bss)
-               return profile_len;
-
-       for_each_element_id(elem, WLAN_EID_MULTIPLE_BSSID, start, len) {
-               if (elem->datalen < 2)
-                       continue;
-               if (elem->data[0] < 1 || elem->data[0] > 8)
-                       continue;
-
-               for_each_element(sub, elem->data + 1, elem->datalen - 1) {
-                       u8 new_bssid[ETH_ALEN];
-                       const u8 *index;
-
-                       if (sub->id != 0 || sub->datalen < 4) {
-                               /* not a valid BSS profile */
-                               continue;
-                       }
-
-                       if (sub->data[0] != WLAN_EID_NON_TX_BSSID_CAP ||
-                           sub->data[1] != 2) {
-                               /* The first element of the
-                                * Nontransmitted BSSID Profile is not
-                                * the Nontransmitted BSSID Capability
-                                * element.
-                                */
-                               continue;
-                       }
-
-                       memset(nontransmitted_profile, 0, len);
-                       profile_len = cfg80211_merge_profile(start, len,
-                                                            elem,
-                                                            sub,
-                                                            nontransmitted_profile,
-                                                            len);
-
-                       /* found a Nontransmitted BSSID Profile */
-                       index = cfg80211_find_ie(WLAN_EID_MULTI_BSSID_IDX,
-                                                nontransmitted_profile,
-                                                profile_len);
-                       if (!index || index[1] < 1 || index[2] == 0) {
-                               /* Invalid MBSSID Index element */
-                               continue;
-                       }
-
-                       cfg80211_gen_new_bssid(bss->transmitted_bss->bssid,
-                                              elem->data[0],
-                                              index[2],
-                                              new_bssid);
-                       if (ether_addr_equal(new_bssid, bss->bssid)) {
-                               found = true;
-                               elems->bssid_index_len = index[1];
-                               elems->bssid_index = (void *)&index[2];
-                               break;
-                       }
-               }
-       }
-
-       return found ? profile_len : 0;
-}
-
-static void ieee80211_mle_get_sta_prof(struct ieee802_11_elems *elems,
-                                      u8 link_id)
-{
-       const struct ieee80211_multi_link_elem *ml = elems->ml_basic;
-       ssize_t ml_len = elems->ml_basic_len;
-       const struct element *sub;
-
-       if (!ml || !ml_len)
-               return;
-
-       if (le16_get_bits(ml->control, IEEE80211_ML_CONTROL_TYPE) !=
-           IEEE80211_ML_CONTROL_TYPE_BASIC)
-               return;
-
-       for_each_mle_subelement(sub, (u8 *)ml, ml_len) {
-               struct ieee80211_mle_per_sta_profile *prof = (void *)sub->data;
-               ssize_t sta_prof_len;
-               u16 control;
-
-               if (sub->id != IEEE80211_MLE_SUBELEM_PER_STA_PROFILE)
-                       continue;
-
-               if (!ieee80211_mle_basic_sta_prof_size_ok(sub->data,
-                                                         sub->datalen))
-                       return;
-
-               control = le16_to_cpu(prof->control);
-
-               if (link_id != u16_get_bits(control,
-                                           IEEE80211_MLE_STA_CONTROL_LINK_ID))
-                       continue;
-
-               if (!(control & IEEE80211_MLE_STA_CONTROL_COMPLETE_PROFILE))
-                       return;
-
-               /* the sub element can be fragmented */
-               sta_prof_len =
-                       cfg80211_defragment_element(sub,
-                                                   (u8 *)ml, ml_len,
-                                                   elems->scratch_pos,
-                                                   elems->scratch +
-                                                       elems->scratch_len -
-                                                       elems->scratch_pos,
-                                                   IEEE80211_MLE_SUBELEM_FRAGMENT);
-
-               if (sta_prof_len < 0)
-                       return;
-
-               elems->prof = (void *)elems->scratch_pos;
-               elems->sta_prof_len = sta_prof_len;
-               elems->scratch_pos += sta_prof_len;
-
-               return;
-       }
-}
-
-static void ieee80211_mle_parse_link(struct ieee802_11_elems *elems,
-                                    struct ieee80211_elems_parse_params *params)
-{
-       struct ieee80211_mle_per_sta_profile *prof;
-       struct ieee80211_elems_parse_params sub = {
-               .mode = params->mode,
-               .action = params->action,
-               .from_ap = params->from_ap,
-               .link_id = -1,
-       };
-       ssize_t ml_len = elems->ml_basic_len;
-       const struct element *non_inherit = NULL;
-       const u8 *end;
-
-       if (params->link_id == -1)
-               return;
-
-       ml_len = cfg80211_defragment_element(elems->ml_basic_elem,
-                                            elems->ie_start,
-                                            elems->total_len,
-                                            elems->scratch_pos,
-                                            elems->scratch +
-                                               elems->scratch_len -
-                                               elems->scratch_pos,
-                                            WLAN_EID_FRAGMENT);
-
-       if (ml_len < 0)
-               return;
-
-       elems->ml_basic = (const void *)elems->scratch_pos;
-       elems->ml_basic_len = ml_len;
-
-       ieee80211_mle_get_sta_prof(elems, params->link_id);
-       prof = elems->prof;
-
-       if (!prof)
-               return;
-
-       /* check if we have the 4 bytes for the fixed part in assoc response */
-       if (elems->sta_prof_len < sizeof(*prof) + prof->sta_info_len - 1 + 4) {
-               elems->prof = NULL;
-               elems->sta_prof_len = 0;
-               return;
-       }
-
-       /*
-        * Skip the capability information and the status code that are expected
-        * as part of the station profile in association response frames. Note
-        * the -1 is because the 'sta_info_len' is accounted to as part of the
-        * per-STA profile, but not part of the 'u8 variable[]' portion.
-        */
-       sub.start = prof->variable + prof->sta_info_len - 1 + 4;
-       end = (const u8 *)prof + elems->sta_prof_len;
-       sub.len = end - sub.start;
-
-       non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
-                                            sub.start, sub.len);
-       _ieee802_11_parse_elems_full(&sub, elems, non_inherit);
-}
-
-struct ieee802_11_elems *
-ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params)
-{
-       struct ieee802_11_elems *elems;
-       const struct element *non_inherit = NULL;
-       u8 *nontransmitted_profile;
-       int nontransmitted_profile_len = 0;
-       size_t scratch_len = 3 * params->len;
-
-       elems = kzalloc(struct_size(elems, scratch, scratch_len), GFP_ATOMIC);
-       if (!elems)
-               return NULL;
-       elems->ie_start = params->start;
-       elems->total_len = params->len;
-       elems->scratch_len = scratch_len;
-       elems->scratch_pos = elems->scratch;
-
-       nontransmitted_profile = elems->scratch_pos;
-       nontransmitted_profile_len =
-               ieee802_11_find_bssid_profile(params->start, params->len,
-                                             elems, params->bss,
-                                             nontransmitted_profile);
-       elems->scratch_pos += nontransmitted_profile_len;
-       elems->scratch_len -= nontransmitted_profile_len;
-       non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
-                                            nontransmitted_profile,
-                                            nontransmitted_profile_len);
-
-       elems->crc = _ieee802_11_parse_elems_full(params, elems, non_inherit);
-
-       /* Override with nontransmitted profile, if found */
-       if (nontransmitted_profile_len) {
-               struct ieee80211_elems_parse_params sub = {
-                       .mode = params->mode,
-                       .start = nontransmitted_profile,
-                       .len = nontransmitted_profile_len,
-                       .action = params->action,
-                       .link_id = params->link_id,
-               };
-
-               _ieee802_11_parse_elems_full(&sub, elems, NULL);
-       }
-
-       ieee80211_mle_parse_link(elems, params);
-
-       if (elems->tim && !elems->parse_error) {
-               const struct ieee80211_tim_ie *tim_ie = elems->tim;
-
-               elems->dtim_period = tim_ie->dtim_period;
-               elems->dtim_count = tim_ie->dtim_count;
-       }
-
-       /* Override DTIM period and count if needed */
-       if (elems->bssid_index &&
-           elems->bssid_index_len >=
-           offsetofend(struct ieee80211_bssid_index, dtim_period))
-               elems->dtim_period = elems->bssid_index->dtim_period;
-
-       if (elems->bssid_index &&
-           elems->bssid_index_len >=
-           offsetofend(struct ieee80211_bssid_index, dtim_count))
-               elems->dtim_count = elems->bssid_index->dtim_count;
-
-       return elems;
-}
-EXPORT_SYMBOL_IF_KUNIT(ieee802_11_parse_elems_full);
-
 void ieee80211_regulatory_limit_wmm_params(struct ieee80211_sub_if_data *sdata,
                                           struct ieee80211_tx_queue_params
                                           *qparam, int ac)
@@ -4063,35 +3201,6 @@ bool ieee80211_chandef_s1g_oper(const struct ieee80211_s1g_oper_ie *oper,
        return true;
 }
 
-int ieee80211_parse_bitrates(enum nl80211_chan_width width,
-                            const struct ieee80211_supported_band *sband,
-                            const u8 *srates, int srates_len, u32 *rates)
-{
-       u32 rate_flags = ieee80211_chanwidth_rate_flags(width);
-       struct ieee80211_rate *br;
-       int brate, rate, i, j, count = 0;
-
-       *rates = 0;
-
-       for (i = 0; i < srates_len; i++) {
-               rate = srates[i] & 0x7f;
-
-               for (j = 0; j < sband->n_bitrates; j++) {
-                       br = &sband->bitrates[j];
-                       if ((rate_flags & br->flags) != rate_flags)
-                               continue;
-
-                       brate = DIV_ROUND_UP(br->bitrate, 5);
-                       if (brate == rate) {
-                               *rates |= BIT(j);
-                               count++;
-                               break;
-                       }
-               }
-       }
-       return count;
-}
-
 int ieee80211_put_srates_elem(struct sk_buff *skb,
                              const struct ieee80211_supported_band *sband,
                              u32 basic_rates, u32 rate_flags, u32 masked_rates,