xref: /linux/net/mac80211/parse.c (revision 2a52ca7c98960aafb0eca9ef96b2d0c932171357)
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
10  *
11  * element parsing for mac80211
12  */
13 
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>
28 
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"
36 
37 struct ieee80211_elems_parse {
38 	/* must be first for kfree to work */
39 	struct ieee802_11_elems elems;
40 
41 	/* The basic Multi-Link element in the original elements */
42 	const struct element *ml_basic_elem;
43 
44 	/* The reconfiguration Multi-Link element in the original elements */
45 	const struct element *ml_reconf_elem;
46 
47 	/*
48 	 * scratch buffer that can be used for various element parsing related
49 	 * tasks, e.g., element de-fragmentation etc.
50 	 */
51 	size_t scratch_len;
52 	u8 *scratch_pos;
53 	u8 scratch[] __counted_by(scratch_len);
54 };
55 
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)
61 {
62 	struct ieee802_11_elems *elems = &elems_parse->elems;
63 	const void *data = elem->data + 1;
64 	bool calc_crc = false;
65 	u8 len;
66 
67 	if (!elem->datalen)
68 		return;
69 
70 	len = elem->datalen - 1;
71 
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;
86 		}
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;
131 		}
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;
144 
145 		if (ieee80211_mle_size_ok(data, len)) {
146 			const struct ieee80211_multi_link_elem *mle =
147 				(void *)data;
148 
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;
156 				}
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;
164 			}
165 		}
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++;
182 		}
183 		break;
184 	}
185 
186 	if (crc && calc_crc)
187 		*crc = crc32_be(*crc, (void *)elem, elem->datalen + 2);
188 }
189 
190 static void ieee80211_parse_tpe(struct ieee80211_parsed_tpe *tpe,
191 				const u8 *data, u8 len)
192 {
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;
196 
197 	if (!ieee80211_valid_tpe_element(data, len))
198 		return;
199 
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);
203 
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;
227 	}
228 
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[count + 2];
239 			*cnt_out = 5;
240 		}
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;
249 		}
250 
251 		N = 1 << (count - 1);
252 		memcpy(out, env->variable, N);
253 		*cnt_out = N;
254 		*N_out = N;
255 
256 		if (len > sizeof(*env) + N) {
257 			int K = u8_get_bits(env->variable[N],
258 					    IEEE80211_TX_PWR_ENV_EXT_COUNT);
259 
260 			K = min(K, IEEE80211_TPE_PSD_ENTRIES_320MHZ - N);
261 			memcpy(out + N, env->variable + N + 1, K);
262 			(*cnt_out) += K;
263 		}
264 		break;
265 	}
266 }
267 
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)
272 {
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;
278 
279 	bitmap_zero(seen_elems, 256);
280 
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;
287 
288 		if (check_inherit &&
289 		    !cfg80211_is_element_inherited(elem,
290 						   check_inherit))
291 			continue;
292 
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:
334 		/*
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
337 		 */
338 			if (test_bit(id, seen_elems)) {
339 				elems->parse_error |=
340 					IEEE80211_PARSE_ERR_DUP_ELEM;
341 				continue;
342 			}
343 			break;
344 		}
345 
346 		if (calc_crc && id < 64 && (params->filter & (1ULL << id)))
347 			crc = crc32_be(crc, pos - 2, elen + 2);
348 
349 		elem_parse_failed = 0;
350 
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;
357 			}
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;
365 			}
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) */
399 
400 				if (calc_crc)
401 					crc = crc32_be(crc, pos - 2, elen + 2);
402 
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;
411 					}
412 				}
413 			}
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;
465 			}
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;
477 			}
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;
524 			}
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;
532 			}
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;
542 			}
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;
551 			}
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;
557 
558 			if (!params->action) {
559 				elem_parse_failed =
560 					IEEE80211_PARSE_ERR_UNEXPECTED_ELEM;
561 				break;
562 			}
563 
564 			if (elen < sizeof(*elems->wide_bw_chansw_ie)) {
565 				elem_parse_failed =
566 					IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
567 				break;
568 			}
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;
578 			}
579 			/*
580 			 * This is a bit tricky, but as we only care about
581 			 * a few elements, parse them out manually.
582 			 */
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;
592 			}
593 
594 			if (params->mode < IEEE80211_CONN_MODE_EHT)
595 				break;
596 
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;
602 
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;
609 			}
610 
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;
627 			}
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).
635 			 */
636 			if (elen < 4) {
637 				elem_parse_failed =
638 					IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
639 				break;
640 			}
641 
642 			if (pos[0] != 0x00 || pos[1] != 0x40 ||
643 			    pos[2] != 0x96 || pos[3] != 0x00)
644 				break;
645 
646 			if (elen != 6) {
647 				elem_parse_failed =
648 					IEEE80211_PARSE_ERR_BAD_ELEM_SIZE;
649 				break;
650 			}
651 
652 			if (calc_crc)
653 				crc = crc32_be(crc, pos - 2, elen + 2);
654 
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;
662 			}
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;
729 		}
730 
731 		if (elem_parse_failed)
732 			elems->parse_error |= elem_parse_failed;
733 		else
734 			__set_bit(id, seen_elems);
735 	}
736 
737 	if (!for_each_element_completed(elem, params->start, params->len))
738 		elems->parse_error |= IEEE80211_PARSE_ERR_INVALID_END;
739 
740 	return crc;
741 }
742 
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)
747 {
748 	const struct element *elem, *sub;
749 	size_t profile_len = 0;
750 	bool found = false;
751 
752 	if (!bss || !bss->transmitted_bss)
753 		return profile_len;
754 
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;
760 
761 		for_each_element(sub, elem->data + 1, elem->datalen - 1) {
762 			u8 new_bssid[ETH_ALEN];
763 			const u8 *index;
764 
765 			if (sub->id != 0 || sub->datalen < 4) {
766 				/* not a valid BSS profile */
767 				continue;
768 			}
769 
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.
776 				 */
777 				continue;
778 			}
779 
780 			memset(nontransmitted_profile, 0, len);
781 			profile_len = cfg80211_merge_profile(start, len,
782 							     elem,
783 							     sub,
784 							     nontransmitted_profile,
785 							     len);
786 
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;
794 			}
795 
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;
805 			}
806 		}
807 	}
808 
809 	return found ? profile_len : 0;
810 }
811 
812 static void
813 ieee80211_mle_get_sta_prof(struct ieee80211_elems_parse *elems_parse,
814 			   u8 link_id)
815 {
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;
820 
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;
825 
826 		if (sub->id != IEEE80211_MLE_SUBELEM_PER_STA_PROFILE)
827 			continue;
828 
829 		if (!ieee80211_mle_basic_sta_prof_size_ok(sub->data,
830 							  sub->datalen))
831 			return;
832 
833 		control = le16_to_cpu(prof->control);
834 
835 		if (link_id != u16_get_bits(control,
836 					    IEEE80211_MLE_STA_CONTROL_LINK_ID))
837 			continue;
838 
839 		if (!(control & IEEE80211_MLE_STA_CONTROL_COMPLETE_PROFILE))
840 			return;
841 
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);
851 
852 		if (sta_prof_len < 0)
853 			return;
854 
855 		elems->prof = (void *)elems_parse->scratch_pos;
856 		elems->sta_prof_len = sta_prof_len;
857 		elems_parse->scratch_pos += sta_prof_len;
858 
859 		return;
860 	}
861 }
862 
863 static void ieee80211_mle_parse_link(struct ieee80211_elems_parse *elems_parse,
864 				     struct ieee80211_elems_parse_params *params)
865 {
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,
873 	};
874 	ssize_t ml_len = elems->ml_basic_len;
875 	const struct element *non_inherit = NULL;
876 	const u8 *end;
877 
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);
886 
887 	if (ml_len < 0)
888 		return;
889 
890 	elems->ml_basic = (const void *)elems_parse->scratch_pos;
891 	elems->ml_basic_len = ml_len;
892 	elems_parse->scratch_pos += ml_len;
893 
894 	if (params->link_id == -1)
895 		return;
896 
897 	ieee80211_mle_get_sta_prof(elems_parse, params->link_id);
898 	prof = elems->prof;
899 
900 	if (!prof)
901 		return;
902 
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;
908 	}
909 
910 	/*
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.
915 	 */
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;
919 
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);
923 }
924 
925 static void
926 ieee80211_mle_defrag_reconf(struct ieee80211_elems_parse *elems_parse)
927 {
928 	struct ieee802_11_elems *elems = &elems_parse->elems;
929 	ssize_t ml_len;
930 
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;
944 }
945 
946 struct ieee802_11_elems *
947 ieee802_11_parse_elems_full(struct ieee80211_elems_parse_params *params)
948 {
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;
955 
956 	BUILD_BUG_ON(offsetof(typeof(*elems_parse), elems) != 0);
957 
958 	elems_parse = kzalloc(struct_size(elems_parse, scratch, scratch_len),
959 			      GFP_ATOMIC);
960 	if (!elems_parse)
961 		return NULL;
962 
963 	elems_parse->scratch_len = scratch_len;
964 	elems_parse->scratch_pos = elems_parse->scratch;
965 
966 	elems = &elems_parse->elems;
967 	elems->ie_start = params->start;
968 	elems->total_len = params->len;
969 
970 	/* set all TPE entries to unlimited (but invalid) */
971 	ieee80211_clear_tpe(&elems->tpe);
972 	ieee80211_clear_tpe(&elems->csa_tpe);
973 
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);
983 
984 	elems->crc = _ieee802_11_parse_elems_full(params, elems_parse,
985 						  non_inherit);
986 
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,
995 		};
996 
997 		_ieee802_11_parse_elems_full(&sub, elems_parse, NULL);
998 	}
999 
1000 	ieee80211_mle_parse_link(elems_parse, params);
1001 
1002 	ieee80211_mle_defrag_reconf(elems_parse);
1003 
1004 	if (elems->tim && !elems->parse_error) {
1005 		const struct ieee80211_tim_ie *tim_ie = elems->tim;
1006 
1007 		elems->dtim_period = tim_ie->dtim_period;
1008 		elems->dtim_count = tim_ie->dtim_count;
1009 	}
1010 
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;
1016 
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;
1021 
1022 	return elems;
1023 }
1024 EXPORT_SYMBOL_IF_KUNIT(ieee802_11_parse_elems_full);
1025 
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)
1029 {
1030 	u32 rate_flags = ieee80211_chanwidth_rate_flags(width);
1031 	struct ieee80211_rate *br;
1032 	int brate, rate, i, j, count = 0;
1033 
1034 	*rates = 0;
1035 
1036 	for (i = 0; i < srates_len; i++) {
1037 		rate = srates[i] & 0x7f;
1038 
1039 		for (j = 0; j < sband->n_bitrates; j++) {
1040 			br = &sband->bitrates[j];
1041 			if ((rate_flags & br->flags) != rate_flags)
1042 				continue;
1043 
1044 			brate = DIV_ROUND_UP(br->bitrate, 5);
1045 			if (brate == rate) {
1046 				*rates |= BIT(j);
1047 				count++;
1048 				break;
1049 			}
1050 		}
1051 	}
1052 	return count;
1053 }
1054