drm/tests: hdmi: Fix memory leaks in drm_display_mode_from_cea_vic()
[drm/drm-misc.git] / net / mac80211 / parse.c
blob279c5143b3356dcc03011f25d1de276d7876b766
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Copyright 2002-2005, Instant802 Networks, Inc.
4 * Copyright 2005-2006, Devicescape Software, Inc.
5 * Copyright 2006-2007 Jiri Benc <jbenc@suse.cz>
6 * Copyright 2007 Johannes Berg <johannes@sipsolutions.net>
7 * Copyright 2013-2014 Intel Mobile Communications GmbH
8 * Copyright (C) 2015-2017 Intel Deutschland GmbH
9 * Copyright (C) 2018-2024 Intel Corporation
11 * element parsing for mac80211
14 #include <net/mac80211.h>
15 #include <linux/netdevice.h>
16 #include <linux/export.h>
17 #include <linux/types.h>
18 #include <linux/slab.h>
19 #include <linux/skbuff.h>
20 #include <linux/etherdevice.h>
21 #include <linux/if_arp.h>
22 #include <linux/bitmap.h>
23 #include <linux/crc32.h>
24 #include <net/net_namespace.h>
25 #include <net/cfg80211.h>
26 #include <net/rtnetlink.h>
27 #include <kunit/visibility.h>
29 #include "ieee80211_i.h"
30 #include "driver-ops.h"
31 #include "rate.h"
32 #include "mesh.h"
33 #include "wme.h"
34 #include "led.h"
35 #include "wep.h"
37 struct ieee80211_elems_parse {
38 /* must be first for kfree to work */
39 struct ieee802_11_elems elems;
41 /* The basic Multi-Link element in the original elements */
42 const struct element *ml_basic_elem;
44 /* The reconfiguration Multi-Link element in the original elements */
45 const struct element *ml_reconf_elem;
48 * scratch buffer that can be used for various element parsing related
49 * tasks, e.g., element de-fragmentation etc.
51 size_t scratch_len;
52 u8 *scratch_pos;
53 u8 scratch[] __counted_by(scratch_len);
56 static void
57 ieee80211_parse_extension_element(u32 *crc,
58 const struct element *elem,
59 struct ieee80211_elems_parse *elems_parse,
60 struct ieee80211_elems_parse_params *params)
62 struct ieee802_11_elems *elems = &elems_parse->elems;
63 const void *data = elem->data + 1;
64 bool calc_crc = false;
65 u8 len;
67 if (!elem->datalen)
68 return;
70 len = elem->datalen - 1;
72 switch (elem->data[0]) {
73 case WLAN_EID_EXT_HE_MU_EDCA:
74 if (params->mode < IEEE80211_CONN_MODE_HE)
75 break;
76 calc_crc = true;
77 if (len >= sizeof(*elems->mu_edca_param_set))
78 elems->mu_edca_param_set = data;
79 break;
80 case WLAN_EID_EXT_HE_CAPABILITY:
81 if (params->mode < IEEE80211_CONN_MODE_HE)
82 break;
83 if (ieee80211_he_capa_size_ok(data, len)) {
84 elems->he_cap = data;
85 elems->he_cap_len = len;
87 break;
88 case WLAN_EID_EXT_HE_OPERATION:
89 if (params->mode < IEEE80211_CONN_MODE_HE)
90 break;
91 calc_crc = true;
92 if (len >= sizeof(*elems->he_operation) &&
93 len >= ieee80211_he_oper_size(data) - 1)
94 elems->he_operation = data;
95 break;
96 case WLAN_EID_EXT_UORA:
97 if (params->mode < IEEE80211_CONN_MODE_HE)
98 break;
99 if (len >= 1)
100 elems->uora_element = data;
101 break;
102 case WLAN_EID_EXT_MAX_CHANNEL_SWITCH_TIME:
103 if (len == 3)
104 elems->max_channel_switch_time = data;
105 break;
106 case WLAN_EID_EXT_MULTIPLE_BSSID_CONFIGURATION:
107 if (len >= sizeof(*elems->mbssid_config_ie))
108 elems->mbssid_config_ie = data;
109 break;
110 case WLAN_EID_EXT_HE_SPR:
111 if (params->mode < IEEE80211_CONN_MODE_HE)
112 break;
113 if (len >= sizeof(*elems->he_spr) &&
114 len >= ieee80211_he_spr_size(data) - 1)
115 elems->he_spr = data;
116 break;
117 case WLAN_EID_EXT_HE_6GHZ_CAPA:
118 if (params->mode < IEEE80211_CONN_MODE_HE)
119 break;
120 if (len >= sizeof(*elems->he_6ghz_capa))
121 elems->he_6ghz_capa = data;
122 break;
123 case WLAN_EID_EXT_EHT_CAPABILITY:
124 if (params->mode < IEEE80211_CONN_MODE_EHT)
125 break;
126 if (ieee80211_eht_capa_size_ok(elems->he_cap,
127 data, len,
128 params->from_ap)) {
129 elems->eht_cap = data;
130 elems->eht_cap_len = len;
132 break;
133 case WLAN_EID_EXT_EHT_OPERATION:
134 if (params->mode < IEEE80211_CONN_MODE_EHT)
135 break;
136 if (ieee80211_eht_oper_size_ok(data, len))
137 elems->eht_operation = data;
138 calc_crc = true;
139 break;
140 case WLAN_EID_EXT_EHT_MULTI_LINK:
141 if (params->mode < IEEE80211_CONN_MODE_EHT)
142 break;
143 calc_crc = true;
145 if (ieee80211_mle_size_ok(data, len)) {
146 const struct ieee80211_multi_link_elem *mle =
147 (void *)data;
149 switch (le16_get_bits(mle->control,
150 IEEE80211_ML_CONTROL_TYPE)) {
151 case IEEE80211_ML_CONTROL_TYPE_BASIC:
152 if (elems_parse->ml_basic_elem) {
153 elems->parse_error |=
154 IEEE80211_PARSE_ERR_DUP_NEST_ML_BASIC;
155 break;
157 elems_parse->ml_basic_elem = elem;
158 break;
159 case IEEE80211_ML_CONTROL_TYPE_RECONF:
160 elems_parse->ml_reconf_elem = elem;
161 break;
162 default:
163 break;
166 break;
167 case WLAN_EID_EXT_BANDWIDTH_INDICATION:
168 if (params->mode < IEEE80211_CONN_MODE_EHT)
169 break;
170 if (ieee80211_bandwidth_indication_size_ok(data, len))
171 elems->bandwidth_indication = data;
172 calc_crc = true;
173 break;
174 case WLAN_EID_EXT_TID_TO_LINK_MAPPING:
175 if (params->mode < IEEE80211_CONN_MODE_EHT)
176 break;
177 calc_crc = true;
178 if (ieee80211_tid_to_link_map_size_ok(data, len) &&
179 elems->ttlm_num < ARRAY_SIZE(elems->ttlm)) {
180 elems->ttlm[elems->ttlm_num] = (void *)data;
181 elems->ttlm_num++;
183 break;
186 if (crc && calc_crc)
187 *crc = crc32_be(*crc, (void *)elem, elem->datalen + 2);
190 static void ieee80211_parse_tpe(struct ieee80211_parsed_tpe *tpe,
191 const u8 *data, u8 len)
193 const struct ieee80211_tx_pwr_env *env = (const void *)data;
194 u8 count, interpret, category;
195 u8 *out, N, *cnt_out = NULL, *N_out = NULL;
197 if (!ieee80211_valid_tpe_element(data, len))
198 return;
200 count = u8_get_bits(env->info, IEEE80211_TX_PWR_ENV_INFO_COUNT);
201 interpret = u8_get_bits(env->info, IEEE80211_TX_PWR_ENV_INFO_INTERPRET);
202 category = u8_get_bits(env->info, IEEE80211_TX_PWR_ENV_INFO_CATEGORY);
204 switch (interpret) {
205 case IEEE80211_TPE_LOCAL_EIRP:
206 out = tpe->max_local[category].power;
207 cnt_out = &tpe->max_local[category].count;
208 tpe->max_local[category].valid = true;
209 break;
210 case IEEE80211_TPE_REG_CLIENT_EIRP:
211 out = tpe->max_reg_client[category].power;
212 cnt_out = &tpe->max_reg_client[category].count;
213 tpe->max_reg_client[category].valid = true;
214 break;
215 case IEEE80211_TPE_LOCAL_EIRP_PSD:
216 out = tpe->psd_local[category].power;
217 cnt_out = &tpe->psd_local[category].count;
218 N_out = &tpe->psd_local[category].n;
219 tpe->psd_local[category].valid = true;
220 break;
221 case IEEE80211_TPE_REG_CLIENT_EIRP_PSD:
222 out = tpe->psd_reg_client[category].power;
223 cnt_out = &tpe->psd_reg_client[category].count;
224 N_out = &tpe->psd_reg_client[category].n;
225 tpe->psd_reg_client[category].valid = true;
226 break;
229 switch (interpret) {
230 case IEEE80211_TPE_LOCAL_EIRP:
231 case IEEE80211_TPE_REG_CLIENT_EIRP:
232 /* count was validated <= 3, plus 320 MHz */
233 BUILD_BUG_ON(IEEE80211_TPE_EIRP_ENTRIES_320MHZ < 5);
234 memcpy(out, env->variable, count + 1);
235 *cnt_out = count + 1;
236 /* separately take 320 MHz if present */
237 if (count == 3 && len > sizeof(*env) + count + 1) {
238 out[4] = env->variable[4];
239 *cnt_out = 5;
241 break;
242 case IEEE80211_TPE_LOCAL_EIRP_PSD:
243 case IEEE80211_TPE_REG_CLIENT_EIRP_PSD:
244 if (!count) {
245 memset(out, env->variable[0],
246 IEEE80211_TPE_PSD_ENTRIES_320MHZ);
247 *cnt_out = IEEE80211_TPE_PSD_ENTRIES_320MHZ;
248 break;
251 N = 1 << (count - 1);
252 memcpy(out, env->variable, N);
253 *cnt_out = N;
254 *N_out = N;
256 if (len > sizeof(*env) + N) {
257 int K = u8_get_bits(env->variable[N],
258 IEEE80211_TX_PWR_ENV_EXT_COUNT);
260 K = min(K, IEEE80211_TPE_PSD_ENTRIES_320MHZ - N);
261 memcpy(out + N, env->variable + N + 1, K);
262 (*cnt_out) += K;
264 break;
268 static u32
269 _ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params,
270 struct ieee80211_elems_parse *elems_parse,
271 const struct element *check_inherit)
273 struct ieee802_11_elems *elems = &elems_parse->elems;
274 const struct element *elem;
275 bool calc_crc = params->filter != 0;
276 DECLARE_BITMAP(seen_elems, 256);
277 u32 crc = params->crc;
279 bitmap_zero(seen_elems, 256);
281 for_each_element(elem, params->start, params->len) {
282 const struct element *subelem;
283 u8 elem_parse_failed;
284 u8 id = elem->id;
285 u8 elen = elem->datalen;
286 const u8 *pos = elem->data;
288 if (check_inherit &&
289 !cfg80211_is_element_inherited(elem,
290 check_inherit))
291 continue;
293 switch (id) {
294 case WLAN_EID_SSID:
295 case WLAN_EID_SUPP_RATES:
296 case WLAN_EID_FH_PARAMS:
297 case WLAN_EID_DS_PARAMS:
298 case WLAN_EID_CF_PARAMS:
299 case WLAN_EID_TIM:
300 case WLAN_EID_IBSS_PARAMS:
301 case WLAN_EID_CHALLENGE:
302 case WLAN_EID_RSN:
303 case WLAN_EID_ERP_INFO:
304 case WLAN_EID_EXT_SUPP_RATES:
305 case WLAN_EID_HT_CAPABILITY:
306 case WLAN_EID_HT_OPERATION:
307 case WLAN_EID_VHT_CAPABILITY:
308 case WLAN_EID_VHT_OPERATION:
309 case WLAN_EID_MESH_ID:
310 case WLAN_EID_MESH_CONFIG:
311 case WLAN_EID_PEER_MGMT:
312 case WLAN_EID_PREQ:
313 case WLAN_EID_PREP:
314 case WLAN_EID_PERR:
315 case WLAN_EID_RANN:
316 case WLAN_EID_CHANNEL_SWITCH:
317 case WLAN_EID_EXT_CHANSWITCH_ANN:
318 case WLAN_EID_COUNTRY:
319 case WLAN_EID_PWR_CONSTRAINT:
320 case WLAN_EID_TIMEOUT_INTERVAL:
321 case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
322 case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
323 case WLAN_EID_CHAN_SWITCH_PARAM:
324 case WLAN_EID_EXT_CAPABILITY:
325 case WLAN_EID_CHAN_SWITCH_TIMING:
326 case WLAN_EID_LINK_ID:
327 case WLAN_EID_BSS_MAX_IDLE_PERIOD:
328 case WLAN_EID_RSNX:
329 case WLAN_EID_S1G_BCN_COMPAT:
330 case WLAN_EID_S1G_CAPABILITIES:
331 case WLAN_EID_S1G_OPERATION:
332 case WLAN_EID_AID_RESPONSE:
333 case WLAN_EID_S1G_SHORT_BCN_INTERVAL:
335 * not listing WLAN_EID_CHANNEL_SWITCH_WRAPPER -- it seems possible
336 * that if the content gets bigger it might be needed more than once
338 if (test_bit(id, seen_elems)) {
339 elems->parse_error |=
340 IEEE80211_PARSE_ERR_DUP_ELEM;
341 continue;
343 break;
346 if (calc_crc && id < 64 && (params->filter & (1ULL << id)))
347 crc = crc32_be(crc, pos - 2, elen + 2);
349 elem_parse_failed = 0;
351 switch (id) {
352 case WLAN_EID_LINK_ID:
353 if (elen + 2 < sizeof(struct ieee80211_tdls_lnkie)) {
354 elem_parse_failed =
355 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
356 break;
358 elems->lnk_id = (void *)(pos - 2);
359 break;
360 case WLAN_EID_CHAN_SWITCH_TIMING:
361 if (elen < sizeof(struct ieee80211_ch_switch_timing)) {
362 elem_parse_failed =
363 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
364 break;
366 elems->ch_sw_timing = (void *)pos;
367 break;
368 case WLAN_EID_EXT_CAPABILITY:
369 elems->ext_capab = pos;
370 elems->ext_capab_len = elen;
371 break;
372 case WLAN_EID_SSID:
373 elems->ssid = pos;
374 elems->ssid_len = elen;
375 break;
376 case WLAN_EID_SUPP_RATES:
377 elems->supp_rates = pos;
378 elems->supp_rates_len = elen;
379 break;
380 case WLAN_EID_DS_PARAMS:
381 if (elen >= 1)
382 elems->ds_params = pos;
383 else
384 elem_parse_failed =
385 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
386 break;
387 case WLAN_EID_TIM:
388 if (elen >= sizeof(struct ieee80211_tim_ie)) {
389 elems->tim = (void *)pos;
390 elems->tim_len = elen;
391 } else
392 elem_parse_failed =
393 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
394 break;
395 case WLAN_EID_VENDOR_SPECIFIC:
396 if (elen >= 4 && pos[0] == 0x00 && pos[1] == 0x50 &&
397 pos[2] == 0xf2) {
398 /* Microsoft OUI (00:50:F2) */
400 if (calc_crc)
401 crc = crc32_be(crc, pos - 2, elen + 2);
403 if (elen >= 5 && pos[3] == 2) {
404 /* OUI Type 2 - WMM IE */
405 if (pos[4] == 0) {
406 elems->wmm_info = pos;
407 elems->wmm_info_len = elen;
408 } else if (pos[4] == 1) {
409 elems->wmm_param = pos;
410 elems->wmm_param_len = elen;
414 break;
415 case WLAN_EID_RSN:
416 elems->rsn = pos;
417 elems->rsn_len = elen;
418 break;
419 case WLAN_EID_ERP_INFO:
420 if (elen >= 1)
421 elems->erp_info = pos;
422 else
423 elem_parse_failed =
424 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
425 break;
426 case WLAN_EID_EXT_SUPP_RATES:
427 elems->ext_supp_rates = pos;
428 elems->ext_supp_rates_len = elen;
429 break;
430 case WLAN_EID_HT_CAPABILITY:
431 if (params->mode < IEEE80211_CONN_MODE_HT)
432 break;
433 if (elen >= sizeof(struct ieee80211_ht_cap))
434 elems->ht_cap_elem = (void *)pos;
435 else
436 elem_parse_failed =
437 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
438 break;
439 case WLAN_EID_HT_OPERATION:
440 if (params->mode < IEEE80211_CONN_MODE_HT)
441 break;
442 if (elen >= sizeof(struct ieee80211_ht_operation))
443 elems->ht_operation = (void *)pos;
444 else
445 elem_parse_failed =
446 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
447 break;
448 case WLAN_EID_VHT_CAPABILITY:
449 if (params->mode < IEEE80211_CONN_MODE_VHT)
450 break;
451 if (elen >= sizeof(struct ieee80211_vht_cap))
452 elems->vht_cap_elem = (void *)pos;
453 else
454 elem_parse_failed =
455 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
456 break;
457 case WLAN_EID_VHT_OPERATION:
458 if (params->mode < IEEE80211_CONN_MODE_VHT)
459 break;
460 if (elen >= sizeof(struct ieee80211_vht_operation)) {
461 elems->vht_operation = (void *)pos;
462 if (calc_crc)
463 crc = crc32_be(crc, pos - 2, elen + 2);
464 break;
466 elem_parse_failed =
467 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
468 break;
469 case WLAN_EID_OPMODE_NOTIF:
470 if (params->mode < IEEE80211_CONN_MODE_VHT)
471 break;
472 if (elen > 0) {
473 elems->opmode_notif = pos;
474 if (calc_crc)
475 crc = crc32_be(crc, pos - 2, elen + 2);
476 break;
478 elem_parse_failed =
479 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
480 break;
481 case WLAN_EID_MESH_ID:
482 elems->mesh_id = pos;
483 elems->mesh_id_len = elen;
484 break;
485 case WLAN_EID_MESH_CONFIG:
486 if (elen >= sizeof(struct ieee80211_meshconf_ie))
487 elems->mesh_config = (void *)pos;
488 else
489 elem_parse_failed =
490 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
491 break;
492 case WLAN_EID_PEER_MGMT:
493 elems->peering = pos;
494 elems->peering_len = elen;
495 break;
496 case WLAN_EID_MESH_AWAKE_WINDOW:
497 if (elen >= 2)
498 elems->awake_window = (void *)pos;
499 break;
500 case WLAN_EID_PREQ:
501 elems->preq = pos;
502 elems->preq_len = elen;
503 break;
504 case WLAN_EID_PREP:
505 elems->prep = pos;
506 elems->prep_len = elen;
507 break;
508 case WLAN_EID_PERR:
509 elems->perr = pos;
510 elems->perr_len = elen;
511 break;
512 case WLAN_EID_RANN:
513 if (elen >= sizeof(struct ieee80211_rann_ie))
514 elems->rann = (void *)pos;
515 else
516 elem_parse_failed =
517 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
518 break;
519 case WLAN_EID_CHANNEL_SWITCH:
520 if (elen != sizeof(struct ieee80211_channel_sw_ie)) {
521 elem_parse_failed =
522 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
523 break;
525 elems->ch_switch_ie = (void *)pos;
526 break;
527 case WLAN_EID_EXT_CHANSWITCH_ANN:
528 if (elen != sizeof(struct ieee80211_ext_chansw_ie)) {
529 elem_parse_failed =
530 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
531 break;
533 elems->ext_chansw_ie = (void *)pos;
534 break;
535 case WLAN_EID_SECONDARY_CHANNEL_OFFSET:
536 if (params->mode < IEEE80211_CONN_MODE_HT)
537 break;
538 if (elen != sizeof(struct ieee80211_sec_chan_offs_ie)) {
539 elem_parse_failed =
540 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
541 break;
543 elems->sec_chan_offs = (void *)pos;
544 break;
545 case WLAN_EID_CHAN_SWITCH_PARAM:
546 if (elen <
547 sizeof(*elems->mesh_chansw_params_ie)) {
548 elem_parse_failed =
549 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
550 break;
552 elems->mesh_chansw_params_ie = (void *)pos;
553 break;
554 case WLAN_EID_WIDE_BW_CHANNEL_SWITCH:
555 if (params->mode < IEEE80211_CONN_MODE_VHT)
556 break;
558 if (!params->action) {
559 elem_parse_failed =
560 IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
561 break;
564 if (elen < sizeof(*elems->wide_bw_chansw_ie)) {
565 elem_parse_failed =
566 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
567 break;
569 elems->wide_bw_chansw_ie = (void *)pos;
570 break;
571 case WLAN_EID_CHANNEL_SWITCH_WRAPPER:
572 if (params->mode < IEEE80211_CONN_MODE_VHT)
573 break;
574 if (params->action) {
575 elem_parse_failed =
576 IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
577 break;
580 * This is a bit tricky, but as we only care about
581 * a few elements, parse them out manually.
583 subelem = cfg80211_find_elem(WLAN_EID_WIDE_BW_CHANNEL_SWITCH,
584 pos, elen);
585 if (subelem) {
586 if (subelem->datalen >= sizeof(*elems->wide_bw_chansw_ie))
587 elems->wide_bw_chansw_ie =
588 (void *)subelem->data;
589 else
590 elem_parse_failed =
591 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
594 if (params->mode < IEEE80211_CONN_MODE_EHT)
595 break;
597 subelem = cfg80211_find_ext_elem(WLAN_EID_EXT_BANDWIDTH_INDICATION,
598 pos, elen);
599 if (subelem) {
600 const void *edata = subelem->data + 1;
601 u8 edatalen = subelem->datalen - 1;
603 if (ieee80211_bandwidth_indication_size_ok(edata,
604 edatalen))
605 elems->bandwidth_indication = edata;
606 else
607 elem_parse_failed =
608 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
611 subelem = cfg80211_find_ext_elem(WLAN_EID_TX_POWER_ENVELOPE,
612 pos, elen);
613 if (subelem)
614 ieee80211_parse_tpe(&elems->csa_tpe,
615 subelem->data + 1,
616 subelem->datalen - 1);
617 break;
618 case WLAN_EID_COUNTRY:
619 elems->country_elem = pos;
620 elems->country_elem_len = elen;
621 break;
622 case WLAN_EID_PWR_CONSTRAINT:
623 if (elen != 1) {
624 elem_parse_failed =
625 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
626 break;
628 elems->pwr_constr_elem = pos;
629 break;
630 case WLAN_EID_CISCO_VENDOR_SPECIFIC:
631 /* Lots of different options exist, but we only care
632 * about the Dynamic Transmit Power Control element.
633 * First check for the Cisco OUI, then for the DTPC
634 * tag (0x00).
636 if (elen < 4) {
637 elem_parse_failed =
638 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
639 break;
642 if (pos[0] != 0x00 || pos[1] != 0x40 ||
643 pos[2] != 0x96 || pos[3] != 0x00)
644 break;
646 if (elen != 6) {
647 elem_parse_failed =
648 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
649 break;
652 if (calc_crc)
653 crc = crc32_be(crc, pos - 2, elen + 2);
655 elems->cisco_dtpc_elem = pos;
656 break;
657 case WLAN_EID_ADDBA_EXT:
658 if (elen < sizeof(struct ieee80211_addba_ext_ie)) {
659 elem_parse_failed =
660 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
661 break;
663 elems->addba_ext_ie = (void *)pos;
664 break;
665 case WLAN_EID_TIMEOUT_INTERVAL:
666 if (elen >= sizeof(struct ieee80211_timeout_interval_ie))
667 elems->timeout_int = (void *)pos;
668 else
669 elem_parse_failed =
670 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
671 break;
672 case WLAN_EID_BSS_MAX_IDLE_PERIOD:
673 if (elen >= sizeof(*elems->max_idle_period_ie))
674 elems->max_idle_period_ie = (void *)pos;
675 break;
676 case WLAN_EID_RSNX:
677 elems->rsnx = pos;
678 elems->rsnx_len = elen;
679 break;
680 case WLAN_EID_TX_POWER_ENVELOPE:
681 if (params->mode < IEEE80211_CONN_MODE_HE)
682 break;
683 ieee80211_parse_tpe(&elems->tpe, pos, elen);
684 break;
685 case WLAN_EID_EXTENSION:
686 ieee80211_parse_extension_element(calc_crc ?
687 &crc : NULL,
688 elem, elems_parse,
689 params);
690 break;
691 case WLAN_EID_S1G_CAPABILITIES:
692 if (params->mode != IEEE80211_CONN_MODE_S1G)
693 break;
694 if (elen >= sizeof(*elems->s1g_capab))
695 elems->s1g_capab = (void *)pos;
696 else
697 elem_parse_failed =
698 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
699 break;
700 case WLAN_EID_S1G_OPERATION:
701 if (params->mode != IEEE80211_CONN_MODE_S1G)
702 break;
703 if (elen == sizeof(*elems->s1g_oper))
704 elems->s1g_oper = (void *)pos;
705 else
706 elem_parse_failed =
707 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
708 break;
709 case WLAN_EID_S1G_BCN_COMPAT:
710 if (params->mode != IEEE80211_CONN_MODE_S1G)
711 break;
712 if (elen == sizeof(*elems->s1g_bcn_compat))
713 elems->s1g_bcn_compat = (void *)pos;
714 else
715 elem_parse_failed =
716 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
717 break;
718 case WLAN_EID_AID_RESPONSE:
719 if (params->mode != IEEE80211_CONN_MODE_S1G)
720 break;
721 if (elen == sizeof(struct ieee80211_aid_response_ie))
722 elems->aid_resp = (void *)pos;
723 else
724 elem_parse_failed =
725 IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
726 break;
727 default:
728 break;
731 if (elem_parse_failed)
732 elems->parse_error |= elem_parse_failed;
733 else
734 __set_bit(id, seen_elems);
737 if (!for_each_element_completed(elem, params->start, params->len))
738 elems->parse_error |= IEEE80211_PARSE_ERR_INVALID_END;
740 return crc;
743 static size_t ieee802_11_find_bssid_profile(const u8 *start, size_t len,
744 struct ieee802_11_elems *elems,
745 struct cfg80211_bss *bss,
746 u8 *nontransmitted_profile)
748 const struct element *elem, *sub;
749 size_t profile_len = 0;
750 bool found = false;
752 if (!bss || !bss->transmitted_bss)
753 return profile_len;
755 for_each_element_id(elem, WLAN_EID_MULTIPLE_BSSID, start, len) {
756 if (elem->datalen < 2)
757 continue;
758 if (elem->data[0] < 1 || elem->data[0] > 8)
759 continue;
761 for_each_element(sub, elem->data + 1, elem->datalen - 1) {
762 u8 new_bssid[ETH_ALEN];
763 const u8 *index;
765 if (sub->id != 0 || sub->datalen < 4) {
766 /* not a valid BSS profile */
767 continue;
770 if (sub->data[0] != WLAN_EID_NON_TX_BSSID_CAP ||
771 sub->data[1] != 2) {
772 /* The first element of the
773 * Nontransmitted BSSID Profile is not
774 * the Nontransmitted BSSID Capability
775 * element.
777 continue;
780 memset(nontransmitted_profile, 0, len);
781 profile_len = cfg80211_merge_profile(start, len,
782 elem,
783 sub,
784 nontransmitted_profile,
785 len);
787 /* found a Nontransmitted BSSID Profile */
788 index = cfg80211_find_ie(WLAN_EID_MULTI_BSSID_IDX,
789 nontransmitted_profile,
790 profile_len);
791 if (!index || index[1] < 1 || index[2] == 0) {
792 /* Invalid MBSSID Index element */
793 continue;
796 cfg80211_gen_new_bssid(bss->transmitted_bss->bssid,
797 elem->data[0],
798 index[2],
799 new_bssid);
800 if (ether_addr_equal(new_bssid, bss->bssid)) {
801 found = true;
802 elems->bssid_index_len = index[1];
803 elems->bssid_index = (void *)&index[2];
804 break;
809 return found ? profile_len : 0;
812 static void
813 ieee80211_mle_get_sta_prof(struct ieee80211_elems_parse *elems_parse,
814 u8 link_id)
816 struct ieee802_11_elems *elems = &elems_parse->elems;
817 const struct ieee80211_multi_link_elem *ml = elems->ml_basic;
818 ssize_t ml_len = elems->ml_basic_len;
819 const struct element *sub;
821 for_each_mle_subelement(sub, (u8 *)ml, ml_len) {
822 struct ieee80211_mle_per_sta_profile *prof = (void *)sub->data;
823 ssize_t sta_prof_len;
824 u16 control;
826 if (sub->id != IEEE80211_MLE_SUBELEM_PER_STA_PROFILE)
827 continue;
829 if (!ieee80211_mle_basic_sta_prof_size_ok(sub->data,
830 sub->datalen))
831 return;
833 control = le16_to_cpu(prof->control);
835 if (link_id != u16_get_bits(control,
836 IEEE80211_MLE_STA_CONTROL_LINK_ID))
837 continue;
839 if (!(control & IEEE80211_MLE_STA_CONTROL_COMPLETE_PROFILE))
840 return;
842 /* the sub element can be fragmented */
843 sta_prof_len =
844 cfg80211_defragment_element(sub,
845 (u8 *)ml, ml_len,
846 elems_parse->scratch_pos,
847 elems_parse->scratch +
848 elems_parse->scratch_len -
849 elems_parse->scratch_pos,
850 IEEE80211_MLE_SUBELEM_FRAGMENT);
852 if (sta_prof_len < 0)
853 return;
855 elems->prof = (void *)elems_parse->scratch_pos;
856 elems->sta_prof_len = sta_prof_len;
857 elems_parse->scratch_pos += sta_prof_len;
859 return;
863 static void ieee80211_mle_parse_link(struct ieee80211_elems_parse *elems_parse,
864 struct ieee80211_elems_parse_params *params)
866 struct ieee802_11_elems *elems = &elems_parse->elems;
867 struct ieee80211_mle_per_sta_profile *prof;
868 struct ieee80211_elems_parse_params sub = {
869 .mode = params->mode,
870 .action = params->action,
871 .from_ap = params->from_ap,
872 .link_id = -1,
874 ssize_t ml_len = elems->ml_basic_len;
875 const struct element *non_inherit = NULL;
876 const u8 *end;
878 ml_len = cfg80211_defragment_element(elems_parse->ml_basic_elem,
879 elems->ie_start,
880 elems->total_len,
881 elems_parse->scratch_pos,
882 elems_parse->scratch +
883 elems_parse->scratch_len -
884 elems_parse->scratch_pos,
885 WLAN_EID_FRAGMENT);
887 if (ml_len < 0)
888 return;
890 elems->ml_basic = (const void *)elems_parse->scratch_pos;
891 elems->ml_basic_len = ml_len;
892 elems_parse->scratch_pos += ml_len;
894 if (params->link_id == -1)
895 return;
897 ieee80211_mle_get_sta_prof(elems_parse, params->link_id);
898 prof = elems->prof;
900 if (!prof)
901 return;
903 /* check if we have the 4 bytes for the fixed part in assoc response */
904 if (elems->sta_prof_len < sizeof(*prof) + prof->sta_info_len - 1 + 4) {
905 elems->prof = NULL;
906 elems->sta_prof_len = 0;
907 return;
911 * Skip the capability information and the status code that are expected
912 * as part of the station profile in association response frames. Note
913 * the -1 is because the 'sta_info_len' is accounted to as part of the
914 * per-STA profile, but not part of the 'u8 variable[]' portion.
916 sub.start = prof->variable + prof->sta_info_len - 1 + 4;
917 end = (const u8 *)prof + elems->sta_prof_len;
918 sub.len = end - sub.start;
920 non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
921 sub.start, sub.len);
922 _ieee802_11_parse_elems_full(&sub, elems_parse, non_inherit);
925 static void
926 ieee80211_mle_defrag_reconf(struct ieee80211_elems_parse *elems_parse)
928 struct ieee802_11_elems *elems = &elems_parse->elems;
929 ssize_t ml_len;
931 ml_len = cfg80211_defragment_element(elems_parse->ml_reconf_elem,
932 elems->ie_start,
933 elems->total_len,
934 elems_parse->scratch_pos,
935 elems_parse->scratch +
936 elems_parse->scratch_len -
937 elems_parse->scratch_pos,
938 WLAN_EID_FRAGMENT);
939 if (ml_len < 0)
940 return;
941 elems->ml_reconf = (void *)elems_parse->scratch_pos;
942 elems->ml_reconf_len = ml_len;
943 elems_parse->scratch_pos += ml_len;
946 struct ieee802_11_elems *
947 ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params)
949 struct ieee80211_elems_parse *elems_parse;
950 struct ieee802_11_elems *elems;
951 const struct element *non_inherit = NULL;
952 u8 *nontransmitted_profile;
953 int nontransmitted_profile_len = 0;
954 size_t scratch_len = 3 * params->len;
956 BUILD_BUG_ON(offsetof(typeof(*elems_parse), elems) != 0);
958 elems_parse = kzalloc(struct_size(elems_parse, scratch, scratch_len),
959 GFP_ATOMIC);
960 if (!elems_parse)
961 return NULL;
963 elems_parse->scratch_len = scratch_len;
964 elems_parse->scratch_pos = elems_parse->scratch;
966 elems = &elems_parse->elems;
967 elems->ie_start = params->start;
968 elems->total_len = params->len;
970 /* set all TPE entries to unlimited (but invalid) */
971 ieee80211_clear_tpe(&elems->tpe);
972 ieee80211_clear_tpe(&elems->csa_tpe);
974 nontransmitted_profile = elems_parse->scratch_pos;
975 nontransmitted_profile_len =
976 ieee802_11_find_bssid_profile(params->start, params->len,
977 elems, params->bss,
978 nontransmitted_profile);
979 elems_parse->scratch_pos += nontransmitted_profile_len;
980 non_inherit = cfg80211_find_ext_elem(WLAN_EID_EXT_NON_INHERITANCE,
981 nontransmitted_profile,
982 nontransmitted_profile_len);
984 elems->crc = _ieee802_11_parse_elems_full(params, elems_parse,
985 non_inherit);
987 /* Override with nontransmitted profile, if found */
988 if (nontransmitted_profile_len) {
989 struct ieee80211_elems_parse_params sub = {
990 .mode = params->mode,
991 .start = nontransmitted_profile,
992 .len = nontransmitted_profile_len,
993 .action = params->action,
994 .link_id = params->link_id,
997 _ieee802_11_parse_elems_full(&sub, elems_parse, NULL);
1000 ieee80211_mle_parse_link(elems_parse, params);
1002 ieee80211_mle_defrag_reconf(elems_parse);
1004 if (elems->tim && !elems->parse_error) {
1005 const struct ieee80211_tim_ie *tim_ie = elems->tim;
1007 elems->dtim_period = tim_ie->dtim_period;
1008 elems->dtim_count = tim_ie->dtim_count;
1011 /* Override DTIM period and count if needed */
1012 if (elems->bssid_index &&
1013 elems->bssid_index_len >=
1014 offsetofend(struct ieee80211_bssid_index, dtim_period))
1015 elems->dtim_period = elems->bssid_index->dtim_period;
1017 if (elems->bssid_index &&
1018 elems->bssid_index_len >=
1019 offsetofend(struct ieee80211_bssid_index, dtim_count))
1020 elems->dtim_count = elems->bssid_index->dtim_count;
1022 return elems;
1024 EXPORT_SYMBOL_IF_KUNIT(ieee802_11_parse_elems_full);
1026 int ieee80211_parse_bitrates(enum nl80211_chan_width width,
1027 const struct ieee80211_supported_band *sband,
1028 const u8 *srates, int srates_len, u32 *rates)
1030 u32 rate_flags = ieee80211_chanwidth_rate_flags(width);
1031 struct ieee80211_rate *br;
1032 int brate, rate, i, j, count = 0;
1034 *rates = 0;
1036 for (i = 0; i < srates_len; i++) {
1037 rate = srates[i] & 0x7f;
1039 for (j = 0; j < sband->n_bitrates; j++) {
1040 br = &sband->bitrates[j];
1041 if ((rate_flags & br->flags) != rate_flags)
1042 continue;
1044 brate = DIV_ROUND_UP(br->bitrate, 5);
1045 if (brate == rate) {
1046 *rates |= BIT(j);
1047 count++;
1048 break;
1052 return count;