1 /*
2 * IEEE 802.11 Common routines
3 * Copyright (c) 2002-2019, Jouni Malinen <j@w1.fi>
4 *
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
7 */
8
9 #ifndef IEEE802_11_COMMON_H
10 #define IEEE802_11_COMMON_H
11
12 #include "defs.h"
13 #include "ieee802_11_defs.h"
14
15 struct element {
16 u8 id;
17 u8 datalen;
18 u8 data[];
19 } STRUCT_PACKED;
20
21 struct hostapd_hw_modes;
22
23 #define MAX_NOF_MB_IES_SUPPORTED 5
24
25 #define MAX_NOSS_RX_LENGTH 4
26 #define CHAN_CENTER_FREQ_VALUE 8
27 #define VHT_CHAN_WIDTH_TYPE_1 1
28 #define VHT_CHAN_WIDTH_TYPE_2 2
29 #define VHT_CHAN_WIDTH_TYPE_3 3
30 #define SIX_GHZ_CHANN_WIDTH_TYPE_1 1
31 #define SIX_GHZ_CHANN_WIDTH_TYPE_2 2
32 #define SIX_GHZ_CHANN_WIDTH_TYPE_3 3
33
34 struct mb_ies_info {
35 struct {
36 const u8 *ie;
37 u8 ie_len;
38 } ies[MAX_NOF_MB_IES_SUPPORTED];
39 u8 nof_ies;
40 };
41
42 struct multi_ap_params {
43 u8 capability;
44 u8 profile;
45 u16 vlanid;
46 };
47
48 /* Parsed Information Elements */
49 struct ieee802_11_elems {
50 const u8 *ssid;
51 const u8 *supp_rates;
52 const u8 *ds_params;
53 const u8 *challenge;
54 const u8 *erp_info;
55 const u8 *ext_supp_rates;
56 const u8 *wpa_ie;
57 const u8 *rsn_ie;
58 const u8 *rsnxe;
59 const u8 *wmm; /* WMM Information or Parameter Element */
60 const u8 *wmm_tspec;
61 const u8 *wps_ie;
62 const u8 *supp_channels;
63 const u8 *mdie;
64 const u8 *ftie;
65 const u8 *timeout_int;
66 const u8 *ht_capabilities;
67 const u8 *ht_operation;
68 const u8 *mesh_config;
69 const u8 *mesh_id;
70 const u8 *peer_mgmt;
71 const u8 *vht_capabilities;
72 const u8 *vht_operation;
73 const u8 *opmode_notif;
74 const u8 *vendor_ht_cap;
75 const u8 *vendor_vht;
76 const u8 *p2p;
77 const u8 *wfd;
78 const u8 *link_id;
79 const u8 *interworking;
80 const u8 *qos_map_set;
81 const u8 *hs20;
82 const u8 *ext_capab;
83 const u8 *bss_max_idle_period;
84 const u8 *ssid_list;
85 const u8 *osen;
86 const u8 *mbo;
87 const u8 *ampe;
88 const u8 *mic;
89 const u8 *pref_freq_list;
90 const u8 *supp_op_classes;
91 const u8 *rrm_enabled;
92 const u8 *cag_number;
93 const u8 *ap_csn;
94 const u8 *fils_indic;
95 const u8 *dils;
96 const u8 *assoc_delay_info;
97 const u8 *fils_req_params;
98 const u8 *fils_key_confirm;
99 const u8 *fils_session;
100 const u8 *fils_hlp;
101 const u8 *fils_ip_addr_assign;
102 const u8 *key_delivery;
103 const u8 *wrapped_data;
104 const u8 *fils_pk;
105 const u8 *fils_nonce;
106 const u8 *owe_dh;
107 #if defined(CONFIG_OPEN_HARMONY_PATCH) && defined(OPEN_HARMONY_MIRACAST_SINK_OPT)
108 const u8 *pvt_peer_band;
109 const u8 *pvt_peer_ch;
110 const u8 *pvt_peer_cap;
111 #endif
112 const u8 *power_capab;
113 const u8 *roaming_cons_sel;
114 const u8 *password_id;
115 const u8 *oci;
116 const u8 *multi_ap;
117 const u8 *he_capabilities;
118 const u8 *he_operation;
119 const u8 *short_ssid_list;
120 const u8 *he_6ghz_band_cap;
121 const u8 *sae_pk;
122 const u8 *s1g_capab;
123 const u8 *pasn_params;
124 const u8 *eht_capabilities;
125 const u8 *eht_operation;
126 const u8 *basic_mle;
127 const u8 *probe_req_mle;
128 const u8 *reconf_mle;
129 const u8 *tdls_mle;
130 const u8 *prior_access_mle;
131 const u8 *mbssid_known_bss;
132 const u8 *mbssid;
133
134 u8 ssid_len;
135 u8 supp_rates_len;
136 u8 challenge_len;
137 u8 ext_supp_rates_len;
138 u8 wpa_ie_len;
139 u8 rsn_ie_len;
140 u8 rsnxe_len;
141 u8 wmm_len; /* 7 = WMM Information; 24 = WMM Parameter */
142 u8 wmm_tspec_len;
143 u8 wps_ie_len;
144 u8 supp_channels_len;
145 u8 mdie_len;
146 u8 ftie_len;
147 u8 mesh_config_len;
148 u8 mesh_id_len;
149 u8 peer_mgmt_len;
150 u8 vendor_ht_cap_len;
151 u8 vendor_vht_len;
152 u8 p2p_len;
153 u8 wfd_len;
154 u8 interworking_len;
155 u8 qos_map_set_len;
156 u8 hs20_len;
157 u8 ext_capab_len;
158 u8 ssid_list_len;
159 u8 osen_len;
160 u8 mbo_len;
161 u8 ampe_len;
162 u8 mic_len;
163 u8 pref_freq_list_len;
164 u8 supp_op_classes_len;
165 u8 rrm_enabled_len;
166 u8 cag_number_len;
167 u8 fils_indic_len;
168 u8 dils_len;
169 u8 fils_req_params_len;
170 u8 fils_key_confirm_len;
171 size_t fils_hlp_len;
172 u8 fils_ip_addr_assign_len;
173 u8 key_delivery_len;
174 size_t wrapped_data_len;
175 u8 fils_pk_len;
176 u8 owe_dh_len;
177 u8 power_capab_len;
178 u8 roaming_cons_sel_len;
179 u8 password_id_len;
180 u8 oci_len;
181 u8 multi_ap_len;
182 u8 he_capabilities_len;
183 u8 he_operation_len;
184 u8 short_ssid_list_len;
185 u8 sae_pk_len;
186 u8 pasn_params_len;
187 u8 eht_capabilities_len;
188 u8 eht_operation_len;
189 size_t basic_mle_len;
190 size_t probe_req_mle_len;
191 size_t reconf_mle_len;
192 size_t tdls_mle_len;
193 size_t prior_access_mle_len;
194 u8 mbssid_known_bss_len;
195 u8 mbssid_len;
196
197 struct mb_ies_info mb_ies;
198
199 size_t fte_defrag_len;
200
201 /*
202 * The number of fragment elements to be skipped after a known
203 * fragmented element.
204 */
205 unsigned int num_frag_elems;
206 };
207
208 typedef enum { ParseOK = 0, ParseUnknown = 1, ParseFailed = -1 } ParseRes;
209
210 ParseRes ieee802_11_parse_elems(const u8 *start, size_t len,
211 struct ieee802_11_elems *elems,
212 int show_errors);
213 void ieee802_11_elems_clear_ids(struct ieee802_11_elems *elems,
214 const u8 *ids, size_t num);
215 void ieee802_11_elems_clear_ext_ids(struct ieee802_11_elems *elems,
216 const u8 *ids, size_t num);
217 ParseRes ieee802_11_parse_link_assoc_req(const u8 *start, size_t len,
218 struct ieee802_11_elems *elems,
219 struct wpabuf *mlbuf,
220 u8 link_id, bool show_errors);
221 int ieee802_11_ie_count(const u8 *ies, size_t ies_len);
222 struct wpabuf * ieee802_11_vendor_ie_concat(const u8 *ies, size_t ies_len,
223 u32 oui_type);
224 struct ieee80211_hdr;
225 const u8 * get_hdr_bssid(const struct ieee80211_hdr *hdr, size_t len);
226
227 struct hostapd_wmm_ac_params {
228 int cwmin;
229 int cwmax;
230 int aifs;
231 int txop_limit; /* in units of 32us */
232 int admission_control_mandatory;
233 };
234
235 int hostapd_config_wmm_ac(struct hostapd_wmm_ac_params wmm_ac_params[],
236 const char *name, const char *val);
237
238 struct hostapd_tx_queue_params {
239 int aifs;
240 int cwmin;
241 int cwmax;
242 int burst; /* maximum burst time in 0.1 ms, i.e., 10 = 1 ms */
243 };
244
245 #define NUM_TX_QUEUES 4
246
247 int hostapd_config_tx_queue(struct hostapd_tx_queue_params queue[],
248 const char *name, const char *val);
249 enum hostapd_hw_mode ieee80211_freq_to_chan(int freq, u8 *channel);
250 int ieee80211_chan_to_freq(const char *country, u8 op_class, u8 chan);
251 enum hostapd_hw_mode
252 ieee80211_freq_to_channel_ext(unsigned int freq, int sec_channel,
253 enum oper_chan_width chanwidth,
254 u8 *op_class, u8 *channel);
255 int ieee80211_chaninfo_to_channel(unsigned int freq, enum chan_width chanwidth,
256 int sec_channel, u8 *op_class, u8 *channel);
257 int ieee80211_is_dfs(int freq, const struct hostapd_hw_modes *modes,
258 u16 num_modes);
259 int is_dfs_global_op_class(u8 op_class);
260 bool is_80plus_op_class(u8 op_class);
261 enum phy_type ieee80211_get_phy_type(int freq, int ht, int vht);
262
263 int supp_rates_11b_only(struct ieee802_11_elems *elems);
264 int mb_ies_info_by_ies(struct mb_ies_info *info, const u8 *ies_buf,
265 size_t ies_len);
266 struct wpabuf * mb_ies_by_info(struct mb_ies_info *info);
267
268 const char * fc2str(u16 fc);
269 const char * reason2str(u16 reason);
270 const char * status2str(u16 status);
271
272 struct oper_class_map {
273 enum hostapd_hw_mode mode;
274 u8 op_class;
275 u8 min_chan;
276 u8 max_chan;
277 u8 inc;
278 enum { BW20, BW40PLUS, BW40MINUS, BW40, BW80, BW2160, BW160, BW80P80,
279 BW320, BW4320, BW6480, BW8640} bw;
280 enum { P2P_SUPP, NO_P2P_SUPP } p2p;
281 };
282
283 #ifdef CONFIG_P2P_160M
284 extern const struct oper_class_map *global_op_class;
285 extern const struct oper_class_map global_op_class_data[];
286 extern const struct oper_class_map global_op_class_for_dfs[];
287 #else
288 extern const struct oper_class_map global_op_class[];
289 #endif
290 extern size_t global_op_class_size;
291
292 const u8 * get_ie(const u8 *ies, size_t len, u8 eid);
293 const u8 * get_ie_ext(const u8 *ies, size_t len, u8 ext);
294 const u8 * get_vendor_ie(const u8 *ies, size_t len, u32 vendor_type);
295
296 size_t mbo_add_ie(u8 *buf, size_t len, const u8 *attr, size_t attr_len);
297
298 u16 check_multi_ap_ie(const u8 *multi_ap_ie, size_t multi_ap_len,
299 struct multi_ap_params *multi_ap);
300 size_t add_multi_ap_ie(u8 *buf, size_t len,
301 const struct multi_ap_params *multi_ap);
302
303 struct country_op_class {
304 u8 country_op_class;
305 u8 global_op_class;
306 };
307
308 u8 country_to_global_op_class(const char *country, u8 op_class);
309
310 const struct oper_class_map * get_oper_class(const char *country, u8 op_class);
311 int oper_class_bw_to_int(const struct oper_class_map *map);
312 int center_idx_to_bw_6ghz(u8 idx);
313 bool is_6ghz_freq(int freq);
314 bool is_6ghz_op_class(u8 op_class);
315 bool is_6ghz_psc_frequency(int freq);
316 int get_6ghz_sec_channel(int channel);
317
318 bool is_same_band(int freq1, int freq2);
319 #define IS_2P4GHZ(n) (n >= 2412 && n <= 2484)
320 #define IS_5GHZ(n) (n > 4000 && n < 5895)
321
322 int ieee802_11_parse_candidate_list(const char *pos, u8 *nei_rep,
323 size_t nei_rep_len);
324
325 int ieee802_11_ext_capab(const u8 *ie, unsigned int capab);
326 bool ieee802_11_rsnx_capab_len(const u8 *rsnxe, size_t rsnxe_len,
327 unsigned int capab);
328 bool ieee802_11_rsnx_capab(const u8 *rsnxe, unsigned int capab);
329 int op_class_to_bandwidth(u8 op_class);
330 enum oper_chan_width op_class_to_ch_width(u8 op_class);
331 int chwidth_freq2_to_ch_width(int chwidth, int freq2);
332
333 /* element iteration helpers */
334 #define for_each_element(_elem, _data, _datalen) \
335 for (_elem = (const struct element *) (_data); \
336 (const u8 *) (_data) + (_datalen) - (const u8 *) _elem >= \
337 (int) sizeof(*_elem) && \
338 (const u8 *) (_data) + (_datalen) - (const u8 *) _elem >= \
339 (int) sizeof(*_elem) + _elem->datalen; \
340 _elem = (const struct element *) (_elem->data + _elem->datalen))
341
342 #define for_each_element_id(element, _id, data, datalen) \
343 for_each_element(element, data, datalen) \
344 if (element->id == (_id))
345
346 #define for_each_element_extid(element, extid, _data, _datalen) \
347 for_each_element(element, _data, _datalen) \
348 if (element->id == WLAN_EID_EXTENSION && \
349 element->datalen > 0 && \
350 element->data[0] == (extid))
351
352 #define for_each_subelement(sub, element) \
353 for_each_element(sub, (element)->data, (element)->datalen)
354
355 #define for_each_subelement_id(sub, id, element) \
356 for_each_element_id(sub, id, (element)->data, (element)->datalen)
357
358 #define for_each_subelement_extid(sub, extid, element) \
359 for_each_element_extid(sub, extid, (element)->data, (element)->datalen)
360
361 /**
362 * for_each_element_completed - Determine if element parsing consumed all data
363 * @element: Element pointer after for_each_element() or friends
364 * @data: Same data pointer as passed to for_each_element() or friends
365 * @datalen: Same data length as passed to for_each_element() or friends
366 *
367 * This function returns 1 if all the data was parsed or considered
368 * while walking the elements. Only use this if your for_each_element()
369 * loop cannot be broken out of, otherwise it always returns 0.
370 *
371 * If some data was malformed, this returns %false since the last parsed
372 * element will not fill the whole remaining data.
373 */
for_each_element_completed(const struct element * element,const void * data,size_t datalen)374 static inline int for_each_element_completed(const struct element *element,
375 const void *data, size_t datalen)
376 {
377 return (const u8 *) element == (const u8 *) data + datalen;
378 }
379
380 struct ieee80211_edmg_config;
381
382 void hostapd_encode_edmg_chan(int edmg_enable, u8 edmg_channel,
383 int primary_channel,
384 struct ieee80211_edmg_config *edmg);
385
386 int ieee802_edmg_is_allowed(struct ieee80211_edmg_config allowed,
387 struct ieee80211_edmg_config requested);
388
389 struct wpabuf * ieee802_11_defrag(const u8 *data, size_t len, bool ext_elem);
390 const u8 * get_ml_ie(const u8 *ies, size_t len, u8 type);
391 const u8 * get_basic_mle_mld_addr(const u8 *buf, size_t len);
392
393 int get_oh_max_noss_capa(struct ieee802_11_elems *elements, int parse_rx);
394
395 struct supp_channel_width {
396 u8 is_80p80_supp;
397 u8 is_160_supp;
398 };
399
400 struct supp_channel_width get_oh_support_channel_width(struct ieee802_11_elems *elements);
401
402 enum chan_width get_oh_oper_channel_width(struct ieee802_11_elems *elements);
403
404 enum chan_width get_oh_sta_oper_chan_width(enum chan_width ap_oper_chan_width,
405 struct supp_channel_width sta_supp_chan_width);
406
407 #endif /* IEEE802_11_COMMON_H */
408