blob: fff35b728c359a0f61f40674b855ca052cb4971b [file] [log] [blame]
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001/*
2 * hostapd / IEEE 802.11 Management
Dmitry Shmidt29333592017-01-09 12:27:11 -08003 * Copyright (c) 2002-2017, Jouni Malinen <j@w1.fi>
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004 *
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07007 */
8
9#include "utils/includes.h"
10
11#ifndef CONFIG_NATIVE_WINDOWS
12
13#include "utils/common.h"
14#include "utils/eloop.h"
15#include "crypto/crypto.h"
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -080016#include "crypto/sha256.h"
Dmitry Shmidtd2986c22017-10-23 14:22:09 -070017#include "crypto/sha384.h"
18#include "crypto/sha512.h"
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -080019#include "crypto/random.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070020#include "common/ieee802_11_defs.h"
21#include "common/ieee802_11_common.h"
22#include "common/wpa_ctrl.h"
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -080023#include "common/sae.h"
Hai Shalom021b0b52019-04-10 11:17:58 -070024#include "common/dpp.h"
Hai Shalom74f70d42019-02-11 14:42:39 -080025#include "common/ocv.h"
Hai Shalom81f62d82019-07-22 12:10:00 -070026#include "common/wpa_common.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070027#include "radius/radius.h"
28#include "radius/radius_client.h"
29#include "p2p/p2p.h"
30#include "wps/wps.h"
Dmitry Shmidtd80a4012015-11-05 16:35:40 -080031#include "fst/fst.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070032#include "hostapd.h"
33#include "beacon.h"
34#include "ieee802_11_auth.h"
35#include "sta_info.h"
36#include "ieee802_1x.h"
37#include "wpa_auth.h"
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -080038#include "pmksa_cache_auth.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070039#include "wmm.h"
40#include "ap_list.h"
41#include "accounting.h"
42#include "ap_config.h"
43#include "ap_mlme.h"
44#include "p2p_hostapd.h"
45#include "ap_drv_ops.h"
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -080046#include "wnm_ap.h"
Dmitry Shmidtd80a4012015-11-05 16:35:40 -080047#include "hw_features.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070048#include "ieee802_11.h"
Dmitry Shmidtf21452a2014-02-26 10:55:25 -080049#include "dfs.h"
Dmitry Shmidt57c2d392016-02-23 13:40:19 -080050#include "mbo_ap.h"
Dmitry Shmidt849734c2016-05-27 09:59:01 -070051#include "rrm.h"
Dmitry Shmidtaca489e2016-09-28 15:44:14 -070052#include "taxonomy.h"
Dmitry Shmidtebd93af2017-02-21 13:40:44 -080053#include "fils_hlp.h"
Dmitry Shmidtd2986c22017-10-23 14:22:09 -070054#include "dpp_hostapd.h"
55#include "gas_query_ap.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070056
57
Dmitry Shmidtd2986c22017-10-23 14:22:09 -070058#ifdef CONFIG_FILS
59static struct wpabuf *
60prepare_auth_resp_fils(struct hostapd_data *hapd,
61 struct sta_info *sta, u16 *resp,
62 struct rsn_pmksa_cache_entry *pmksa,
63 struct wpabuf *erp_resp,
64 const u8 *msk, size_t msk_len,
65 int *is_pub);
66#endif /* CONFIG_FILS */
Hai Shalom021b0b52019-04-10 11:17:58 -070067static void handle_auth(struct hostapd_data *hapd,
68 const struct ieee80211_mgmt *mgmt, size_t len,
69 int rssi, int from_queue);
Dmitry Shmidtd2986c22017-10-23 14:22:09 -070070
Hai Shalom74f70d42019-02-11 14:42:39 -080071
72u8 * hostapd_eid_multi_ap(struct hostapd_data *hapd, u8 *eid)
73{
74 u8 multi_ap_val = 0;
75
76 if (!hapd->conf->multi_ap)
77 return eid;
78 if (hapd->conf->multi_ap & BACKHAUL_BSS)
79 multi_ap_val |= MULTI_AP_BACKHAUL_BSS;
80 if (hapd->conf->multi_ap & FRONTHAUL_BSS)
81 multi_ap_val |= MULTI_AP_FRONTHAUL_BSS;
82
83 return eid + add_multi_ap_ie(eid, 9, multi_ap_val);
84}
85
86
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070087u8 * hostapd_eid_supp_rates(struct hostapd_data *hapd, u8 *eid)
88{
89 u8 *pos = eid;
90 int i, num, count;
91
92 if (hapd->iface->current_rates == NULL)
93 return eid;
94
95 *pos++ = WLAN_EID_SUPP_RATES;
96 num = hapd->iface->num_rates;
97 if (hapd->iconf->ieee80211n && hapd->iconf->require_ht)
98 num++;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -080099 if (hapd->iconf->ieee80211ac && hapd->iconf->require_vht)
100 num++;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700101 if (num > 8) {
102 /* rest of the rates are encoded in Extended supported
103 * rates element */
104 num = 8;
105 }
106
107 *pos++ = num;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700108 for (i = 0, count = 0; i < hapd->iface->num_rates && count < num;
109 i++) {
110 count++;
111 *pos = hapd->iface->current_rates[i].rate / 5;
112 if (hapd->iface->current_rates[i].flags & HOSTAPD_RATE_BASIC)
113 *pos |= 0x80;
114 pos++;
115 }
116
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800117 if (hapd->iconf->ieee80211n && hapd->iconf->require_ht && count < 8) {
118 count++;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700119 *pos++ = 0x80 | BSS_MEMBERSHIP_SELECTOR_HT_PHY;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800120 }
121
122 if (hapd->iconf->ieee80211ac && hapd->iconf->require_vht && count < 8) {
123 count++;
124 *pos++ = 0x80 | BSS_MEMBERSHIP_SELECTOR_VHT_PHY;
125 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700126
127 return pos;
128}
129
130
131u8 * hostapd_eid_ext_supp_rates(struct hostapd_data *hapd, u8 *eid)
132{
133 u8 *pos = eid;
134 int i, num, count;
135
136 if (hapd->iface->current_rates == NULL)
137 return eid;
138
139 num = hapd->iface->num_rates;
140 if (hapd->iconf->ieee80211n && hapd->iconf->require_ht)
141 num++;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800142 if (hapd->iconf->ieee80211ac && hapd->iconf->require_vht)
143 num++;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700144 if (num <= 8)
145 return eid;
146 num -= 8;
147
148 *pos++ = WLAN_EID_EXT_SUPP_RATES;
149 *pos++ = num;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700150 for (i = 0, count = 0; i < hapd->iface->num_rates && count < num + 8;
151 i++) {
152 count++;
153 if (count <= 8)
154 continue; /* already in SuppRates IE */
155 *pos = hapd->iface->current_rates[i].rate / 5;
156 if (hapd->iface->current_rates[i].flags & HOSTAPD_RATE_BASIC)
157 *pos |= 0x80;
158 pos++;
159 }
160
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800161 if (hapd->iconf->ieee80211n && hapd->iconf->require_ht) {
162 count++;
163 if (count > 8)
164 *pos++ = 0x80 | BSS_MEMBERSHIP_SELECTOR_HT_PHY;
165 }
166
167 if (hapd->iconf->ieee80211ac && hapd->iconf->require_vht) {
168 count++;
169 if (count > 8)
170 *pos++ = 0x80 | BSS_MEMBERSHIP_SELECTOR_VHT_PHY;
171 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700172
173 return pos;
174}
175
176
Dmitry Shmidt9d9e6022015-04-23 10:34:55 -0700177u16 hostapd_own_capab_info(struct hostapd_data *hapd)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700178{
179 int capab = WLAN_CAPABILITY_ESS;
180 int privacy;
Dmitry Shmidtf21452a2014-02-26 10:55:25 -0800181 int dfs;
Dmitry Shmidt849734c2016-05-27 09:59:01 -0700182 int i;
Dmitry Shmidtf21452a2014-02-26 10:55:25 -0800183
184 /* Check if any of configured channels require DFS */
185 dfs = hostapd_is_dfs_required(hapd->iface);
186 if (dfs < 0) {
187 wpa_printf(MSG_WARNING, "Failed to check if DFS is required; ret=%d",
188 dfs);
189 dfs = 0;
190 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700191
192 if (hapd->iface->num_sta_no_short_preamble == 0 &&
193 hapd->iconf->preamble == SHORT_PREAMBLE)
194 capab |= WLAN_CAPABILITY_SHORT_PREAMBLE;
195
196 privacy = hapd->conf->ssid.wep.keys_set;
197
198 if (hapd->conf->ieee802_1x &&
199 (hapd->conf->default_wep_key_len ||
200 hapd->conf->individual_wep_key_len))
201 privacy = 1;
202
203 if (hapd->conf->wpa)
204 privacy = 1;
205
Dmitry Shmidtf21452a2014-02-26 10:55:25 -0800206#ifdef CONFIG_HS20
207 if (hapd->conf->osen)
208 privacy = 1;
209#endif /* CONFIG_HS20 */
210
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700211 if (privacy)
212 capab |= WLAN_CAPABILITY_PRIVACY;
213
214 if (hapd->iface->current_mode &&
215 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211G &&
216 hapd->iface->num_sta_no_short_slot_time == 0)
217 capab |= WLAN_CAPABILITY_SHORT_SLOT_TIME;
218
Dmitry Shmidtf21452a2014-02-26 10:55:25 -0800219 /*
220 * Currently, Spectrum Management capability bit is set when directly
221 * requested in configuration by spectrum_mgmt_required or when AP is
222 * running on DFS channel.
223 * TODO: Also consider driver support for TPC to set Spectrum Mgmt bit
224 */
225 if (hapd->iface->current_mode &&
226 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211A &&
227 (hapd->iconf->spectrum_mgmt_required || dfs))
228 capab |= WLAN_CAPABILITY_SPECTRUM_MGMT;
229
Dmitry Shmidt849734c2016-05-27 09:59:01 -0700230 for (i = 0; i < RRM_CAPABILITIES_IE_LEN; i++) {
231 if (hapd->conf->radio_measurements[i]) {
232 capab |= IEEE80211_CAP_RRM;
233 break;
234 }
235 }
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800236
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700237 return capab;
238}
239
240
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800241#ifndef CONFIG_NO_RC4
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700242static u16 auth_shared_key(struct hostapd_data *hapd, struct sta_info *sta,
243 u16 auth_transaction, const u8 *challenge,
244 int iswep)
245{
246 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
247 HOSTAPD_LEVEL_DEBUG,
248 "authentication (shared key, transaction %d)",
249 auth_transaction);
250
251 if (auth_transaction == 1) {
252 if (!sta->challenge) {
253 /* Generate a pseudo-random challenge */
254 u8 key[8];
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800255
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700256 sta->challenge = os_zalloc(WLAN_AUTH_CHALLENGE_LEN);
257 if (sta->challenge == NULL)
258 return WLAN_STATUS_UNSPECIFIED_FAILURE;
259
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800260 if (os_get_random(key, sizeof(key)) < 0) {
261 os_free(sta->challenge);
262 sta->challenge = NULL;
263 return WLAN_STATUS_UNSPECIFIED_FAILURE;
264 }
265
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700266 rc4_skip(key, sizeof(key), 0,
267 sta->challenge, WLAN_AUTH_CHALLENGE_LEN);
268 }
269 return 0;
270 }
271
272 if (auth_transaction != 3)
273 return WLAN_STATUS_UNSPECIFIED_FAILURE;
274
275 /* Transaction 3 */
276 if (!iswep || !sta->challenge || !challenge ||
Dmitry Shmidtc2817022014-07-02 10:32:10 -0700277 os_memcmp_const(sta->challenge, challenge,
278 WLAN_AUTH_CHALLENGE_LEN)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700279 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
280 HOSTAPD_LEVEL_INFO,
281 "shared key authentication - invalid "
282 "challenge-response");
283 return WLAN_STATUS_CHALLENGE_FAIL;
284 }
285
286 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
287 HOSTAPD_LEVEL_DEBUG,
288 "authentication OK (shared key)");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700289 sta->flags |= WLAN_STA_AUTH;
290 wpa_auth_sm_event(sta->wpa_sm, WPA_AUTH);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700291 os_free(sta->challenge);
292 sta->challenge = NULL;
293
294 return 0;
295}
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800296#endif /* CONFIG_NO_RC4 */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700297
298
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800299static int send_auth_reply(struct hostapd_data *hapd,
300 const u8 *dst, const u8 *bssid,
301 u16 auth_alg, u16 auth_transaction, u16 resp,
Roshan Pius3a1667e2018-07-03 15:17:14 -0700302 const u8 *ies, size_t ies_len, const char *dbg)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700303{
304 struct ieee80211_mgmt *reply;
305 u8 *buf;
306 size_t rlen;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800307 int reply_res = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700308
309 rlen = IEEE80211_HDRLEN + sizeof(reply->u.auth) + ies_len;
310 buf = os_zalloc(rlen);
311 if (buf == NULL)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800312 return -1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700313
314 reply = (struct ieee80211_mgmt *) buf;
315 reply->frame_control = IEEE80211_FC(WLAN_FC_TYPE_MGMT,
316 WLAN_FC_STYPE_AUTH);
317 os_memcpy(reply->da, dst, ETH_ALEN);
318 os_memcpy(reply->sa, hapd->own_addr, ETH_ALEN);
319 os_memcpy(reply->bssid, bssid, ETH_ALEN);
320
321 reply->u.auth.auth_alg = host_to_le16(auth_alg);
322 reply->u.auth.auth_transaction = host_to_le16(auth_transaction);
323 reply->u.auth.status_code = host_to_le16(resp);
324
325 if (ies && ies_len)
326 os_memcpy(reply->u.auth.variable, ies, ies_len);
327
328 wpa_printf(MSG_DEBUG, "authentication reply: STA=" MACSTR
Roshan Pius3a1667e2018-07-03 15:17:14 -0700329 " auth_alg=%d auth_transaction=%d resp=%d (IE len=%lu) (dbg=%s)",
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700330 MAC2STR(dst), auth_alg, auth_transaction,
Roshan Pius3a1667e2018-07-03 15:17:14 -0700331 resp, (unsigned long) ies_len, dbg);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -0800332 if (hostapd_drv_send_mlme(hapd, reply, rlen, 0) < 0)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800333 wpa_printf(MSG_INFO, "send_auth_reply: send failed");
334 else
335 reply_res = WLAN_STATUS_SUCCESS;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700336
337 os_free(buf);
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800338
339 return reply_res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700340}
341
342
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -0800343#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700344static void handle_auth_ft_finish(void *ctx, const u8 *dst, const u8 *bssid,
345 u16 auth_transaction, u16 status,
346 const u8 *ies, size_t ies_len)
347{
348 struct hostapd_data *hapd = ctx;
349 struct sta_info *sta;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800350 int reply_res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700351
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800352 reply_res = send_auth_reply(hapd, dst, bssid, WLAN_AUTH_FT,
Roshan Pius3a1667e2018-07-03 15:17:14 -0700353 auth_transaction, status, ies, ies_len,
354 "auth-ft-finish");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700355
356 sta = ap_get_sta(hapd, dst);
357 if (sta == NULL)
358 return;
359
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800360 if (sta->added_unassoc && (reply_res != WLAN_STATUS_SUCCESS ||
361 status != WLAN_STATUS_SUCCESS)) {
362 hostapd_drv_sta_remove(hapd, sta->addr);
363 sta->added_unassoc = 0;
364 return;
365 }
366
367 if (status != WLAN_STATUS_SUCCESS)
368 return;
369
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700370 hostapd_logger(hapd, dst, HOSTAPD_MODULE_IEEE80211,
371 HOSTAPD_LEVEL_DEBUG, "authentication OK (FT)");
372 sta->flags |= WLAN_STA_AUTH;
373 mlme_authenticate_indication(hapd, sta);
374}
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -0800375#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700376
377
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800378#ifdef CONFIG_SAE
379
Roshan Pius3a1667e2018-07-03 15:17:14 -0700380static void sae_set_state(struct sta_info *sta, enum sae_state state,
381 const char *reason)
382{
383 wpa_printf(MSG_DEBUG, "SAE: State %s -> %s for peer " MACSTR " (%s)",
384 sae_state_txt(sta->sae->state), sae_state_txt(state),
385 MAC2STR(sta->addr), reason);
386 sta->sae->state = state;
387}
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800388
389
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800390static struct wpabuf * auth_build_sae_commit(struct hostapd_data *hapd,
391 struct sta_info *sta, int update)
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800392{
393 struct wpabuf *buf;
Roshan Pius3a1667e2018-07-03 15:17:14 -0700394 const char *password = NULL;
395 struct sae_password_entry *pw;
396 const char *rx_id = NULL;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800397
Roshan Pius3a1667e2018-07-03 15:17:14 -0700398 if (sta->sae->tmp)
399 rx_id = sta->sae->tmp->pw_id;
400
401 for (pw = hapd->conf->sae_passwords; pw; pw = pw->next) {
402 if (!is_broadcast_ether_addr(pw->peer_addr) &&
403 os_memcmp(pw->peer_addr, sta->addr, ETH_ALEN) != 0)
404 continue;
405 if ((rx_id && !pw->identifier) || (!rx_id && pw->identifier))
406 continue;
407 if (rx_id && pw->identifier &&
408 os_strcmp(rx_id, pw->identifier) != 0)
409 continue;
410 password = pw->password;
411 break;
412 }
Dmitry Shmidtd2986c22017-10-23 14:22:09 -0700413 if (!password)
414 password = hapd->conf->ssid.wpa_passphrase;
415 if (!password) {
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800416 wpa_printf(MSG_DEBUG, "SAE: No password available");
417 return NULL;
418 }
419
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800420 if (update &&
421 sae_prepare_commit(hapd->own_addr, sta->addr,
Roshan Pius3a1667e2018-07-03 15:17:14 -0700422 (u8 *) password, os_strlen(password), rx_id,
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800423 sta->sae) < 0) {
424 wpa_printf(MSG_DEBUG, "SAE: Could not pick PWE");
425 return NULL;
426 }
427
Hai Shalom021b0b52019-04-10 11:17:58 -0700428 if (pw && pw->vlan_id) {
429 if (!sta->sae->tmp) {
430 wpa_printf(MSG_INFO,
431 "SAE: No temporary data allocated - cannot store VLAN ID");
432 return NULL;
433 }
434 sta->sae->tmp->vlan_id = pw->vlan_id;
435 }
436
Roshan Pius3a1667e2018-07-03 15:17:14 -0700437 buf = wpabuf_alloc(SAE_COMMIT_MAX_LEN +
438 (rx_id ? 3 + os_strlen(rx_id) : 0));
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800439 if (buf == NULL)
440 return NULL;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800441 sae_write_commit(sta->sae, buf, sta->sae->tmp ?
Roshan Pius3a1667e2018-07-03 15:17:14 -0700442 sta->sae->tmp->anti_clogging_token : NULL, rx_id);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800443
444 return buf;
445}
446
447
448static struct wpabuf * auth_build_sae_confirm(struct hostapd_data *hapd,
449 struct sta_info *sta)
450{
451 struct wpabuf *buf;
452
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800453 buf = wpabuf_alloc(SAE_CONFIRM_MAX_LEN);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800454 if (buf == NULL)
455 return NULL;
456
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800457 sae_write_confirm(sta->sae, buf);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800458
459 return buf;
460}
461
462
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800463static int auth_sae_send_commit(struct hostapd_data *hapd,
464 struct sta_info *sta,
465 const u8 *bssid, int update)
466{
467 struct wpabuf *data;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800468 int reply_res;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800469
470 data = auth_build_sae_commit(hapd, sta, update);
Roshan Pius3a1667e2018-07-03 15:17:14 -0700471 if (!data && sta->sae->tmp && sta->sae->tmp->pw_id)
472 return WLAN_STATUS_UNKNOWN_PASSWORD_IDENTIFIER;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800473 if (data == NULL)
474 return WLAN_STATUS_UNSPECIFIED_FAILURE;
475
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800476 reply_res = send_auth_reply(hapd, sta->addr, bssid, WLAN_AUTH_SAE, 1,
477 WLAN_STATUS_SUCCESS, wpabuf_head(data),
Roshan Pius3a1667e2018-07-03 15:17:14 -0700478 wpabuf_len(data), "sae-send-commit");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800479
480 wpabuf_free(data);
481
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800482 return reply_res;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800483}
484
485
486static int auth_sae_send_confirm(struct hostapd_data *hapd,
487 struct sta_info *sta,
488 const u8 *bssid)
489{
490 struct wpabuf *data;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800491 int reply_res;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800492
493 data = auth_build_sae_confirm(hapd, sta);
494 if (data == NULL)
495 return WLAN_STATUS_UNSPECIFIED_FAILURE;
496
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800497 reply_res = send_auth_reply(hapd, sta->addr, bssid, WLAN_AUTH_SAE, 2,
498 WLAN_STATUS_SUCCESS, wpabuf_head(data),
Roshan Pius3a1667e2018-07-03 15:17:14 -0700499 wpabuf_len(data), "sae-send-confirm");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800500
501 wpabuf_free(data);
502
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800503 return reply_res;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800504}
505
506
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800507static int use_sae_anti_clogging(struct hostapd_data *hapd)
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800508{
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800509 struct sta_info *sta;
510 unsigned int open = 0;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800511
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800512 if (hapd->conf->sae_anti_clogging_threshold == 0)
513 return 1;
514
515 for (sta = hapd->sta_list; sta; sta = sta->next) {
516 if (!sta->sae)
517 continue;
518 if (sta->sae->state != SAE_COMMITTED &&
519 sta->sae->state != SAE_CONFIRMED)
520 continue;
521 open++;
522 if (open >= hapd->conf->sae_anti_clogging_threshold)
523 return 1;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800524 }
525
Hai Shalom021b0b52019-04-10 11:17:58 -0700526 /* In addition to already existing open SAE sessions, check whether
527 * there are enough pending commit messages in the processing queue to
528 * potentially result in too many open sessions. */
529 if (open + dl_list_len(&hapd->sae_commit_queue) >=
530 hapd->conf->sae_anti_clogging_threshold)
531 return 1;
532
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800533 return 0;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800534}
535
536
Hai Shalom021b0b52019-04-10 11:17:58 -0700537static u8 sae_token_hash(struct hostapd_data *hapd, const u8 *addr)
538{
539 u8 hash[SHA256_MAC_LEN];
540
541 hmac_sha256(hapd->sae_token_key, sizeof(hapd->sae_token_key),
542 addr, ETH_ALEN, hash);
543 return hash[0];
544}
545
546
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800547static int check_sae_token(struct hostapd_data *hapd, const u8 *addr,
548 const u8 *token, size_t token_len)
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800549{
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800550 u8 mac[SHA256_MAC_LEN];
Hai Shalom021b0b52019-04-10 11:17:58 -0700551 const u8 *addrs[2];
552 size_t len[2];
553 u16 token_idx;
554 u8 idx;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800555
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800556 if (token_len != SHA256_MAC_LEN)
557 return -1;
Hai Shalom021b0b52019-04-10 11:17:58 -0700558 idx = sae_token_hash(hapd, addr);
559 token_idx = hapd->sae_pending_token_idx[idx];
560 if (token_idx == 0 || token_idx != WPA_GET_BE16(token)) {
561 wpa_printf(MSG_DEBUG, "SAE: Invalid anti-clogging token from "
562 MACSTR " - token_idx 0x%04x, expected 0x%04x",
563 MAC2STR(addr), WPA_GET_BE16(token), token_idx);
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800564 return -1;
Hai Shalom021b0b52019-04-10 11:17:58 -0700565 }
566
567 addrs[0] = addr;
568 len[0] = ETH_ALEN;
569 addrs[1] = token;
570 len[1] = 2;
571 if (hmac_sha256_vector(hapd->sae_token_key, sizeof(hapd->sae_token_key),
572 2, addrs, len, mac) < 0 ||
573 os_memcmp_const(token + 2, &mac[2], SHA256_MAC_LEN - 2) != 0)
574 return -1;
575
576 hapd->sae_pending_token_idx[idx] = 0; /* invalidate used token */
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800577
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800578 return 0;
579}
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800580
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800581
582static struct wpabuf * auth_build_token_req(struct hostapd_data *hapd,
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800583 int group, const u8 *addr)
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800584{
585 struct wpabuf *buf;
586 u8 *token;
Dmitry Shmidt04f534e2013-12-09 15:50:16 -0800587 struct os_reltime now;
Hai Shalom021b0b52019-04-10 11:17:58 -0700588 u8 idx[2];
589 const u8 *addrs[2];
590 size_t len[2];
591 u8 p_idx;
592 u16 token_idx;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800593
Dmitry Shmidt04f534e2013-12-09 15:50:16 -0800594 os_get_reltime(&now);
595 if (!os_reltime_initialized(&hapd->last_sae_token_key_update) ||
Hai Shalom021b0b52019-04-10 11:17:58 -0700596 os_reltime_expired(&now, &hapd->last_sae_token_key_update, 60) ||
597 hapd->sae_token_idx == 0xffff) {
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800598 if (random_get_bytes(hapd->sae_token_key,
599 sizeof(hapd->sae_token_key)) < 0)
600 return NULL;
601 wpa_hexdump(MSG_DEBUG, "SAE: Updated token key",
602 hapd->sae_token_key, sizeof(hapd->sae_token_key));
Dmitry Shmidt04f534e2013-12-09 15:50:16 -0800603 hapd->last_sae_token_key_update = now;
Hai Shalom021b0b52019-04-10 11:17:58 -0700604 hapd->sae_token_idx = 0;
605 os_memset(hapd->sae_pending_token_idx, 0,
606 sizeof(hapd->sae_pending_token_idx));
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800607 }
608
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800609 buf = wpabuf_alloc(sizeof(le16) + SHA256_MAC_LEN);
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800610 if (buf == NULL)
611 return NULL;
612
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800613 wpabuf_put_le16(buf, group); /* Finite Cyclic Group */
614
Hai Shalom021b0b52019-04-10 11:17:58 -0700615 p_idx = sae_token_hash(hapd, addr);
616 token_idx = hapd->sae_pending_token_idx[p_idx];
617 if (!token_idx) {
618 hapd->sae_token_idx++;
619 token_idx = hapd->sae_token_idx;
620 hapd->sae_pending_token_idx[p_idx] = token_idx;
621 }
622 WPA_PUT_BE16(idx, token_idx);
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800623 token = wpabuf_put(buf, SHA256_MAC_LEN);
Hai Shalom021b0b52019-04-10 11:17:58 -0700624 addrs[0] = addr;
625 len[0] = ETH_ALEN;
626 addrs[1] = idx;
627 len[1] = sizeof(idx);
628 if (hmac_sha256_vector(hapd->sae_token_key, sizeof(hapd->sae_token_key),
629 2, addrs, len, token) < 0) {
630 wpabuf_free(buf);
631 return NULL;
632 }
633 WPA_PUT_BE16(token, token_idx);
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800634
635 return buf;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800636}
637
638
Roshan Pius3a1667e2018-07-03 15:17:14 -0700639static int sae_check_big_sync(struct hostapd_data *hapd, struct sta_info *sta)
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800640{
Roshan Pius3a1667e2018-07-03 15:17:14 -0700641 if (sta->sae->sync > hapd->conf->sae_sync) {
642 sae_set_state(sta, SAE_NOTHING, "Sync > dot11RSNASAESync");
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800643 sta->sae->sync = 0;
644 return -1;
645 }
646 return 0;
647}
648
649
650static void auth_sae_retransmit_timer(void *eloop_ctx, void *eloop_data)
651{
652 struct hostapd_data *hapd = eloop_ctx;
653 struct sta_info *sta = eloop_data;
654 int ret;
655
Roshan Pius3a1667e2018-07-03 15:17:14 -0700656 if (sae_check_big_sync(hapd, sta))
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800657 return;
658 sta->sae->sync++;
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -0700659 wpa_printf(MSG_DEBUG, "SAE: Auth SAE retransmit timer for " MACSTR
Roshan Pius3a1667e2018-07-03 15:17:14 -0700660 " (sync=%d state=%s)",
661 MAC2STR(sta->addr), sta->sae->sync,
662 sae_state_txt(sta->sae->state));
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800663
664 switch (sta->sae->state) {
665 case SAE_COMMITTED:
666 ret = auth_sae_send_commit(hapd, sta, hapd->own_addr, 0);
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800667 eloop_register_timeout(0,
668 hapd->dot11RSNASAERetransPeriod * 1000,
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800669 auth_sae_retransmit_timer, hapd, sta);
670 break;
671 case SAE_CONFIRMED:
672 ret = auth_sae_send_confirm(hapd, sta, hapd->own_addr);
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800673 eloop_register_timeout(0,
674 hapd->dot11RSNASAERetransPeriod * 1000,
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800675 auth_sae_retransmit_timer, hapd, sta);
676 break;
677 default:
678 ret = -1;
679 break;
680 }
681
682 if (ret != WLAN_STATUS_SUCCESS)
683 wpa_printf(MSG_INFO, "SAE: Failed to retransmit: ret=%d", ret);
684}
685
686
687void sae_clear_retransmit_timer(struct hostapd_data *hapd, struct sta_info *sta)
688{
689 eloop_cancel_timeout(auth_sae_retransmit_timer, hapd, sta);
690}
691
692
693static void sae_set_retransmit_timer(struct hostapd_data *hapd,
694 struct sta_info *sta)
695{
696 if (!(hapd->conf->mesh & MESH_ENABLED))
697 return;
698
699 eloop_cancel_timeout(auth_sae_retransmit_timer, hapd, sta);
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800700 eloop_register_timeout(0, hapd->dot11RSNASAERetransPeriod * 1000,
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800701 auth_sae_retransmit_timer, hapd, sta);
702}
703
704
Hai Shalom5f92bc92019-04-18 11:54:11 -0700705static void sae_sme_send_external_auth_status(struct hostapd_data *hapd,
706 struct sta_info *sta, u16 status)
707{
708 struct external_auth params;
709
710 os_memset(&params, 0, sizeof(params));
711 params.status = status;
712 params.bssid = sta->addr;
Hai Shalom81f62d82019-07-22 12:10:00 -0700713 if (status == WLAN_STATUS_SUCCESS && sta->sae &&
714 !hapd->conf->disable_pmksa_caching)
Hai Shalom5f92bc92019-04-18 11:54:11 -0700715 params.pmkid = sta->sae->pmkid;
716
717 hostapd_drv_send_external_auth_status(hapd, &params);
718}
719
720
Dmitry Shmidte4663042016-04-04 10:07:49 -0700721void sae_accept_sta(struct hostapd_data *hapd, struct sta_info *sta)
722{
Hai Shalom021b0b52019-04-10 11:17:58 -0700723#ifndef CONFIG_NO_VLAN
724 struct vlan_description vlan_desc;
725
726 if (sta->sae->tmp && sta->sae->tmp->vlan_id > 0) {
727 wpa_printf(MSG_DEBUG, "SAE: Assign STA " MACSTR
728 " to VLAN ID %d",
729 MAC2STR(sta->addr), sta->sae->tmp->vlan_id);
730
731 os_memset(&vlan_desc, 0, sizeof(vlan_desc));
732 vlan_desc.notempty = 1;
733 vlan_desc.untagged = sta->sae->tmp->vlan_id;
734 if (!hostapd_vlan_valid(hapd->conf->vlan, &vlan_desc)) {
735 wpa_printf(MSG_INFO,
736 "Invalid VLAN ID %d in sae_password",
737 sta->sae->tmp->vlan_id);
738 return;
739 }
740
741 if (ap_sta_set_vlan(hapd, sta, &vlan_desc) < 0 ||
742 ap_sta_bind_vlan(hapd, sta) < 0) {
743 wpa_printf(MSG_INFO,
744 "Failed to assign VLAN ID %d from sae_password to "
745 MACSTR, sta->sae->tmp->vlan_id,
746 MAC2STR(sta->addr));
747 return;
748 }
749 }
750#endif /* CONFIG_NO_VLAN */
751
Dmitry Shmidte4663042016-04-04 10:07:49 -0700752 sta->flags |= WLAN_STA_AUTH;
753 sta->auth_alg = WLAN_AUTH_SAE;
754 mlme_authenticate_indication(hapd, sta);
755 wpa_auth_sm_event(sta->wpa_sm, WPA_AUTH);
Roshan Pius3a1667e2018-07-03 15:17:14 -0700756 sae_set_state(sta, SAE_ACCEPTED, "Accept Confirm");
Dmitry Shmidte4663042016-04-04 10:07:49 -0700757 wpa_auth_pmksa_add_sae(hapd->wpa_auth, sta->addr,
758 sta->sae->pmk, sta->sae->pmkid);
Hai Shalom5f92bc92019-04-18 11:54:11 -0700759 sae_sme_send_external_auth_status(hapd, sta, WLAN_STATUS_SUCCESS);
Dmitry Shmidte4663042016-04-04 10:07:49 -0700760}
761
762
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800763static int sae_sm_step(struct hostapd_data *hapd, struct sta_info *sta,
Hai Shalom5f92bc92019-04-18 11:54:11 -0700764 const u8 *bssid, u8 auth_transaction, int allow_reuse,
765 int *sta_removed)
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800766{
767 int ret;
768
Hai Shalom5f92bc92019-04-18 11:54:11 -0700769 *sta_removed = 0;
770
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800771 if (auth_transaction != 1 && auth_transaction != 2)
772 return WLAN_STATUS_UNSPECIFIED_FAILURE;
773
Roshan Pius3a1667e2018-07-03 15:17:14 -0700774 wpa_printf(MSG_DEBUG, "SAE: Peer " MACSTR " state=%s auth_trans=%u",
775 MAC2STR(sta->addr), sae_state_txt(sta->sae->state),
776 auth_transaction);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800777 switch (sta->sae->state) {
778 case SAE_NOTHING:
779 if (auth_transaction == 1) {
Hai Shalom021b0b52019-04-10 11:17:58 -0700780 ret = auth_sae_send_commit(hapd, sta, bssid,
781 !allow_reuse);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800782 if (ret)
783 return ret;
Roshan Pius3a1667e2018-07-03 15:17:14 -0700784 sae_set_state(sta, SAE_COMMITTED, "Sent Commit");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800785
786 if (sae_process_commit(sta->sae) < 0)
787 return WLAN_STATUS_UNSPECIFIED_FAILURE;
788
789 /*
790 * In mesh case, both Commit and Confirm can be sent
791 * immediately. In infrastructure BSS, only a single
792 * Authentication frame (Commit) is expected from the AP
793 * here and the second one (Confirm) will be sent once
794 * the STA has sent its second Authentication frame
795 * (Confirm).
796 */
797 if (hapd->conf->mesh & MESH_ENABLED) {
798 /*
799 * Send both Commit and Confirm immediately
800 * based on SAE finite state machine
801 * Nothing -> Confirm transition.
802 */
803 ret = auth_sae_send_confirm(hapd, sta, bssid);
804 if (ret)
805 return ret;
Roshan Pius3a1667e2018-07-03 15:17:14 -0700806 sae_set_state(sta, SAE_CONFIRMED,
807 "Sent Confirm (mesh)");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800808 } else {
809 /*
810 * For infrastructure BSS, send only the Commit
811 * message now to get alternating sequence of
812 * Authentication frames between the AP and STA.
813 * Confirm will be sent in
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -0800814 * Committed -> Confirmed/Accepted transition
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800815 * when receiving Confirm from STA.
816 */
817 }
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800818 sta->sae->sync = 0;
819 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800820 } else {
821 hostapd_logger(hapd, sta->addr,
822 HOSTAPD_MODULE_IEEE80211,
823 HOSTAPD_LEVEL_DEBUG,
824 "SAE confirm before commit");
825 }
826 break;
827 case SAE_COMMITTED:
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800828 sae_clear_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800829 if (auth_transaction == 1) {
830 if (sae_process_commit(sta->sae) < 0)
831 return WLAN_STATUS_UNSPECIFIED_FAILURE;
832
833 ret = auth_sae_send_confirm(hapd, sta, bssid);
834 if (ret)
835 return ret;
Roshan Pius3a1667e2018-07-03 15:17:14 -0700836 sae_set_state(sta, SAE_CONFIRMED, "Sent Confirm");
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800837 sta->sae->sync = 0;
838 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800839 } else if (hapd->conf->mesh & MESH_ENABLED) {
840 /*
841 * In mesh case, follow SAE finite state machine and
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800842 * send Commit now, if sync count allows.
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800843 */
Roshan Pius3a1667e2018-07-03 15:17:14 -0700844 if (sae_check_big_sync(hapd, sta))
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800845 return WLAN_STATUS_SUCCESS;
846 sta->sae->sync++;
847
Dmitry Shmidt1d755d02015-04-28 10:34:29 -0700848 ret = auth_sae_send_commit(hapd, sta, bssid, 0);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800849 if (ret)
850 return ret;
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800851
852 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800853 } else {
854 /*
855 * For instructure BSS, send the postponed Confirm from
856 * Nothing -> Confirmed transition that was reduced to
857 * Nothing -> Committed above.
858 */
859 ret = auth_sae_send_confirm(hapd, sta, bssid);
860 if (ret)
861 return ret;
862
Roshan Pius3a1667e2018-07-03 15:17:14 -0700863 sae_set_state(sta, SAE_CONFIRMED, "Sent Confirm");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800864
865 /*
866 * Since this was triggered on Confirm RX, run another
867 * step to get to Accepted without waiting for
868 * additional events.
869 */
Hai Shalom021b0b52019-04-10 11:17:58 -0700870 return sae_sm_step(hapd, sta, bssid, auth_transaction,
Hai Shalom5f92bc92019-04-18 11:54:11 -0700871 0, sta_removed);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800872 }
873 break;
874 case SAE_CONFIRMED:
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800875 sae_clear_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800876 if (auth_transaction == 1) {
Roshan Pius3a1667e2018-07-03 15:17:14 -0700877 if (sae_check_big_sync(hapd, sta))
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800878 return WLAN_STATUS_SUCCESS;
879 sta->sae->sync++;
880
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800881 ret = auth_sae_send_commit(hapd, sta, bssid, 1);
882 if (ret)
883 return ret;
884
885 if (sae_process_commit(sta->sae) < 0)
886 return WLAN_STATUS_UNSPECIFIED_FAILURE;
887
888 ret = auth_sae_send_confirm(hapd, sta, bssid);
889 if (ret)
890 return ret;
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800891
892 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800893 } else {
Roshan Pius3a1667e2018-07-03 15:17:14 -0700894 sta->sae->send_confirm = 0xffff;
Dmitry Shmidte4663042016-04-04 10:07:49 -0700895 sae_accept_sta(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800896 }
897 break;
898 case SAE_ACCEPTED:
Roshan Pius3a1667e2018-07-03 15:17:14 -0700899 if (auth_transaction == 1 &&
900 (hapd->conf->mesh & MESH_ENABLED)) {
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800901 wpa_printf(MSG_DEBUG, "SAE: remove the STA (" MACSTR
902 ") doing reauthentication",
903 MAC2STR(sta->addr));
Dmitry Shmidte4663042016-04-04 10:07:49 -0700904 wpa_auth_pmksa_remove(hapd->wpa_auth, sta->addr);
Hai Shalom5f92bc92019-04-18 11:54:11 -0700905 ap_free_sta(hapd, sta);
906 *sta_removed = 1;
Roshan Pius3a1667e2018-07-03 15:17:14 -0700907 } else if (auth_transaction == 1) {
908 wpa_printf(MSG_DEBUG, "SAE: Start reauthentication");
909 ret = auth_sae_send_commit(hapd, sta, bssid, 1);
910 if (ret)
911 return ret;
912 sae_set_state(sta, SAE_COMMITTED, "Sent Commit");
913
914 if (sae_process_commit(sta->sae) < 0)
915 return WLAN_STATUS_UNSPECIFIED_FAILURE;
916 sta->sae->sync = 0;
917 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800918 } else {
Roshan Pius3a1667e2018-07-03 15:17:14 -0700919 if (sae_check_big_sync(hapd, sta))
Dmitry Shmidtff787d52015-01-12 13:01:47 -0800920 return WLAN_STATUS_SUCCESS;
921 sta->sae->sync++;
922
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800923 ret = auth_sae_send_confirm(hapd, sta, bssid);
924 sae_clear_temp_data(sta->sae);
925 if (ret)
926 return ret;
927 }
928 break;
929 default:
930 wpa_printf(MSG_ERROR, "SAE: invalid state %d",
931 sta->sae->state);
932 return WLAN_STATUS_UNSPECIFIED_FAILURE;
933 }
934 return WLAN_STATUS_SUCCESS;
935}
936
937
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -0700938static void sae_pick_next_group(struct hostapd_data *hapd, struct sta_info *sta)
939{
940 struct sae_data *sae = sta->sae;
941 int i, *groups = hapd->conf->sae_groups;
Hai Shalom021b0b52019-04-10 11:17:58 -0700942 int default_groups[] = { 19, 0 };
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -0700943
944 if (sae->state != SAE_COMMITTED)
945 return;
946
947 wpa_printf(MSG_DEBUG, "SAE: Previously selected group: %d", sae->group);
948
Hai Shalom021b0b52019-04-10 11:17:58 -0700949 if (!groups)
950 groups = default_groups;
951 for (i = 0; groups[i] > 0; i++) {
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -0700952 if (sae->group == groups[i])
953 break;
954 }
955
Hai Shalom021b0b52019-04-10 11:17:58 -0700956 if (groups[i] <= 0) {
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -0700957 wpa_printf(MSG_DEBUG,
958 "SAE: Previously selected group not found from the current configuration");
959 return;
960 }
961
962 for (;;) {
963 i++;
964 if (groups[i] <= 0) {
965 wpa_printf(MSG_DEBUG,
966 "SAE: No alternative group enabled");
967 return;
968 }
969
970 if (sae_set_group(sae, groups[i]) < 0)
971 continue;
972
973 break;
974 }
975 wpa_printf(MSG_DEBUG, "SAE: Selected new group: %d", groups[i]);
976}
977
978
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800979static void handle_auth_sae(struct hostapd_data *hapd, struct sta_info *sta,
980 const struct ieee80211_mgmt *mgmt, size_t len,
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -0800981 u16 auth_transaction, u16 status_code)
Dmitry Shmidtd5e49232012-12-03 15:08:10 -0800982{
Dmitry Shmidt57c2d392016-02-23 13:40:19 -0800983 int resp = WLAN_STATUS_SUCCESS;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800984 struct wpabuf *data = NULL;
Hai Shalom021b0b52019-04-10 11:17:58 -0700985 int *groups = hapd->conf->sae_groups;
986 int default_groups[] = { 19, 0 };
987 const u8 *pos, *end;
Hai Shalom5f92bc92019-04-18 11:54:11 -0700988 int sta_removed = 0;
Hai Shalom021b0b52019-04-10 11:17:58 -0700989
990 if (!groups)
991 groups = default_groups;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -0800992
Dmitry Shmidtd2986c22017-10-23 14:22:09 -0700993#ifdef CONFIG_TESTING_OPTIONS
994 if (hapd->conf->sae_reflection_attack && auth_transaction == 1) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -0700995 wpa_printf(MSG_DEBUG, "SAE: TESTING - reflection attack");
996 pos = mgmt->u.auth.variable;
997 end = ((const u8 *) mgmt) + len;
998 send_auth_reply(hapd, mgmt->sa, mgmt->bssid, WLAN_AUTH_SAE,
Roshan Pius3a1667e2018-07-03 15:17:14 -0700999 auth_transaction, resp, pos, end - pos,
1000 "auth-sae-reflection-attack");
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001001 goto remove_sta;
1002 }
1003
1004 if (hapd->conf->sae_commit_override && auth_transaction == 1) {
1005 wpa_printf(MSG_DEBUG, "SAE: TESTING - commit override");
1006 send_auth_reply(hapd, mgmt->sa, mgmt->bssid, WLAN_AUTH_SAE,
1007 auth_transaction, resp,
1008 wpabuf_head(hapd->conf->sae_commit_override),
Roshan Pius3a1667e2018-07-03 15:17:14 -07001009 wpabuf_len(hapd->conf->sae_commit_override),
1010 "sae-commit-override");
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001011 goto remove_sta;
1012 }
1013#endif /* CONFIG_TESTING_OPTIONS */
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001014 if (!sta->sae) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001015 if (auth_transaction != 1 ||
1016 status_code != WLAN_STATUS_SUCCESS) {
1017 resp = -1;
1018 goto remove_sta;
1019 }
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001020 sta->sae = os_zalloc(sizeof(*sta->sae));
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001021 if (!sta->sae) {
1022 resp = -1;
1023 goto remove_sta;
1024 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07001025 sae_set_state(sta, SAE_NOTHING, "Init");
Dmitry Shmidtff787d52015-01-12 13:01:47 -08001026 sta->sae->sync = 0;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001027 }
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001028
Dmitry Shmidte4663042016-04-04 10:07:49 -07001029 if (sta->mesh_sae_pmksa_caching) {
1030 wpa_printf(MSG_DEBUG,
1031 "SAE: Cancel use of mesh PMKSA caching because peer starts SAE authentication");
1032 wpa_auth_pmksa_remove(hapd->wpa_auth, sta->addr);
1033 sta->mesh_sae_pmksa_caching = 0;
1034 }
1035
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001036 if (auth_transaction == 1) {
Hai Shalom021b0b52019-04-10 11:17:58 -07001037 const u8 *token = NULL;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001038 size_t token_len = 0;
Hai Shalom021b0b52019-04-10 11:17:58 -07001039 int allow_reuse = 0;
1040
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001041 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
1042 HOSTAPD_LEVEL_DEBUG,
Hai Shalom81f62d82019-07-22 12:10:00 -07001043 "start SAE authentication (RX commit, status=%u (%s))",
1044 status_code, status2str(status_code));
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001045
1046 if ((hapd->conf->mesh & MESH_ENABLED) &&
1047 status_code == WLAN_STATUS_ANTI_CLOGGING_TOKEN_REQ &&
1048 sta->sae->tmp) {
1049 pos = mgmt->u.auth.variable;
1050 end = ((const u8 *) mgmt) + len;
1051 if (pos + sizeof(le16) > end) {
1052 wpa_printf(MSG_ERROR,
1053 "SAE: Too short anti-clogging token request");
1054 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1055 goto reply;
1056 }
Hai Shalom021b0b52019-04-10 11:17:58 -07001057 resp = sae_group_allowed(sta->sae, groups,
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001058 WPA_GET_LE16(pos));
1059 if (resp != WLAN_STATUS_SUCCESS) {
1060 wpa_printf(MSG_ERROR,
1061 "SAE: Invalid group in anti-clogging token request");
1062 goto reply;
1063 }
1064 pos += sizeof(le16);
1065
1066 wpabuf_free(sta->sae->tmp->anti_clogging_token);
1067 sta->sae->tmp->anti_clogging_token =
1068 wpabuf_alloc_copy(pos, end - pos);
1069 if (sta->sae->tmp->anti_clogging_token == NULL) {
1070 wpa_printf(MSG_ERROR,
1071 "SAE: Failed to alloc for anti-clogging token");
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001072 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1073 goto remove_sta;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001074 }
1075
1076 /*
1077 * IEEE Std 802.11-2012, 11.3.8.6.4: If the Status code
1078 * is 76, a new Commit Message shall be constructed
1079 * with the Anti-Clogging Token from the received
1080 * Authentication frame, and the commit-scalar and
1081 * COMMIT-ELEMENT previously sent.
1082 */
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001083 resp = auth_sae_send_commit(hapd, sta, mgmt->bssid, 0);
1084 if (resp != WLAN_STATUS_SUCCESS) {
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001085 wpa_printf(MSG_ERROR,
1086 "SAE: Failed to send commit message");
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001087 goto remove_sta;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001088 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07001089 sae_set_state(sta, SAE_COMMITTED,
1090 "Sent Commit (anti-clogging token case in mesh)");
Dmitry Shmidtff787d52015-01-12 13:01:47 -08001091 sta->sae->sync = 0;
1092 sae_set_retransmit_timer(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001093 return;
1094 }
1095
Dmitry Shmidtd5ab1b52016-06-21 12:38:41 -07001096 if ((hapd->conf->mesh & MESH_ENABLED) &&
1097 status_code ==
1098 WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED &&
1099 sta->sae->tmp) {
1100 wpa_printf(MSG_DEBUG,
1101 "SAE: Peer did not accept our SAE group");
1102 sae_pick_next_group(hapd, sta);
1103 goto remove_sta;
1104 }
1105
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001106 if (status_code != WLAN_STATUS_SUCCESS)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001107 goto remove_sta;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001108
Roshan Pius3a1667e2018-07-03 15:17:14 -07001109 if (!(hapd->conf->mesh & MESH_ENABLED) &&
1110 sta->sae->state == SAE_COMMITTED) {
1111 /* This is needed in the infrastructure BSS case to
1112 * address a sequence where a STA entry may remain in
1113 * hostapd across two attempts to do SAE authentication
1114 * by the same STA. The second attempt may end up trying
1115 * to use a different group and that would not be
1116 * allowed if we remain in Committed state with the
1117 * previously set parameters. */
Hai Shalom021b0b52019-04-10 11:17:58 -07001118 pos = mgmt->u.auth.variable;
1119 end = ((const u8 *) mgmt) + len;
1120 if (end - pos >= (int) sizeof(le16) &&
1121 sae_group_allowed(sta->sae, groups,
1122 WPA_GET_LE16(pos)) ==
1123 WLAN_STATUS_SUCCESS) {
1124 /* Do not waste resources deriving the same PWE
1125 * again since the same group is reused. */
1126 sae_set_state(sta, SAE_NOTHING,
1127 "Allow previous PWE to be reused");
1128 allow_reuse = 1;
1129 } else {
1130 sae_set_state(sta, SAE_NOTHING,
1131 "Clear existing state to allow restart");
1132 sae_clear_data(sta->sae);
1133 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07001134 }
1135
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001136 resp = sae_parse_commit(sta->sae, mgmt->u.auth.variable,
1137 ((const u8 *) mgmt) + len -
1138 mgmt->u.auth.variable, &token,
Hai Shalom021b0b52019-04-10 11:17:58 -07001139 &token_len, groups);
Dmitry Shmidt41712582015-06-29 11:02:15 -07001140 if (resp == SAE_SILENTLY_DISCARD) {
1141 wpa_printf(MSG_DEBUG,
1142 "SAE: Drop commit message from " MACSTR " due to reflection attack",
1143 MAC2STR(sta->addr));
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001144 goto remove_sta;
Dmitry Shmidt41712582015-06-29 11:02:15 -07001145 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07001146
1147 if (resp == WLAN_STATUS_UNKNOWN_PASSWORD_IDENTIFIER) {
1148 wpa_msg(hapd->msg_ctx, MSG_INFO,
1149 WPA_EVENT_SAE_UNKNOWN_PASSWORD_IDENTIFIER
1150 MACSTR, MAC2STR(sta->addr));
1151 sae_clear_retransmit_timer(hapd, sta);
1152 sae_set_state(sta, SAE_NOTHING,
1153 "Unknown Password Identifier");
1154 goto remove_sta;
1155 }
1156
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001157 if (token && check_sae_token(hapd, sta->addr, token, token_len)
1158 < 0) {
1159 wpa_printf(MSG_DEBUG, "SAE: Drop commit message with "
1160 "incorrect token from " MACSTR,
1161 MAC2STR(sta->addr));
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001162 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1163 goto remove_sta;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001164 }
1165
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001166 if (resp != WLAN_STATUS_SUCCESS)
1167 goto reply;
1168
Hai Shalom021b0b52019-04-10 11:17:58 -07001169 if (!token && use_sae_anti_clogging(hapd) && !allow_reuse) {
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001170 wpa_printf(MSG_DEBUG,
1171 "SAE: Request anti-clogging token from "
1172 MACSTR, MAC2STR(sta->addr));
1173 data = auth_build_token_req(hapd, sta->sae->group,
1174 sta->addr);
1175 resp = WLAN_STATUS_ANTI_CLOGGING_TOKEN_REQ;
1176 if (hapd->conf->mesh & MESH_ENABLED)
Roshan Pius3a1667e2018-07-03 15:17:14 -07001177 sae_set_state(sta, SAE_NOTHING,
1178 "Request anti-clogging token case in mesh");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001179 goto reply;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001180 }
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001181
Hai Shalom021b0b52019-04-10 11:17:58 -07001182 resp = sae_sm_step(hapd, sta, mgmt->bssid, auth_transaction,
Hai Shalom5f92bc92019-04-18 11:54:11 -07001183 allow_reuse, &sta_removed);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001184 } else if (auth_transaction == 2) {
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001185 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
1186 HOSTAPD_LEVEL_DEBUG,
Hai Shalom81f62d82019-07-22 12:10:00 -07001187 "SAE authentication (RX confirm, status=%u (%s))",
1188 status_code, status2str(status_code));
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001189 if (status_code != WLAN_STATUS_SUCCESS)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001190 goto remove_sta;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001191 if (sta->sae->state >= SAE_CONFIRMED ||
1192 !(hapd->conf->mesh & MESH_ENABLED)) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07001193 const u8 *var;
1194 size_t var_len;
1195 u16 peer_send_confirm;
1196
1197 var = mgmt->u.auth.variable;
1198 var_len = ((u8 *) mgmt) + len - mgmt->u.auth.variable;
1199 if (var_len < 2) {
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001200 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001201 goto reply;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08001202 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07001203
1204 peer_send_confirm = WPA_GET_LE16(var);
1205
1206 if (sta->sae->state == SAE_ACCEPTED &&
1207 (peer_send_confirm <= sta->sae->rc ||
1208 peer_send_confirm == 0xffff)) {
1209 wpa_printf(MSG_DEBUG,
1210 "SAE: Silently ignore unexpected Confirm from peer "
1211 MACSTR
1212 " (peer-send-confirm=%u Rc=%u)",
1213 MAC2STR(sta->addr),
1214 peer_send_confirm, sta->sae->rc);
1215 return;
1216 }
1217
1218 if (sae_check_confirm(sta->sae, var, var_len) < 0) {
1219 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1220 goto reply;
1221 }
1222 sta->sae->rc = peer_send_confirm;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001223 }
Hai Shalom5f92bc92019-04-18 11:54:11 -07001224 resp = sae_sm_step(hapd, sta, mgmt->bssid, auth_transaction, 0,
1225 &sta_removed);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001226 } else {
1227 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
1228 HOSTAPD_LEVEL_DEBUG,
Hai Shalom81f62d82019-07-22 12:10:00 -07001229 "unexpected SAE authentication transaction %u (status=%u (%s))",
1230 auth_transaction, status_code,
1231 status2str(status_code));
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001232 if (status_code != WLAN_STATUS_SUCCESS)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001233 goto remove_sta;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001234 resp = WLAN_STATUS_UNKNOWN_AUTH_TRANSACTION;
1235 }
1236
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001237reply:
Hai Shalom5f92bc92019-04-18 11:54:11 -07001238 if (!sta_removed && resp != WLAN_STATUS_SUCCESS) {
Hai Shalom021b0b52019-04-10 11:17:58 -07001239 pos = mgmt->u.auth.variable;
1240 end = ((const u8 *) mgmt) + len;
1241
1242 /* Copy the Finite Cyclic Group field from the request if we
1243 * rejected it as unsupported group. */
1244 if (resp == WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED &&
1245 !data && end - pos >= 2)
1246 data = wpabuf_alloc_copy(pos, 2);
1247
Hai Shalom5f92bc92019-04-18 11:54:11 -07001248 sae_sme_send_external_auth_status(hapd, sta, resp);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001249 send_auth_reply(hapd, mgmt->sa, mgmt->bssid, WLAN_AUTH_SAE,
1250 auth_transaction, resp,
1251 data ? wpabuf_head(data) : (u8 *) "",
Roshan Pius3a1667e2018-07-03 15:17:14 -07001252 data ? wpabuf_len(data) : 0, "auth-sae");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08001253 }
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001254
1255remove_sta:
Hai Shalom5f92bc92019-04-18 11:54:11 -07001256 if (!sta_removed && sta->added_unassoc &&
1257 (resp != WLAN_STATUS_SUCCESS ||
1258 status_code != WLAN_STATUS_SUCCESS)) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08001259 hostapd_drv_sta_remove(hapd, sta->addr);
1260 sta->added_unassoc = 0;
1261 }
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001262 wpabuf_free(data);
1263}
Dmitry Shmidtff787d52015-01-12 13:01:47 -08001264
1265
1266/**
1267 * auth_sae_init_committed - Send COMMIT and start SAE in committed state
1268 * @hapd: BSS data for the device initiating the authentication
1269 * @sta: the peer to which commit authentication frame is sent
1270 *
1271 * This function implements Init event handling (IEEE Std 802.11-2012,
1272 * 11.3.8.6.3) in which initial COMMIT message is sent. Prior to calling, the
1273 * sta->sae structure should be initialized appropriately via a call to
1274 * sae_prepare_commit().
1275 */
1276int auth_sae_init_committed(struct hostapd_data *hapd, struct sta_info *sta)
1277{
1278 int ret;
1279
1280 if (!sta->sae || !sta->sae->tmp)
1281 return -1;
1282
1283 if (sta->sae->state != SAE_NOTHING)
1284 return -1;
1285
1286 ret = auth_sae_send_commit(hapd, sta, hapd->own_addr, 0);
1287 if (ret)
1288 return -1;
1289
Roshan Pius3a1667e2018-07-03 15:17:14 -07001290 sae_set_state(sta, SAE_COMMITTED, "Init and sent commit");
Dmitry Shmidtff787d52015-01-12 13:01:47 -08001291 sta->sae->sync = 0;
1292 sae_set_retransmit_timer(hapd, sta);
1293
1294 return 0;
1295}
1296
Hai Shalom021b0b52019-04-10 11:17:58 -07001297
1298void auth_sae_process_commit(void *eloop_ctx, void *user_ctx)
1299{
1300 struct hostapd_data *hapd = eloop_ctx;
1301 struct hostapd_sae_commit_queue *q;
1302 unsigned int queue_len;
1303
1304 q = dl_list_first(&hapd->sae_commit_queue,
1305 struct hostapd_sae_commit_queue, list);
1306 if (!q)
1307 return;
1308 wpa_printf(MSG_DEBUG,
1309 "SAE: Process next available message from queue");
1310 dl_list_del(&q->list);
1311 handle_auth(hapd, (const struct ieee80211_mgmt *) q->msg, q->len,
1312 q->rssi, 1);
1313 os_free(q);
1314
1315 if (eloop_is_timeout_registered(auth_sae_process_commit, hapd, NULL))
1316 return;
1317 queue_len = dl_list_len(&hapd->sae_commit_queue);
1318 eloop_register_timeout(0, queue_len * 10000, auth_sae_process_commit,
1319 hapd, NULL);
1320}
1321
1322
1323static void auth_sae_queue(struct hostapd_data *hapd,
1324 const struct ieee80211_mgmt *mgmt, size_t len,
1325 int rssi)
1326{
1327 struct hostapd_sae_commit_queue *q, *q2;
1328 unsigned int queue_len;
1329 const struct ieee80211_mgmt *mgmt2;
1330
1331 queue_len = dl_list_len(&hapd->sae_commit_queue);
1332 if (queue_len >= 15) {
1333 wpa_printf(MSG_DEBUG,
1334 "SAE: No more room in message queue - drop the new frame from "
1335 MACSTR, MAC2STR(mgmt->sa));
1336 return;
1337 }
1338
1339 wpa_printf(MSG_DEBUG, "SAE: Queue Authentication message from "
1340 MACSTR " for processing (queue_len %u)", MAC2STR(mgmt->sa),
1341 queue_len);
1342 q = os_zalloc(sizeof(*q) + len);
1343 if (!q)
1344 return;
1345 q->rssi = rssi;
1346 q->len = len;
1347 os_memcpy(q->msg, mgmt, len);
1348
1349 /* Check whether there is already a queued Authentication frame from the
1350 * same station with the same transaction number and if so, replace that
1351 * queue entry with the new one. This avoids issues with a peer that
1352 * sends multiple times (e.g., due to frequent SAE retries). There is no
1353 * point in us trying to process the old attempts after a new one has
1354 * obsoleted them. */
1355 dl_list_for_each(q2, &hapd->sae_commit_queue,
1356 struct hostapd_sae_commit_queue, list) {
1357 mgmt2 = (const struct ieee80211_mgmt *) q2->msg;
1358 if (os_memcmp(mgmt->sa, mgmt2->sa, ETH_ALEN) == 0 &&
1359 mgmt->u.auth.auth_transaction ==
1360 mgmt2->u.auth.auth_transaction) {
1361 wpa_printf(MSG_DEBUG,
1362 "SAE: Replace queued message from same STA with same transaction number");
1363 dl_list_add(&q2->list, &q->list);
1364 dl_list_del(&q2->list);
1365 os_free(q2);
1366 goto queued;
1367 }
1368 }
1369
1370 /* No pending identical entry, so add to the end of the queue */
1371 dl_list_add_tail(&hapd->sae_commit_queue, &q->list);
1372
1373queued:
1374 if (eloop_is_timeout_registered(auth_sae_process_commit, hapd, NULL))
1375 return;
1376 eloop_register_timeout(0, queue_len * 10000, auth_sae_process_commit,
1377 hapd, NULL);
1378}
1379
1380
1381static int auth_sae_queued_addr(struct hostapd_data *hapd, const u8 *addr)
1382{
1383 struct hostapd_sae_commit_queue *q;
1384 const struct ieee80211_mgmt *mgmt;
1385
1386 dl_list_for_each(q, &hapd->sae_commit_queue,
1387 struct hostapd_sae_commit_queue, list) {
1388 mgmt = (const struct ieee80211_mgmt *) q->msg;
1389 if (os_memcmp(addr, mgmt->sa, ETH_ALEN) == 0)
1390 return 1;
1391 }
1392
1393 return 0;
1394}
1395
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08001396#endif /* CONFIG_SAE */
1397
1398
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001399static u16 wpa_res_to_status_code(int res)
1400{
1401 if (res == WPA_INVALID_GROUP)
1402 return WLAN_STATUS_GROUP_CIPHER_NOT_VALID;
1403 if (res == WPA_INVALID_PAIRWISE)
1404 return WLAN_STATUS_PAIRWISE_CIPHER_NOT_VALID;
1405 if (res == WPA_INVALID_AKMP)
1406 return WLAN_STATUS_AKMP_NOT_VALID;
1407 if (res == WPA_ALLOC_FAIL)
1408 return WLAN_STATUS_UNSPECIFIED_FAILURE;
1409#ifdef CONFIG_IEEE80211W
1410 if (res == WPA_MGMT_FRAME_PROTECTION_VIOLATION)
1411 return WLAN_STATUS_ROBUST_MGMT_FRAME_POLICY_VIOLATION;
1412 if (res == WPA_INVALID_MGMT_GROUP_CIPHER)
Roshan Pius3a1667e2018-07-03 15:17:14 -07001413 return WLAN_STATUS_CIPHER_REJECTED_PER_POLICY;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001414#endif /* CONFIG_IEEE80211W */
1415 if (res == WPA_INVALID_MDIE)
1416 return WLAN_STATUS_INVALID_MDIE;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001417 if (res == WPA_INVALID_PMKID)
1418 return WLAN_STATUS_INVALID_PMKID;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001419 if (res != WPA_IE_OK)
1420 return WLAN_STATUS_INVALID_IE;
1421 return WLAN_STATUS_SUCCESS;
1422}
1423
1424
1425#ifdef CONFIG_FILS
1426
1427static void handle_auth_fils_finish(struct hostapd_data *hapd,
1428 struct sta_info *sta, u16 resp,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001429 struct wpabuf *data, int pub);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001430
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001431void handle_auth_fils(struct hostapd_data *hapd, struct sta_info *sta,
1432 const u8 *pos, size_t len, u16 auth_alg,
1433 u16 auth_transaction, u16 status_code,
1434 void (*cb)(struct hostapd_data *hapd,
1435 struct sta_info *sta, u16 resp,
1436 struct wpabuf *data, int pub))
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001437{
1438 u16 resp = WLAN_STATUS_SUCCESS;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001439 const u8 *end;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001440 struct ieee802_11_elems elems;
1441 int res;
1442 struct wpa_ie_data rsn;
1443 struct rsn_pmksa_cache_entry *pmksa = NULL;
1444
1445 if (auth_transaction != 1 || status_code != WLAN_STATUS_SUCCESS)
1446 return;
1447
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001448 end = pos + len;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001449
1450 wpa_hexdump(MSG_DEBUG, "FILS: Authentication frame fields",
1451 pos, end - pos);
1452
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001453 /* TODO: FILS PK */
1454#ifdef CONFIG_FILS_SK_PFS
1455 if (auth_alg == WLAN_AUTH_FILS_SK_PFS) {
1456 u16 group;
1457 struct wpabuf *pub;
1458 size_t elem_len;
1459
1460 /* Using FILS PFS */
1461
1462 /* Finite Cyclic Group */
1463 if (end - pos < 2) {
1464 wpa_printf(MSG_DEBUG,
1465 "FILS: No room for Finite Cyclic Group");
1466 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1467 goto fail;
1468 }
1469 group = WPA_GET_LE16(pos);
1470 pos += 2;
1471 if (group != hapd->conf->fils_dh_group) {
1472 wpa_printf(MSG_DEBUG,
1473 "FILS: Unsupported Finite Cyclic Group: %u (expected %u)",
1474 group, hapd->conf->fils_dh_group);
1475 resp = WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED;
1476 goto fail;
1477 }
1478
1479 crypto_ecdh_deinit(sta->fils_ecdh);
1480 sta->fils_ecdh = crypto_ecdh_init(group);
1481 if (!sta->fils_ecdh) {
1482 wpa_printf(MSG_INFO,
1483 "FILS: Could not initialize ECDH with group %d",
1484 group);
1485 resp = WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED;
1486 goto fail;
1487 }
1488
1489 pub = crypto_ecdh_get_pubkey(sta->fils_ecdh, 1);
1490 if (!pub) {
1491 wpa_printf(MSG_DEBUG,
1492 "FILS: Failed to derive ECDH public key");
1493 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1494 goto fail;
1495 }
1496 elem_len = wpabuf_len(pub);
1497 wpabuf_free(pub);
1498
1499 /* Element */
1500 if ((size_t) (end - pos) < elem_len) {
1501 wpa_printf(MSG_DEBUG, "FILS: No room for Element");
1502 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1503 goto fail;
1504 }
1505
1506 wpabuf_free(sta->fils_g_sta);
1507 sta->fils_g_sta = wpabuf_alloc_copy(pos, elem_len);
1508 wpabuf_clear_free(sta->fils_dh_ss);
1509 sta->fils_dh_ss = crypto_ecdh_set_peerkey(sta->fils_ecdh, 1,
1510 pos, elem_len);
1511 if (!sta->fils_dh_ss) {
1512 wpa_printf(MSG_DEBUG, "FILS: ECDH operation failed");
1513 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1514 goto fail;
1515 }
1516 wpa_hexdump_buf_key(MSG_DEBUG, "FILS: DH_SS", sta->fils_dh_ss);
1517 pos += elem_len;
1518 } else {
1519 crypto_ecdh_deinit(sta->fils_ecdh);
1520 sta->fils_ecdh = NULL;
1521 wpabuf_clear_free(sta->fils_dh_ss);
1522 sta->fils_dh_ss = NULL;
1523 }
1524#endif /* CONFIG_FILS_SK_PFS */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001525
1526 wpa_hexdump(MSG_DEBUG, "FILS: Remaining IEs", pos, end - pos);
1527 if (ieee802_11_parse_elems(pos, end - pos, &elems, 1) == ParseFailed) {
1528 wpa_printf(MSG_DEBUG, "FILS: Could not parse elements");
1529 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1530 goto fail;
1531 }
1532
1533 /* RSNE */
1534 wpa_hexdump(MSG_DEBUG, "FILS: RSN element",
1535 elems.rsn_ie, elems.rsn_ie_len);
1536 if (!elems.rsn_ie ||
1537 wpa_parse_wpa_ie_rsn(elems.rsn_ie - 2, elems.rsn_ie_len + 2,
1538 &rsn) < 0) {
1539 wpa_printf(MSG_DEBUG, "FILS: No valid RSN element");
1540 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1541 goto fail;
1542 }
1543
1544 if (!sta->wpa_sm)
1545 sta->wpa_sm = wpa_auth_sta_init(hapd->wpa_auth, sta->addr,
1546 NULL);
1547 if (!sta->wpa_sm) {
1548 wpa_printf(MSG_DEBUG,
1549 "FILS: Failed to initialize RSN state machine");
1550 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1551 goto fail;
1552 }
1553
1554 res = wpa_validate_wpa_ie(hapd->wpa_auth, sta->wpa_sm,
Hai Shalom021b0b52019-04-10 11:17:58 -07001555 hapd->iface->freq,
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001556 elems.rsn_ie - 2, elems.rsn_ie_len + 2,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001557 elems.mdie, elems.mdie_len, NULL, 0);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001558 resp = wpa_res_to_status_code(res);
1559 if (resp != WLAN_STATUS_SUCCESS)
1560 goto fail;
1561
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001562 if (!elems.fils_nonce) {
1563 wpa_printf(MSG_DEBUG, "FILS: No FILS Nonce field");
1564 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1565 goto fail;
1566 }
1567 wpa_hexdump(MSG_DEBUG, "FILS: SNonce", elems.fils_nonce,
1568 FILS_NONCE_LEN);
1569 os_memcpy(sta->fils_snonce, elems.fils_nonce, FILS_NONCE_LEN);
1570
1571 /* PMKID List */
1572 if (rsn.pmkid && rsn.num_pmkid > 0) {
1573 u8 num;
1574 const u8 *pmkid;
1575
1576 wpa_hexdump(MSG_DEBUG, "FILS: PMKID List",
1577 rsn.pmkid, rsn.num_pmkid * PMKID_LEN);
1578
1579 pmkid = rsn.pmkid;
1580 num = rsn.num_pmkid;
1581 while (num) {
1582 wpa_hexdump(MSG_DEBUG, "FILS: PMKID", pmkid, PMKID_LEN);
1583 pmksa = wpa_auth_pmksa_get(hapd->wpa_auth, sta->addr,
1584 pmkid);
1585 if (pmksa)
1586 break;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001587 pmksa = wpa_auth_pmksa_get_fils_cache_id(hapd->wpa_auth,
1588 sta->addr,
1589 pmkid);
1590 if (pmksa)
1591 break;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001592 pmkid += PMKID_LEN;
1593 num--;
1594 }
1595 }
1596 if (pmksa && wpa_auth_sta_key_mgmt(sta->wpa_sm) != pmksa->akmp) {
1597 wpa_printf(MSG_DEBUG,
1598 "FILS: Matching PMKSA cache entry has different AKMP (0x%x != 0x%x) - ignore",
1599 wpa_auth_sta_key_mgmt(sta->wpa_sm), pmksa->akmp);
1600 pmksa = NULL;
1601 }
1602 if (pmksa)
1603 wpa_printf(MSG_DEBUG, "FILS: Found matching PMKSA cache entry");
1604
1605 /* FILS Session */
1606 if (!elems.fils_session) {
1607 wpa_printf(MSG_DEBUG, "FILS: No FILS Session element");
1608 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1609 goto fail;
1610 }
1611 wpa_hexdump(MSG_DEBUG, "FILS: FILS Session", elems.fils_session,
1612 FILS_SESSION_LEN);
1613 os_memcpy(sta->fils_session, elems.fils_session, FILS_SESSION_LEN);
1614
1615 /* FILS Wrapped Data */
1616 if (elems.fils_wrapped_data) {
1617 wpa_hexdump(MSG_DEBUG, "FILS: Wrapped Data",
1618 elems.fils_wrapped_data,
1619 elems.fils_wrapped_data_len);
1620 if (!pmksa) {
1621#ifndef CONFIG_NO_RADIUS
1622 if (!sta->eapol_sm) {
1623 sta->eapol_sm =
1624 ieee802_1x_alloc_eapol_sm(hapd, sta);
1625 }
1626 wpa_printf(MSG_DEBUG,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001627 "FILS: Forward EAP-Initiate/Re-auth to authentication server");
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001628 ieee802_1x_encapsulate_radius(
1629 hapd, sta, elems.fils_wrapped_data,
1630 elems.fils_wrapped_data_len);
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001631 sta->fils_pending_cb = cb;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001632 wpa_printf(MSG_DEBUG,
1633 "FILS: Will send Authentication frame once the response from authentication server is available");
1634 sta->flags |= WLAN_STA_PENDING_FILS_ERP;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001635 /* Calculate pending PMKID here so that we do not need
1636 * to maintain a copy of the EAP-Initiate/Reauth
1637 * message. */
1638 if (fils_pmkid_erp(wpa_auth_sta_key_mgmt(sta->wpa_sm),
1639 elems.fils_wrapped_data,
1640 elems.fils_wrapped_data_len,
1641 sta->fils_erp_pmkid) == 0)
1642 sta->fils_erp_pmkid_set = 1;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001643 return;
1644#else /* CONFIG_NO_RADIUS */
1645 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1646 goto fail;
1647#endif /* CONFIG_NO_RADIUS */
1648 }
1649 }
1650
1651fail:
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001652 if (cb) {
1653 struct wpabuf *data;
1654 int pub = 0;
1655
1656 data = prepare_auth_resp_fils(hapd, sta, &resp, pmksa, NULL,
1657 NULL, 0, &pub);
1658 if (!data) {
1659 wpa_printf(MSG_DEBUG,
1660 "%s: prepare_auth_resp_fils() returned failure",
1661 __func__);
1662 }
1663
1664 cb(hapd, sta, resp, data, pub);
1665 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001666}
1667
1668
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001669static struct wpabuf *
1670prepare_auth_resp_fils(struct hostapd_data *hapd,
1671 struct sta_info *sta, u16 *resp,
1672 struct rsn_pmksa_cache_entry *pmksa,
1673 struct wpabuf *erp_resp,
1674 const u8 *msk, size_t msk_len,
1675 int *is_pub)
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001676{
1677 u8 fils_nonce[FILS_NONCE_LEN];
1678 size_t ielen;
1679 struct wpabuf *data = NULL;
1680 const u8 *ie;
1681 u8 *ie_buf = NULL;
1682 const u8 *pmk = NULL;
1683 size_t pmk_len = 0;
Paul Stewart092955c2017-02-06 09:13:09 -08001684 u8 pmk_buf[PMK_LEN_MAX];
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001685 struct wpabuf *pub = NULL;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001686
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001687 if (*resp != WLAN_STATUS_SUCCESS)
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001688 goto fail;
1689
1690 ie = wpa_auth_get_wpa_ie(hapd->wpa_auth, &ielen);
1691 if (!ie) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001692 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001693 goto fail;
1694 }
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001695
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001696 if (pmksa) {
1697 /* Add PMKID of the selected PMKSA into RSNE */
1698 ie_buf = os_malloc(ielen + 2 + 2 + PMKID_LEN);
1699 if (!ie_buf) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001700 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001701 goto fail;
1702 }
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001703
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001704 os_memcpy(ie_buf, ie, ielen);
1705 if (wpa_insert_pmkid(ie_buf, &ielen, pmksa->pmkid) < 0) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001706 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001707 goto fail;
1708 }
1709 ie = ie_buf;
1710 }
1711
1712 if (random_get_bytes(fils_nonce, FILS_NONCE_LEN) < 0) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001713 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001714 goto fail;
1715 }
1716 wpa_hexdump(MSG_DEBUG, "RSN: Generated FILS Nonce",
1717 fils_nonce, FILS_NONCE_LEN);
1718
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001719#ifdef CONFIG_FILS_SK_PFS
1720 if (sta->fils_dh_ss && sta->fils_ecdh) {
1721 pub = crypto_ecdh_get_pubkey(sta->fils_ecdh, 1);
1722 if (!pub) {
1723 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1724 goto fail;
1725 }
1726 }
1727#endif /* CONFIG_FILS_SK_PFS */
1728
1729 data = wpabuf_alloc(1000 + ielen + (pub ? wpabuf_len(pub) : 0));
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001730 if (!data) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001731 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001732 goto fail;
1733 }
1734
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001735 /* TODO: FILS PK */
1736#ifdef CONFIG_FILS_SK_PFS
1737 if (pub) {
1738 /* Finite Cyclic Group */
1739 wpabuf_put_le16(data, hapd->conf->fils_dh_group);
1740
1741 /* Element */
1742 wpabuf_put_buf(data, pub);
1743 }
1744#endif /* CONFIG_FILS_SK_PFS */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001745
1746 /* RSNE */
1747 wpabuf_put_data(data, ie, ielen);
1748
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001749 /* MDE when using FILS+FT (already included in ie,ielen with RSNE) */
1750
1751#ifdef CONFIG_IEEE80211R_AP
1752 if (wpa_key_mgmt_ft(wpa_auth_sta_key_mgmt(sta->wpa_sm))) {
1753 /* FTE[R1KH-ID,R0KH-ID] when using FILS+FT */
1754 int res;
Roshan Pius3a1667e2018-07-03 15:17:14 -07001755 int use_sha384 = wpa_key_mgmt_sha384(
1756 wpa_auth_sta_key_mgmt(sta->wpa_sm));
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001757
Roshan Pius3a1667e2018-07-03 15:17:14 -07001758 res = wpa_auth_write_fte(hapd->wpa_auth, use_sha384,
1759 wpabuf_put(data, 0),
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001760 wpabuf_tailroom(data));
1761 if (res < 0) {
1762 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
1763 goto fail;
1764 }
1765 wpabuf_put(data, res);
1766 }
1767#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001768
1769 /* FILS Nonce */
1770 wpabuf_put_u8(data, WLAN_EID_EXTENSION); /* Element ID */
1771 wpabuf_put_u8(data, 1 + FILS_NONCE_LEN); /* Length */
1772 /* Element ID Extension */
1773 wpabuf_put_u8(data, WLAN_EID_EXT_FILS_NONCE);
1774 wpabuf_put_data(data, fils_nonce, FILS_NONCE_LEN);
1775
1776 /* FILS Session */
1777 wpabuf_put_u8(data, WLAN_EID_EXTENSION); /* Element ID */
1778 wpabuf_put_u8(data, 1 + FILS_SESSION_LEN); /* Length */
1779 /* Element ID Extension */
1780 wpabuf_put_u8(data, WLAN_EID_EXT_FILS_SESSION);
1781 wpabuf_put_data(data, sta->fils_session, FILS_SESSION_LEN);
1782
1783 /* FILS Wrapped Data */
1784 if (!pmksa && erp_resp) {
1785 wpabuf_put_u8(data, WLAN_EID_EXTENSION); /* Element ID */
1786 wpabuf_put_u8(data, 1 + wpabuf_len(erp_resp)); /* Length */
1787 /* Element ID Extension */
1788 wpabuf_put_u8(data, WLAN_EID_EXT_FILS_WRAPPED_DATA);
1789 wpabuf_put_buf(data, erp_resp);
1790
Paul Stewart092955c2017-02-06 09:13:09 -08001791 if (fils_rmsk_to_pmk(wpa_auth_sta_key_mgmt(sta->wpa_sm),
1792 msk, msk_len, sta->fils_snonce, fils_nonce,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001793 sta->fils_dh_ss ?
1794 wpabuf_head(sta->fils_dh_ss) : NULL,
1795 sta->fils_dh_ss ?
1796 wpabuf_len(sta->fils_dh_ss) : 0,
1797 pmk_buf, &pmk_len)) {
Paul Stewart092955c2017-02-06 09:13:09 -08001798 wpa_printf(MSG_DEBUG, "FILS: Failed to derive PMK");
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001799 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Paul Stewart092955c2017-02-06 09:13:09 -08001800 wpabuf_free(data);
1801 data = NULL;
1802 goto fail;
1803 }
1804 pmk = pmk_buf;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001805
1806 /* Don't use DHss in PTK derivation if PMKSA caching is not
1807 * used. */
1808 wpabuf_clear_free(sta->fils_dh_ss);
1809 sta->fils_dh_ss = NULL;
1810
1811 if (sta->fils_erp_pmkid_set) {
1812 /* TODO: get PMKLifetime from WPA parameters */
1813 unsigned int dot11RSNAConfigPMKLifetime = 43200;
Roshan Pius3a1667e2018-07-03 15:17:14 -07001814 int session_timeout;
1815
1816 session_timeout = dot11RSNAConfigPMKLifetime;
1817 if (sta->session_timeout_set) {
1818 struct os_reltime now, diff;
1819
1820 os_get_reltime(&now);
1821 os_reltime_sub(&sta->session_timeout, &now,
1822 &diff);
1823 session_timeout = diff.sec;
1824 }
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001825
1826 sta->fils_erp_pmkid_set = 0;
Hai Shalom81f62d82019-07-22 12:10:00 -07001827 wpa_auth_add_fils_pmk_pmkid(sta->wpa_sm, pmk, pmk_len,
1828 sta->fils_erp_pmkid);
Hai Shalom021b0b52019-04-10 11:17:58 -07001829 if (!hapd->conf->disable_pmksa_caching &&
1830 wpa_auth_pmksa_add2(
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001831 hapd->wpa_auth, sta->addr,
1832 pmk, pmk_len,
1833 sta->fils_erp_pmkid,
Roshan Pius3a1667e2018-07-03 15:17:14 -07001834 session_timeout,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001835 wpa_auth_sta_key_mgmt(sta->wpa_sm)) < 0) {
1836 wpa_printf(MSG_ERROR,
1837 "FILS: Failed to add PMKSA cache entry based on ERP");
1838 }
1839 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001840 } else if (pmksa) {
1841 pmk = pmksa->pmk;
1842 pmk_len = pmksa->pmk_len;
1843 }
1844
1845 if (!pmk) {
1846 wpa_printf(MSG_DEBUG, "FILS: No PMK available");
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001847 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001848 wpabuf_free(data);
1849 data = NULL;
1850 goto fail;
1851 }
1852
1853 if (fils_auth_pmk_to_ptk(sta->wpa_sm, pmk, pmk_len,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001854 sta->fils_snonce, fils_nonce,
1855 sta->fils_dh_ss ?
1856 wpabuf_head(sta->fils_dh_ss) : NULL,
1857 sta->fils_dh_ss ?
1858 wpabuf_len(sta->fils_dh_ss) : 0,
1859 sta->fils_g_sta, pub) < 0) {
1860 *resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001861 wpabuf_free(data);
1862 data = NULL;
1863 goto fail;
1864 }
1865
1866fail:
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001867 if (is_pub)
1868 *is_pub = pub != NULL;
1869 os_free(ie_buf);
1870 wpabuf_free(pub);
1871 wpabuf_clear_free(sta->fils_dh_ss);
1872 sta->fils_dh_ss = NULL;
1873#ifdef CONFIG_FILS_SK_PFS
1874 crypto_ecdh_deinit(sta->fils_ecdh);
1875 sta->fils_ecdh = NULL;
1876#endif /* CONFIG_FILS_SK_PFS */
1877 return data;
1878}
1879
1880
1881static void handle_auth_fils_finish(struct hostapd_data *hapd,
1882 struct sta_info *sta, u16 resp,
1883 struct wpabuf *data, int pub)
1884{
1885 u16 auth_alg;
1886
1887 auth_alg = (pub ||
1888 resp == WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED) ?
1889 WLAN_AUTH_FILS_SK_PFS : WLAN_AUTH_FILS_SK;
1890 send_auth_reply(hapd, sta->addr, hapd->own_addr, auth_alg, 2, resp,
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001891 data ? wpabuf_head(data) : (u8 *) "",
Roshan Pius3a1667e2018-07-03 15:17:14 -07001892 data ? wpabuf_len(data) : 0, "auth-fils-finish");
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001893 wpabuf_free(data);
1894
1895 if (resp == WLAN_STATUS_SUCCESS) {
1896 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
1897 HOSTAPD_LEVEL_DEBUG,
1898 "authentication OK (FILS)");
1899 sta->flags |= WLAN_STA_AUTH;
1900 wpa_auth_sm_event(sta->wpa_sm, WPA_AUTH);
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001901 sta->auth_alg = pub ? WLAN_AUTH_FILS_SK_PFS : WLAN_AUTH_FILS_SK;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001902 mlme_authenticate_indication(hapd, sta);
1903 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001904}
1905
1906
1907void ieee802_11_finish_fils_auth(struct hostapd_data *hapd,
1908 struct sta_info *sta, int success,
1909 struct wpabuf *erp_resp,
1910 const u8 *msk, size_t msk_len)
1911{
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001912 struct wpabuf *data;
1913 int pub = 0;
1914 u16 resp;
1915
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001916 sta->flags &= ~WLAN_STA_PENDING_FILS_ERP;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07001917
1918 if (!sta->fils_pending_cb)
1919 return;
1920 resp = success ? WLAN_STATUS_SUCCESS : WLAN_STATUS_UNSPECIFIED_FAILURE;
1921 data = prepare_auth_resp_fils(hapd, sta, &resp, NULL, erp_resp,
1922 msk, msk_len, &pub);
1923 if (!data) {
1924 wpa_printf(MSG_DEBUG,
1925 "%s: prepare_auth_resp_fils() returned failure",
1926 __func__);
1927 }
1928 sta->fils_pending_cb(hapd, sta, resp, data, pub);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08001929}
1930
1931#endif /* CONFIG_FILS */
1932
1933
Roshan Pius3a1667e2018-07-03 15:17:14 -07001934int
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08001935ieee802_11_allowed_address(struct hostapd_data *hapd, const u8 *addr,
1936 const u8 *msg, size_t len, u32 *session_timeout,
1937 u32 *acct_interim_interval,
1938 struct vlan_description *vlan_id,
1939 struct hostapd_sta_wpa_psk_short **psk,
Roshan Pius3a1667e2018-07-03 15:17:14 -07001940 char **identity, char **radius_cui, int is_probe_req)
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08001941{
1942 int res;
1943
1944 os_memset(vlan_id, 0, sizeof(*vlan_id));
1945 res = hostapd_allowed_address(hapd, addr, msg, len,
1946 session_timeout, acct_interim_interval,
Roshan Pius3a1667e2018-07-03 15:17:14 -07001947 vlan_id, psk, identity, radius_cui,
1948 is_probe_req);
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08001949
1950 if (res == HOSTAPD_ACL_REJECT) {
Hai Shalom39ba6fc2019-01-22 12:40:38 -08001951 if (!is_probe_req)
1952 wpa_printf(MSG_DEBUG,
1953 "Station " MACSTR
1954 " not allowed to authenticate",
1955 MAC2STR(addr));
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08001956 return HOSTAPD_ACL_REJECT;
1957 }
1958
1959 if (res == HOSTAPD_ACL_PENDING) {
1960 wpa_printf(MSG_DEBUG, "Authentication frame from " MACSTR
1961 " waiting for an external authentication",
1962 MAC2STR(addr));
1963 /* Authentication code will re-send the authentication frame
1964 * after it has received (and cached) information from the
1965 * external source. */
1966 return HOSTAPD_ACL_PENDING;
1967 }
1968
1969 return res;
1970}
1971
1972
1973static int
1974ieee802_11_set_radius_info(struct hostapd_data *hapd, struct sta_info *sta,
1975 int res, u32 session_timeout,
1976 u32 acct_interim_interval,
1977 struct vlan_description *vlan_id,
1978 struct hostapd_sta_wpa_psk_short **psk,
1979 char **identity, char **radius_cui)
1980{
1981 if (vlan_id->notempty &&
1982 !hostapd_vlan_valid(hapd->conf->vlan, vlan_id)) {
1983 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_RADIUS,
1984 HOSTAPD_LEVEL_INFO,
1985 "Invalid VLAN %d%s received from RADIUS server",
1986 vlan_id->untagged,
1987 vlan_id->tagged[0] ? "+" : "");
1988 return -1;
1989 }
1990 if (ap_sta_set_vlan(hapd, sta, vlan_id) < 0)
1991 return -1;
1992 if (sta->vlan_id)
1993 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_RADIUS,
1994 HOSTAPD_LEVEL_INFO, "VLAN ID %d", sta->vlan_id);
1995
1996 hostapd_free_psk_list(sta->psk);
1997 if (hapd->conf->wpa_psk_radius != PSK_RADIUS_IGNORED) {
1998 sta->psk = *psk;
1999 *psk = NULL;
2000 } else {
2001 sta->psk = NULL;
2002 }
2003
Roshan Pius3a1667e2018-07-03 15:17:14 -07002004 os_free(sta->identity);
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002005 sta->identity = *identity;
2006 *identity = NULL;
Roshan Pius3a1667e2018-07-03 15:17:14 -07002007
2008 os_free(sta->radius_cui);
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002009 sta->radius_cui = *radius_cui;
2010 *radius_cui = NULL;
2011
2012 if (hapd->conf->acct_interim_interval == 0 && acct_interim_interval)
2013 sta->acct_interim_interval = acct_interim_interval;
Roshan Pius3a1667e2018-07-03 15:17:14 -07002014 if (res == HOSTAPD_ACL_ACCEPT_TIMEOUT) {
2015 sta->session_timeout_set = 1;
2016 os_get_reltime(&sta->session_timeout);
2017 sta->session_timeout.sec += session_timeout;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002018 ap_sta_session_timeout(hapd, sta, session_timeout);
Roshan Pius3a1667e2018-07-03 15:17:14 -07002019 } else {
2020 sta->session_timeout_set = 0;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002021 ap_sta_no_session_timeout(hapd, sta);
Roshan Pius3a1667e2018-07-03 15:17:14 -07002022 }
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002023
2024 return 0;
2025}
2026
2027
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002028static void handle_auth(struct hostapd_data *hapd,
Hai Shalom74f70d42019-02-11 14:42:39 -08002029 const struct ieee80211_mgmt *mgmt, size_t len,
Hai Shalom021b0b52019-04-10 11:17:58 -07002030 int rssi, int from_queue)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002031{
2032 u16 auth_alg, auth_transaction, status_code;
2033 u16 resp = WLAN_STATUS_SUCCESS;
2034 struct sta_info *sta = NULL;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002035 int res, reply_res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002036 u16 fc;
2037 const u8 *challenge = NULL;
2038 u32 session_timeout, acct_interim_interval;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002039 struct vlan_description vlan_id;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002040 struct hostapd_sta_wpa_psk_short *psk = NULL;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002041 u8 resp_ies[2 + WLAN_AUTH_CHALLENGE_LEN];
2042 size_t resp_ies_len = 0;
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002043 char *identity = NULL;
2044 char *radius_cui = NULL;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002045 u16 seq_ctrl;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002046
2047 if (len < IEEE80211_HDRLEN + sizeof(mgmt->u.auth)) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08002048 wpa_printf(MSG_INFO, "handle_auth - too short payload (len=%lu)",
2049 (unsigned long) len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002050 return;
2051 }
2052
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07002053#ifdef CONFIG_TESTING_OPTIONS
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07002054 if (hapd->iconf->ignore_auth_probability > 0.0 &&
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07002055 drand48() < hapd->iconf->ignore_auth_probability) {
2056 wpa_printf(MSG_INFO,
2057 "TESTING: ignoring auth frame from " MACSTR,
2058 MAC2STR(mgmt->sa));
2059 return;
2060 }
2061#endif /* CONFIG_TESTING_OPTIONS */
2062
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002063 auth_alg = le_to_host16(mgmt->u.auth.auth_alg);
2064 auth_transaction = le_to_host16(mgmt->u.auth.auth_transaction);
2065 status_code = le_to_host16(mgmt->u.auth.status_code);
2066 fc = le_to_host16(mgmt->frame_control);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002067 seq_ctrl = le_to_host16(mgmt->seq_ctrl);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002068
2069 if (len >= IEEE80211_HDRLEN + sizeof(mgmt->u.auth) +
2070 2 + WLAN_AUTH_CHALLENGE_LEN &&
2071 mgmt->u.auth.variable[0] == WLAN_EID_CHALLENGE &&
2072 mgmt->u.auth.variable[1] == WLAN_AUTH_CHALLENGE_LEN)
2073 challenge = &mgmt->u.auth.variable[2];
2074
2075 wpa_printf(MSG_DEBUG, "authentication: STA=" MACSTR " auth_alg=%d "
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002076 "auth_transaction=%d status_code=%d wep=%d%s "
Hai Shalom021b0b52019-04-10 11:17:58 -07002077 "seq_ctrl=0x%x%s%s",
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002078 MAC2STR(mgmt->sa), auth_alg, auth_transaction,
2079 status_code, !!(fc & WLAN_FC_ISWEP),
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002080 challenge ? " challenge" : "",
Hai Shalom021b0b52019-04-10 11:17:58 -07002081 seq_ctrl, (fc & WLAN_FC_RETRY) ? " retry" : "",
2082 from_queue ? " (from queue)" : "");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002083
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002084#ifdef CONFIG_NO_RC4
2085 if (auth_alg == WLAN_AUTH_SHARED_KEY) {
2086 wpa_printf(MSG_INFO,
2087 "Unsupported authentication algorithm (%d)",
2088 auth_alg);
2089 resp = WLAN_STATUS_NOT_SUPPORTED_AUTH_ALG;
2090 goto fail;
2091 }
2092#endif /* CONFIG_NO_RC4 */
2093
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002094 if (hapd->tkip_countermeasures) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07002095 wpa_printf(MSG_DEBUG,
2096 "Ongoing TKIP countermeasures (Michael MIC failure) - reject authentication");
2097 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002098 goto fail;
2099 }
2100
2101 if (!(((hapd->conf->auth_algs & WPA_AUTH_ALG_OPEN) &&
2102 auth_alg == WLAN_AUTH_OPEN) ||
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002103#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002104 (hapd->conf->wpa && wpa_key_mgmt_ft(hapd->conf->wpa_key_mgmt) &&
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002105 auth_alg == WLAN_AUTH_FT) ||
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002106#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002107#ifdef CONFIG_SAE
2108 (hapd->conf->wpa && wpa_key_mgmt_sae(hapd->conf->wpa_key_mgmt) &&
2109 auth_alg == WLAN_AUTH_SAE) ||
2110#endif /* CONFIG_SAE */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002111#ifdef CONFIG_FILS
2112 (hapd->conf->wpa && wpa_key_mgmt_fils(hapd->conf->wpa_key_mgmt) &&
2113 auth_alg == WLAN_AUTH_FILS_SK) ||
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07002114 (hapd->conf->wpa && wpa_key_mgmt_fils(hapd->conf->wpa_key_mgmt) &&
2115 hapd->conf->fils_dh_group &&
2116 auth_alg == WLAN_AUTH_FILS_SK_PFS) ||
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002117#endif /* CONFIG_FILS */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002118 ((hapd->conf->auth_algs & WPA_AUTH_ALG_SHARED) &&
2119 auth_alg == WLAN_AUTH_SHARED_KEY))) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08002120 wpa_printf(MSG_INFO, "Unsupported authentication algorithm (%d)",
2121 auth_alg);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002122 resp = WLAN_STATUS_NOT_SUPPORTED_AUTH_ALG;
2123 goto fail;
2124 }
2125
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002126 if (!(auth_transaction == 1 || auth_alg == WLAN_AUTH_SAE ||
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002127 (auth_alg == WLAN_AUTH_SHARED_KEY && auth_transaction == 3))) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08002128 wpa_printf(MSG_INFO, "Unknown authentication transaction number (%d)",
2129 auth_transaction);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002130 resp = WLAN_STATUS_UNKNOWN_AUTH_TRANSACTION;
2131 goto fail;
2132 }
2133
2134 if (os_memcmp(mgmt->sa, hapd->own_addr, ETH_ALEN) == 0) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08002135 wpa_printf(MSG_INFO, "Station " MACSTR " not allowed to authenticate",
2136 MAC2STR(mgmt->sa));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002137 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
2138 goto fail;
2139 }
2140
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002141 if (hapd->conf->no_auth_if_seen_on) {
2142 struct hostapd_data *other;
2143
2144 other = sta_track_seen_on(hapd->iface, mgmt->sa,
2145 hapd->conf->no_auth_if_seen_on);
2146 if (other) {
2147 u8 *pos;
2148 u32 info;
2149 u8 op_class, channel, phytype;
2150
2151 wpa_printf(MSG_DEBUG, "%s: Reject authentication from "
2152 MACSTR " since STA has been seen on %s",
2153 hapd->conf->iface, MAC2STR(mgmt->sa),
2154 hapd->conf->no_auth_if_seen_on);
2155
2156 resp = WLAN_STATUS_REJECTED_WITH_SUGGESTED_BSS_TRANSITION;
2157 pos = &resp_ies[0];
2158 *pos++ = WLAN_EID_NEIGHBOR_REPORT;
2159 *pos++ = 13;
2160 os_memcpy(pos, other->own_addr, ETH_ALEN);
2161 pos += ETH_ALEN;
2162 info = 0; /* TODO: BSSID Information */
2163 WPA_PUT_LE32(pos, info);
2164 pos += 4;
2165 if (other->iconf->hw_mode == HOSTAPD_MODE_IEEE80211AD)
2166 phytype = 8; /* dmg */
2167 else if (other->iconf->ieee80211ac)
2168 phytype = 9; /* vht */
2169 else if (other->iconf->ieee80211n)
2170 phytype = 7; /* ht */
2171 else if (other->iconf->hw_mode ==
2172 HOSTAPD_MODE_IEEE80211A)
2173 phytype = 4; /* ofdm */
2174 else if (other->iconf->hw_mode ==
2175 HOSTAPD_MODE_IEEE80211G)
2176 phytype = 6; /* erp */
2177 else
2178 phytype = 5; /* hrdsss */
2179 if (ieee80211_freq_to_channel_ext(
2180 hostapd_hw_get_freq(other,
2181 other->iconf->channel),
2182 other->iconf->secondary_channel,
2183 other->iconf->ieee80211ac,
2184 &op_class, &channel) == NUM_HOSTAPD_MODES) {
2185 op_class = 0;
2186 channel = other->iconf->channel;
2187 }
2188 *pos++ = op_class;
2189 *pos++ = channel;
2190 *pos++ = phytype;
2191 resp_ies_len = pos - &resp_ies[0];
2192 goto fail;
2193 }
2194 }
2195
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002196 res = ieee802_11_allowed_address(
2197 hapd, mgmt->sa, (const u8 *) mgmt, len, &session_timeout,
Roshan Pius3a1667e2018-07-03 15:17:14 -07002198 &acct_interim_interval, &vlan_id, &psk, &identity, &radius_cui,
2199 0);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002200 if (res == HOSTAPD_ACL_REJECT) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07002201 wpa_msg(hapd->msg_ctx, MSG_DEBUG,
2202 "Ignore Authentication frame from " MACSTR
2203 " due to ACL reject", MAC2STR(mgmt->sa));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002204 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
2205 goto fail;
2206 }
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002207 if (res == HOSTAPD_ACL_PENDING)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002208 return;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002209
Hai Shalom021b0b52019-04-10 11:17:58 -07002210#ifdef CONFIG_SAE
2211 if (auth_alg == WLAN_AUTH_SAE && !from_queue &&
2212 (auth_transaction == 1 ||
2213 (auth_transaction == 2 && auth_sae_queued_addr(hapd, mgmt->sa)))) {
2214 /* Handle SAE Authentication commit message through a queue to
2215 * provide more control for postponing the needed heavy
2216 * processing under a possible DoS attack scenario. In addition,
2217 * queue SAE Authentication confirm message if there happens to
2218 * be a queued commit message from the same peer. This is needed
2219 * to avoid reordering Authentication frames within the same
2220 * SAE exchange. */
2221 auth_sae_queue(hapd, mgmt, len, rssi);
2222 return;
2223 }
2224#endif /* CONFIG_SAE */
2225
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002226 sta = ap_get_sta(hapd, mgmt->sa);
2227 if (sta) {
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002228 sta->flags &= ~WLAN_STA_PENDING_FILS_ERP;
Hai Shalom74f70d42019-02-11 14:42:39 -08002229 sta->ft_over_ds = 0;
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002230 if ((fc & WLAN_FC_RETRY) &&
2231 sta->last_seq_ctrl != WLAN_INVALID_MGMT_SEQ &&
2232 sta->last_seq_ctrl == seq_ctrl &&
2233 sta->last_subtype == WLAN_FC_STYPE_AUTH) {
2234 hostapd_logger(hapd, sta->addr,
2235 HOSTAPD_MODULE_IEEE80211,
2236 HOSTAPD_LEVEL_DEBUG,
2237 "Drop repeated authentication frame seq_ctrl=0x%x",
2238 seq_ctrl);
2239 return;
2240 }
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002241#ifdef CONFIG_MESH
2242 if ((hapd->conf->mesh & MESH_ENABLED) &&
2243 sta->plink_state == PLINK_BLOCKED) {
2244 wpa_printf(MSG_DEBUG, "Mesh peer " MACSTR
2245 " is blocked - drop Authentication frame",
2246 MAC2STR(mgmt->sa));
2247 return;
2248 }
2249#endif /* CONFIG_MESH */
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002250 } else {
2251#ifdef CONFIG_MESH
2252 if (hapd->conf->mesh & MESH_ENABLED) {
2253 /* if the mesh peer is not available, we don't do auth.
2254 */
2255 wpa_printf(MSG_DEBUG, "Mesh peer " MACSTR
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002256 " not yet known - drop Authentication frame",
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002257 MAC2STR(mgmt->sa));
2258 /*
2259 * Save a copy of the frame so that it can be processed
2260 * if a new peer entry is added shortly after this.
2261 */
2262 wpabuf_free(hapd->mesh_pending_auth);
2263 hapd->mesh_pending_auth = wpabuf_alloc_copy(mgmt, len);
2264 os_get_reltime(&hapd->mesh_pending_auth_time);
2265 return;
2266 }
2267#endif /* CONFIG_MESH */
2268
2269 sta = ap_sta_add(hapd, mgmt->sa);
2270 if (!sta) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07002271 wpa_printf(MSG_DEBUG, "ap_sta_add() failed");
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002272 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
2273 goto fail;
2274 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002275 }
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002276 sta->last_seq_ctrl = seq_ctrl;
2277 sta->last_subtype = WLAN_FC_STYPE_AUTH;
Hai Shalom74f70d42019-02-11 14:42:39 -08002278#ifdef CONFIG_MBO
2279 sta->auth_rssi = rssi;
2280#endif /* CONFIG_MBO */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002281
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08002282 res = ieee802_11_set_radius_info(
2283 hapd, sta, res, session_timeout, acct_interim_interval,
2284 &vlan_id, &psk, &identity, &radius_cui);
2285 if (res) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07002286 wpa_printf(MSG_DEBUG, "ieee802_11_set_radius_info() failed");
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002287 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
2288 goto fail;
2289 }
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002290
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002291 sta->flags &= ~WLAN_STA_PREAUTH;
2292 ieee802_1x_notify_pre_auth(sta->eapol_sm, 0);
2293
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002294 /*
2295 * If the driver supports full AP client state, add a station to the
2296 * driver before sending authentication reply to make sure the driver
2297 * has resources, and not to go through the entire authentication and
2298 * association handshake, and fail it at the end.
2299 *
2300 * If this is not the first transaction, in a multi-step authentication
2301 * algorithm, the station already exists in the driver
2302 * (sta->added_unassoc = 1) so skip it.
2303 *
2304 * In mesh mode, the station was already added to the driver when the
2305 * NEW_PEER_CANDIDATE event is received.
Dmitry Shmidtabb90a32016-12-05 15:34:39 -08002306 *
2307 * If PMF was negotiated for the existing association, skip this to
2308 * avoid dropping the STA entry and the associated keys. This is needed
2309 * to allow the original connection work until the attempt can complete
2310 * (re)association, so that unprotected Authentication frame cannot be
2311 * used to bypass PMF protection.
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002312 */
2313 if (FULL_AP_CLIENT_STATE_SUPP(hapd->iface->drv_flags) &&
Dmitry Shmidtabb90a32016-12-05 15:34:39 -08002314 (!(sta->flags & WLAN_STA_MFP) || !ap_sta_is_authorized(sta)) &&
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002315 !(hapd->conf->mesh & MESH_ENABLED) &&
2316 !(sta->added_unassoc)) {
2317 /*
2318 * If a station that is already associated to the AP, is trying
2319 * to authenticate again, remove the STA entry, in order to make
2320 * sure the STA PS state gets cleared and configuration gets
2321 * updated. To handle this, station's added_unassoc flag is
2322 * cleared once the station has completed association.
2323 */
Hai Shalomce48b4a2018-09-05 11:41:35 -07002324 ap_sta_set_authorized(hapd, sta, 0);
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002325 hostapd_drv_sta_remove(hapd, sta->addr);
2326 sta->flags &= ~(WLAN_STA_ASSOC | WLAN_STA_AUTH |
2327 WLAN_STA_AUTHORIZED);
2328
Hai Shalom81f62d82019-07-22 12:10:00 -07002329 if (hostapd_sta_add(hapd, sta->addr, 0, 0,
2330 sta->supported_rates,
2331 sta->supported_rates_len,
2332 0, NULL, NULL, NULL, 0,
2333 sta->flags, 0, 0, 0, 0)) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002334 hostapd_logger(hapd, sta->addr,
2335 HOSTAPD_MODULE_IEEE80211,
2336 HOSTAPD_LEVEL_NOTICE,
2337 "Could not add STA to kernel driver");
2338 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
2339 goto fail;
2340 }
2341
2342 sta->added_unassoc = 1;
2343 }
2344
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002345 switch (auth_alg) {
2346 case WLAN_AUTH_OPEN:
2347 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2348 HOSTAPD_LEVEL_DEBUG,
2349 "authentication OK (open system)");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002350 sta->flags |= WLAN_STA_AUTH;
2351 wpa_auth_sm_event(sta->wpa_sm, WPA_AUTH);
2352 sta->auth_alg = WLAN_AUTH_OPEN;
2353 mlme_authenticate_indication(hapd, sta);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002354 break;
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002355#ifndef CONFIG_NO_RC4
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002356 case WLAN_AUTH_SHARED_KEY:
2357 resp = auth_shared_key(hapd, sta, auth_transaction, challenge,
2358 fc & WLAN_FC_ISWEP);
Roshan Pius3a1667e2018-07-03 15:17:14 -07002359 if (resp != 0)
2360 wpa_printf(MSG_DEBUG,
2361 "auth_shared_key() failed: status=%d", resp);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002362 sta->auth_alg = WLAN_AUTH_SHARED_KEY;
2363 mlme_authenticate_indication(hapd, sta);
2364 if (sta->challenge && auth_transaction == 1) {
2365 resp_ies[0] = WLAN_EID_CHALLENGE;
2366 resp_ies[1] = WLAN_AUTH_CHALLENGE_LEN;
2367 os_memcpy(resp_ies + 2, sta->challenge,
2368 WLAN_AUTH_CHALLENGE_LEN);
2369 resp_ies_len = 2 + WLAN_AUTH_CHALLENGE_LEN;
2370 }
2371 break;
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002372#endif /* CONFIG_NO_RC4 */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002373#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002374 case WLAN_AUTH_FT:
2375 sta->auth_alg = WLAN_AUTH_FT;
2376 if (sta->wpa_sm == NULL)
2377 sta->wpa_sm = wpa_auth_sta_init(hapd->wpa_auth,
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002378 sta->addr, NULL);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002379 if (sta->wpa_sm == NULL) {
2380 wpa_printf(MSG_DEBUG, "FT: Failed to initialize WPA "
2381 "state machine");
2382 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
2383 goto fail;
2384 }
2385 wpa_ft_process_auth(sta->wpa_sm, mgmt->bssid,
2386 auth_transaction, mgmt->u.auth.variable,
2387 len - IEEE80211_HDRLEN -
2388 sizeof(mgmt->u.auth),
2389 handle_auth_ft_finish, hapd);
2390 /* handle_auth_ft_finish() callback will complete auth. */
2391 return;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002392#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002393#ifdef CONFIG_SAE
2394 case WLAN_AUTH_SAE:
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08002395#ifdef CONFIG_MESH
2396 if (status_code == WLAN_STATUS_SUCCESS &&
2397 hapd->conf->mesh & MESH_ENABLED) {
2398 if (sta->wpa_sm == NULL)
2399 sta->wpa_sm =
2400 wpa_auth_sta_init(hapd->wpa_auth,
2401 sta->addr, NULL);
2402 if (sta->wpa_sm == NULL) {
2403 wpa_printf(MSG_DEBUG,
2404 "SAE: Failed to initialize WPA state machine");
2405 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
2406 goto fail;
2407 }
2408 }
2409#endif /* CONFIG_MESH */
2410 handle_auth_sae(hapd, sta, mgmt, len, auth_transaction,
2411 status_code);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002412 return;
2413#endif /* CONFIG_SAE */
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002414#ifdef CONFIG_FILS
2415 case WLAN_AUTH_FILS_SK:
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07002416 case WLAN_AUTH_FILS_SK_PFS:
2417 handle_auth_fils(hapd, sta, mgmt->u.auth.variable,
2418 len - IEEE80211_HDRLEN - sizeof(mgmt->u.auth),
2419 auth_alg, auth_transaction, status_code,
2420 handle_auth_fils_finish);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08002421 return;
2422#endif /* CONFIG_FILS */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002423 }
2424
2425 fail:
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002426 os_free(identity);
2427 os_free(radius_cui);
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08002428 hostapd_free_psk_list(psk);
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002429
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002430 reply_res = send_auth_reply(hapd, mgmt->sa, mgmt->bssid, auth_alg,
2431 auth_transaction + 1, resp, resp_ies,
Roshan Pius3a1667e2018-07-03 15:17:14 -07002432 resp_ies_len, "handle-auth");
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08002433
2434 if (sta && sta->added_unassoc && (resp != WLAN_STATUS_SUCCESS ||
2435 reply_res != WLAN_STATUS_SUCCESS)) {
2436 hostapd_drv_sta_remove(hapd, sta->addr);
2437 sta->added_unassoc = 0;
2438 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002439}
2440
2441
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002442int hostapd_get_aid(struct hostapd_data *hapd, struct sta_info *sta)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002443{
2444 int i, j = 32, aid;
2445
2446 /* get a unique AID */
2447 if (sta->aid > 0) {
2448 wpa_printf(MSG_DEBUG, " old AID %d", sta->aid);
2449 return 0;
2450 }
2451
Dmitry Shmidt58d12ad2016-07-28 10:07:03 -07002452 if (TEST_FAIL())
2453 return -1;
2454
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002455 for (i = 0; i < AID_WORDS; i++) {
2456 if (hapd->sta_aid[i] == (u32) -1)
2457 continue;
2458 for (j = 0; j < 32; j++) {
2459 if (!(hapd->sta_aid[i] & BIT(j)))
2460 break;
2461 }
2462 if (j < 32)
2463 break;
2464 }
2465 if (j == 32)
2466 return -1;
2467 aid = i * 32 + j + 1;
2468 if (aid > 2007)
2469 return -1;
2470
2471 sta->aid = aid;
2472 hapd->sta_aid[i] |= BIT(j);
2473 wpa_printf(MSG_DEBUG, " new AID %d", sta->aid);
2474 return 0;
2475}
2476
2477
2478static u16 check_ssid(struct hostapd_data *hapd, struct sta_info *sta,
2479 const u8 *ssid_ie, size_t ssid_ie_len)
2480{
2481 if (ssid_ie == NULL)
2482 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2483
2484 if (ssid_ie_len != hapd->conf->ssid.ssid_len ||
2485 os_memcmp(ssid_ie, hapd->conf->ssid.ssid, ssid_ie_len) != 0) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002486 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2487 HOSTAPD_LEVEL_INFO,
2488 "Station tried to associate with unknown SSID "
Dmitry Shmidt3c479372014-02-04 10:50:36 -08002489 "'%s'", wpa_ssid_txt(ssid_ie, ssid_ie_len));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002490 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2491 }
2492
2493 return WLAN_STATUS_SUCCESS;
2494}
2495
2496
2497static u16 check_wmm(struct hostapd_data *hapd, struct sta_info *sta,
2498 const u8 *wmm_ie, size_t wmm_ie_len)
2499{
2500 sta->flags &= ~WLAN_STA_WMM;
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002501 sta->qosinfo = 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002502 if (wmm_ie && hapd->conf->wmm_enabled) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002503 struct wmm_information_element *wmm;
2504
2505 if (!hostapd_eid_wmm_valid(hapd, wmm_ie, wmm_ie_len)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002506 hostapd_logger(hapd, sta->addr,
2507 HOSTAPD_MODULE_WPA,
2508 HOSTAPD_LEVEL_DEBUG,
2509 "invalid WMM element in association "
2510 "request");
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08002511 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2512 }
2513
2514 sta->flags |= WLAN_STA_WMM;
2515 wmm = (struct wmm_information_element *) wmm_ie;
2516 sta->qosinfo = wmm->qos_info;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002517 }
2518 return WLAN_STATUS_SUCCESS;
2519}
2520
Hai Shalom74f70d42019-02-11 14:42:39 -08002521static u16 check_multi_ap(struct hostapd_data *hapd, struct sta_info *sta,
2522 const u8 *multi_ap_ie, size_t multi_ap_len)
2523{
2524 u8 multi_ap_value = 0;
2525
2526 sta->flags &= ~WLAN_STA_MULTI_AP;
2527
2528 if (!hapd->conf->multi_ap)
2529 return WLAN_STATUS_SUCCESS;
2530
2531 if (multi_ap_ie) {
2532 const u8 *multi_ap_subelem;
2533
2534 multi_ap_subelem = get_ie(multi_ap_ie + 4,
2535 multi_ap_len - 4,
2536 MULTI_AP_SUB_ELEM_TYPE);
2537 if (multi_ap_subelem && multi_ap_subelem[1] == 1) {
2538 multi_ap_value = multi_ap_subelem[2];
2539 } else {
2540 hostapd_logger(hapd, sta->addr,
2541 HOSTAPD_MODULE_IEEE80211,
2542 HOSTAPD_LEVEL_INFO,
2543 "Multi-AP IE has missing or invalid Multi-AP subelement");
2544 return WLAN_STATUS_INVALID_IE;
2545 }
2546 }
2547
Hai Shalom021b0b52019-04-10 11:17:58 -07002548 if (multi_ap_value && multi_ap_value != MULTI_AP_BACKHAUL_STA)
2549 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2550 HOSTAPD_LEVEL_INFO,
2551 "Multi-AP IE with unexpected value 0x%02x",
2552 multi_ap_value);
Hai Shalom74f70d42019-02-11 14:42:39 -08002553
Hai Shalom021b0b52019-04-10 11:17:58 -07002554 if (!(multi_ap_value & MULTI_AP_BACKHAUL_STA)) {
2555 if (hapd->conf->multi_ap & FRONTHAUL_BSS)
2556 return WLAN_STATUS_SUCCESS;
Hai Shalom74f70d42019-02-11 14:42:39 -08002557
Hai Shalom021b0b52019-04-10 11:17:58 -07002558 hostapd_logger(hapd, sta->addr,
2559 HOSTAPD_MODULE_IEEE80211,
2560 HOSTAPD_LEVEL_INFO,
2561 "Non-Multi-AP STA tries to associate with backhaul-only BSS");
2562 return WLAN_STATUS_ASSOC_DENIED_UNSPEC;
Hai Shalom74f70d42019-02-11 14:42:39 -08002563 }
2564
Hai Shalom021b0b52019-04-10 11:17:58 -07002565 if (!(hapd->conf->multi_ap & BACKHAUL_BSS))
2566 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2567 HOSTAPD_LEVEL_DEBUG,
2568 "Backhaul STA tries to associate with fronthaul-only BSS");
2569
2570 sta->flags |= WLAN_STA_MULTI_AP;
2571 return WLAN_STATUS_SUCCESS;
Hai Shalom74f70d42019-02-11 14:42:39 -08002572}
2573
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002574
2575static u16 copy_supp_rates(struct hostapd_data *hapd, struct sta_info *sta,
2576 struct ieee802_11_elems *elems)
2577{
Dmitry Shmidt29333592017-01-09 12:27:11 -08002578 /* Supported rates not used in IEEE 802.11ad/DMG */
2579 if (hapd->iface->current_mode &&
2580 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211AD)
2581 return WLAN_STATUS_SUCCESS;
2582
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002583 if (!elems->supp_rates) {
2584 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2585 HOSTAPD_LEVEL_DEBUG,
2586 "No supported rates element in AssocReq");
2587 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2588 }
2589
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002590 if (elems->supp_rates_len + elems->ext_supp_rates_len >
2591 sizeof(sta->supported_rates)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002592 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2593 HOSTAPD_LEVEL_DEBUG,
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002594 "Invalid supported rates element length %d+%d",
2595 elems->supp_rates_len,
2596 elems->ext_supp_rates_len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002597 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2598 }
2599
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002600 sta->supported_rates_len = merge_byte_arrays(
2601 sta->supported_rates, sizeof(sta->supported_rates),
2602 elems->supp_rates, elems->supp_rates_len,
2603 elems->ext_supp_rates, elems->ext_supp_rates_len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002604
2605 return WLAN_STATUS_SUCCESS;
2606}
2607
2608
Dmitry Shmidt051af732013-10-22 13:52:46 -07002609static u16 check_ext_capab(struct hostapd_data *hapd, struct sta_info *sta,
2610 const u8 *ext_capab_ie, size_t ext_capab_ie_len)
2611{
2612#ifdef CONFIG_INTERWORKING
2613 /* check for QoS Map support */
2614 if (ext_capab_ie_len >= 5) {
2615 if (ext_capab_ie[4] & 0x01)
2616 sta->qos_map_enabled = 1;
2617 }
2618#endif /* CONFIG_INTERWORKING */
2619
Roshan Pius3a1667e2018-07-03 15:17:14 -07002620 if (ext_capab_ie_len > 0) {
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002621 sta->ecsa_supported = !!(ext_capab_ie[0] & BIT(2));
Roshan Pius3a1667e2018-07-03 15:17:14 -07002622 os_free(sta->ext_capability);
2623 sta->ext_capability = os_malloc(1 + ext_capab_ie_len);
2624 if (sta->ext_capability) {
2625 sta->ext_capability[0] = ext_capab_ie_len;
2626 os_memcpy(sta->ext_capability + 1, ext_capab_ie,
2627 ext_capab_ie_len);
2628 }
2629 }
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002630
Dmitry Shmidt051af732013-10-22 13:52:46 -07002631 return WLAN_STATUS_SUCCESS;
2632}
2633
2634
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07002635#ifdef CONFIG_OWE
2636
2637static int owe_group_supported(struct hostapd_data *hapd, u16 group)
2638{
2639 int i;
2640 int *groups = hapd->conf->owe_groups;
2641
2642 if (group != 19 && group != 20 && group != 21)
2643 return 0;
2644
2645 if (!groups)
2646 return 1;
2647
2648 for (i = 0; groups[i] > 0; i++) {
2649 if (groups[i] == group)
2650 return 1;
2651 }
2652
2653 return 0;
2654}
2655
2656
2657static u16 owe_process_assoc_req(struct hostapd_data *hapd,
2658 struct sta_info *sta, const u8 *owe_dh,
2659 u8 owe_dh_len)
2660{
2661 struct wpabuf *secret, *pub, *hkey;
2662 int res;
2663 u8 prk[SHA512_MAC_LEN], pmkid[SHA512_MAC_LEN];
2664 const char *info = "OWE Key Generation";
2665 const u8 *addr[2];
2666 size_t len[2];
2667 u16 group;
2668 size_t hash_len, prime_len;
2669
2670 if (wpa_auth_sta_get_pmksa(sta->wpa_sm)) {
2671 wpa_printf(MSG_DEBUG, "OWE: Using PMKSA caching");
2672 return WLAN_STATUS_SUCCESS;
2673 }
2674
2675 group = WPA_GET_LE16(owe_dh);
2676 if (!owe_group_supported(hapd, group)) {
2677 wpa_printf(MSG_DEBUG, "OWE: Unsupported DH group %u", group);
2678 return WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED;
2679 }
2680 if (group == 19)
2681 prime_len = 32;
2682 else if (group == 20)
2683 prime_len = 48;
2684 else if (group == 21)
2685 prime_len = 66;
2686 else
2687 return WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED;
2688
2689 crypto_ecdh_deinit(sta->owe_ecdh);
2690 sta->owe_ecdh = crypto_ecdh_init(group);
2691 if (!sta->owe_ecdh)
2692 return WLAN_STATUS_FINITE_CYCLIC_GROUP_NOT_SUPPORTED;
2693 sta->owe_group = group;
2694
2695 secret = crypto_ecdh_set_peerkey(sta->owe_ecdh, 0, owe_dh + 2,
2696 owe_dh_len - 2);
2697 secret = wpabuf_zeropad(secret, prime_len);
2698 if (!secret) {
2699 wpa_printf(MSG_DEBUG, "OWE: Invalid peer DH public key");
2700 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2701 }
2702 wpa_hexdump_buf_key(MSG_DEBUG, "OWE: DH shared secret", secret);
2703
2704 /* prk = HKDF-extract(C | A | group, z) */
2705
2706 pub = crypto_ecdh_get_pubkey(sta->owe_ecdh, 0);
2707 if (!pub) {
2708 wpabuf_clear_free(secret);
2709 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2710 }
2711
2712 /* PMKID = Truncate-128(Hash(C | A)) */
2713 addr[0] = owe_dh + 2;
2714 len[0] = owe_dh_len - 2;
2715 addr[1] = wpabuf_head(pub);
2716 len[1] = wpabuf_len(pub);
2717 if (group == 19) {
2718 res = sha256_vector(2, addr, len, pmkid);
2719 hash_len = SHA256_MAC_LEN;
2720 } else if (group == 20) {
2721 res = sha384_vector(2, addr, len, pmkid);
2722 hash_len = SHA384_MAC_LEN;
2723 } else if (group == 21) {
2724 res = sha512_vector(2, addr, len, pmkid);
2725 hash_len = SHA512_MAC_LEN;
2726 } else {
2727 wpabuf_free(pub);
2728 wpabuf_clear_free(secret);
2729 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2730 }
2731 pub = wpabuf_zeropad(pub, prime_len);
2732 if (res < 0 || !pub) {
2733 wpabuf_free(pub);
2734 wpabuf_clear_free(secret);
2735 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2736 }
2737
2738 hkey = wpabuf_alloc(owe_dh_len - 2 + wpabuf_len(pub) + 2);
2739 if (!hkey) {
2740 wpabuf_free(pub);
2741 wpabuf_clear_free(secret);
2742 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2743 }
2744
2745 wpabuf_put_data(hkey, owe_dh + 2, owe_dh_len - 2); /* C */
2746 wpabuf_put_buf(hkey, pub); /* A */
2747 wpabuf_free(pub);
2748 wpabuf_put_le16(hkey, group); /* group */
2749 if (group == 19)
2750 res = hmac_sha256(wpabuf_head(hkey), wpabuf_len(hkey),
2751 wpabuf_head(secret), wpabuf_len(secret), prk);
2752 else if (group == 20)
2753 res = hmac_sha384(wpabuf_head(hkey), wpabuf_len(hkey),
2754 wpabuf_head(secret), wpabuf_len(secret), prk);
2755 else if (group == 21)
2756 res = hmac_sha512(wpabuf_head(hkey), wpabuf_len(hkey),
2757 wpabuf_head(secret), wpabuf_len(secret), prk);
2758 wpabuf_clear_free(hkey);
2759 wpabuf_clear_free(secret);
2760 if (res < 0)
2761 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2762
2763 wpa_hexdump_key(MSG_DEBUG, "OWE: prk", prk, hash_len);
2764
2765 /* PMK = HKDF-expand(prk, "OWE Key Generation", n) */
2766
2767 os_free(sta->owe_pmk);
2768 sta->owe_pmk = os_malloc(hash_len);
2769 if (!sta->owe_pmk) {
2770 os_memset(prk, 0, SHA512_MAC_LEN);
2771 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2772 }
2773
2774 if (group == 19)
2775 res = hmac_sha256_kdf(prk, hash_len, NULL, (const u8 *) info,
2776 os_strlen(info), sta->owe_pmk, hash_len);
2777 else if (group == 20)
2778 res = hmac_sha384_kdf(prk, hash_len, NULL, (const u8 *) info,
2779 os_strlen(info), sta->owe_pmk, hash_len);
2780 else if (group == 21)
2781 res = hmac_sha512_kdf(prk, hash_len, NULL, (const u8 *) info,
2782 os_strlen(info), sta->owe_pmk, hash_len);
2783 os_memset(prk, 0, SHA512_MAC_LEN);
2784 if (res < 0) {
2785 os_free(sta->owe_pmk);
2786 sta->owe_pmk = NULL;
2787 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2788 }
2789 sta->owe_pmk_len = hash_len;
2790
2791 wpa_hexdump_key(MSG_DEBUG, "OWE: PMK", sta->owe_pmk, sta->owe_pmk_len);
2792 wpa_hexdump(MSG_DEBUG, "OWE: PMKID", pmkid, PMKID_LEN);
2793 wpa_auth_pmksa_add2(hapd->wpa_auth, sta->addr, sta->owe_pmk,
2794 sta->owe_pmk_len, pmkid, 0, WPA_KEY_MGMT_OWE);
2795
2796 return WLAN_STATUS_SUCCESS;
2797}
2798
Hai Shalom81f62d82019-07-22 12:10:00 -07002799
2800u16 owe_validate_request(struct hostapd_data *hapd, const u8 *peer,
2801 const u8 *rsn_ie, size_t rsn_ie_len,
2802 const u8 *owe_dh, size_t owe_dh_len)
2803{
2804 struct wpa_ie_data data;
2805 int res;
2806
2807 if (!rsn_ie || rsn_ie_len < 2) {
2808 wpa_printf(MSG_DEBUG, "OWE: Invalid RSNE from " MACSTR,
2809 MAC2STR(peer));
2810 return WLAN_STATUS_INVALID_IE;
2811 }
2812 rsn_ie -= 2;
2813 rsn_ie_len += 2;
2814
2815 res = wpa_parse_wpa_ie_rsn(rsn_ie, rsn_ie_len, &data);
2816 if (res) {
2817 wpa_printf(MSG_DEBUG, "Failed to parse RSNE from " MACSTR
2818 " (res=%d)", MAC2STR(peer), res);
2819 wpa_hexdump(MSG_DEBUG, "RSNE", rsn_ie, rsn_ie_len);
2820 return wpa_res_to_status_code(res);
2821 }
2822 if (!(data.key_mgmt & WPA_KEY_MGMT_OWE)) {
2823 wpa_printf(MSG_DEBUG,
2824 "OWE: Unexpected key mgmt 0x%x from " MACSTR,
2825 (unsigned int) data.key_mgmt, MAC2STR(peer));
2826 return WLAN_STATUS_AKMP_NOT_VALID;
2827 }
2828 if (!owe_dh) {
2829 wpa_printf(MSG_DEBUG,
2830 "OWE: No Diffie-Hellman Parameter element from "
2831 MACSTR, MAC2STR(peer));
2832 return WLAN_STATUS_AKMP_NOT_VALID;
2833 }
2834
2835 return WLAN_STATUS_SUCCESS;
2836}
2837
2838
2839u16 owe_process_rsn_ie(struct hostapd_data *hapd,
2840 struct sta_info *sta,
2841 const u8 *rsn_ie, size_t rsn_ie_len,
2842 const u8 *owe_dh, size_t owe_dh_len)
2843{
2844 u16 status;
2845 u8 *owe_buf, ie[256 * 2];
2846 size_t ie_len = 0;
2847 int res;
2848
2849 if (!rsn_ie || rsn_ie_len < 2) {
2850 wpa_printf(MSG_DEBUG, "OWE: No RSNE in (Re)AssocReq");
2851 status = WLAN_STATUS_INVALID_IE;
2852 goto end;
2853 }
2854
2855 if (!sta->wpa_sm)
2856 sta->wpa_sm = wpa_auth_sta_init(hapd->wpa_auth, sta->addr,
2857 NULL);
2858 if (!sta->wpa_sm) {
2859 wpa_printf(MSG_WARNING,
2860 "OWE: Failed to initialize WPA state machine");
2861 status = WLAN_STATUS_UNSPECIFIED_FAILURE;
2862 goto end;
2863 }
2864 rsn_ie -= 2;
2865 rsn_ie_len += 2;
2866 res = wpa_validate_wpa_ie(hapd->wpa_auth, sta->wpa_sm,
2867 hapd->iface->freq, rsn_ie, rsn_ie_len,
2868 NULL, 0, owe_dh, owe_dh_len);
2869 status = wpa_res_to_status_code(res);
2870 if (status != WLAN_STATUS_SUCCESS)
2871 goto end;
2872 status = owe_process_assoc_req(hapd, sta, owe_dh, owe_dh_len);
2873 if (status != WLAN_STATUS_SUCCESS)
2874 goto end;
2875 owe_buf = wpa_auth_write_assoc_resp_owe(sta->wpa_sm, ie, sizeof(ie),
2876 NULL, 0);
2877 if (!owe_buf) {
2878 status = WLAN_STATUS_UNSPECIFIED_FAILURE;
2879 goto end;
2880 }
2881
2882 if (sta->owe_ecdh) {
2883 struct wpabuf *pub;
2884
2885 pub = crypto_ecdh_get_pubkey(sta->owe_ecdh, 0);
2886 if (!pub) {
2887 status = WLAN_STATUS_UNSPECIFIED_FAILURE;
2888 goto end;
2889 }
2890
2891 /* OWE Diffie-Hellman Parameter element */
2892 *owe_buf++ = WLAN_EID_EXTENSION; /* Element ID */
2893 *owe_buf++ = 1 + 2 + wpabuf_len(pub); /* Length */
2894 *owe_buf++ = WLAN_EID_EXT_OWE_DH_PARAM; /* Element ID Extension
2895 */
2896 WPA_PUT_LE16(owe_buf, sta->owe_group);
2897 owe_buf += 2;
2898 os_memcpy(owe_buf, wpabuf_head(pub), wpabuf_len(pub));
2899 owe_buf += wpabuf_len(pub);
2900 wpabuf_free(pub);
2901 sta->external_dh_updated = 1;
2902 }
2903 ie_len = owe_buf - ie;
2904
2905end:
2906 wpa_printf(MSG_DEBUG, "OWE: Update status %d, ie len %d for peer "
2907 MACSTR, status, (unsigned int) ie_len,
2908 MAC2STR(sta->addr));
2909 hostapd_drv_update_dh_ie(hapd, sta->addr, status,
2910 status == WLAN_STATUS_SUCCESS ? ie : NULL,
2911 ie_len);
2912
2913 return status;
2914}
2915
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07002916#endif /* CONFIG_OWE */
2917
2918
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002919static u16 check_assoc_ies(struct hostapd_data *hapd, struct sta_info *sta,
2920 const u8 *ies, size_t ies_len, int reassoc)
2921{
2922 struct ieee802_11_elems elems;
2923 u16 resp;
2924 const u8 *wpa_ie;
2925 size_t wpa_ie_len;
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07002926 const u8 *p2p_dev_addr = NULL;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002927
2928 if (ieee802_11_parse_elems(ies, ies_len, &elems, 1) == ParseFailed) {
2929 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2930 HOSTAPD_LEVEL_INFO, "Station sent an invalid "
2931 "association request");
2932 return WLAN_STATUS_UNSPECIFIED_FAILURE;
2933 }
2934
2935 resp = check_ssid(hapd, sta, elems.ssid, elems.ssid_len);
2936 if (resp != WLAN_STATUS_SUCCESS)
2937 return resp;
2938 resp = check_wmm(hapd, sta, elems.wmm, elems.wmm_len);
2939 if (resp != WLAN_STATUS_SUCCESS)
2940 return resp;
Dmitry Shmidt051af732013-10-22 13:52:46 -07002941 resp = check_ext_capab(hapd, sta, elems.ext_capab, elems.ext_capab_len);
2942 if (resp != WLAN_STATUS_SUCCESS)
2943 return resp;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002944 resp = copy_supp_rates(hapd, sta, &elems);
2945 if (resp != WLAN_STATUS_SUCCESS)
2946 return resp;
Hai Shalom74f70d42019-02-11 14:42:39 -08002947
2948 resp = check_multi_ap(hapd, sta, elems.multi_ap, elems.multi_ap_len);
2949 if (resp != WLAN_STATUS_SUCCESS)
2950 return resp;
2951
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002952#ifdef CONFIG_IEEE80211N
Dmitry Shmidt9d9e6022015-04-23 10:34:55 -07002953 resp = copy_sta_ht_capab(hapd, sta, elems.ht_capabilities);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07002954 if (resp != WLAN_STATUS_SUCCESS)
2955 return resp;
2956 if (hapd->iconf->ieee80211n && hapd->iconf->require_ht &&
2957 !(sta->flags & WLAN_STA_HT)) {
2958 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2959 HOSTAPD_LEVEL_INFO, "Station does not support "
2960 "mandatory HT PHY - reject association");
2961 return WLAN_STATUS_ASSOC_DENIED_NO_HT;
2962 }
2963#endif /* CONFIG_IEEE80211N */
2964
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002965#ifdef CONFIG_IEEE80211AC
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002966 if (hapd->iconf->ieee80211ac) {
2967 resp = copy_sta_vht_capab(hapd, sta, elems.vht_capabilities);
2968 if (resp != WLAN_STATUS_SUCCESS)
2969 return resp;
Dmitry Shmidtbd14a572014-02-18 10:33:49 -08002970
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08002971 resp = set_sta_vht_opmode(hapd, sta, elems.vht_opmode_notif);
2972 if (resp != WLAN_STATUS_SUCCESS)
2973 return resp;
2974 }
Dmitry Shmidtbd14a572014-02-18 10:33:49 -08002975
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002976 if (hapd->iconf->ieee80211ac && hapd->iconf->require_vht &&
2977 !(sta->flags & WLAN_STA_VHT)) {
2978 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
2979 HOSTAPD_LEVEL_INFO, "Station does not support "
2980 "mandatory VHT PHY - reject association");
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08002981 return WLAN_STATUS_ASSOC_DENIED_NO_VHT;
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002982 }
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08002983
2984 if (hapd->conf->vendor_vht && !elems.vht_capabilities) {
2985 resp = copy_sta_vendor_vht(hapd, sta, elems.vendor_vht,
2986 elems.vendor_vht_len);
2987 if (resp != WLAN_STATUS_SUCCESS)
2988 return resp;
2989 }
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07002990#endif /* CONFIG_IEEE80211AC */
Hai Shalom81f62d82019-07-22 12:10:00 -07002991#ifdef CONFIG_IEEE80211AX
2992 if (hapd->iconf->ieee80211ax) {
2993 resp = copy_sta_he_capab(hapd, sta, IEEE80211_MODE_AP,
2994 elems.he_capabilities,
2995 elems.he_capabilities_len);
2996 if (resp != WLAN_STATUS_SUCCESS)
2997 return resp;
2998 }
2999#endif /* CONFIG_IEEE80211AX */
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003000
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07003001#ifdef CONFIG_P2P
3002 if (elems.p2p) {
3003 wpabuf_free(sta->p2p_ie);
3004 sta->p2p_ie = ieee802_11_vendor_ie_concat(ies, ies_len,
3005 P2P_IE_VENDOR_TYPE);
3006 if (sta->p2p_ie)
3007 p2p_dev_addr = p2p_get_go_dev_addr(sta->p2p_ie);
3008 } else {
3009 wpabuf_free(sta->p2p_ie);
3010 sta->p2p_ie = NULL;
3011 }
3012#endif /* CONFIG_P2P */
3013
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003014 if ((hapd->conf->wpa & WPA_PROTO_RSN) && elems.rsn_ie) {
3015 wpa_ie = elems.rsn_ie;
3016 wpa_ie_len = elems.rsn_ie_len;
3017 } else if ((hapd->conf->wpa & WPA_PROTO_WPA) &&
3018 elems.wpa_ie) {
3019 wpa_ie = elems.wpa_ie;
3020 wpa_ie_len = elems.wpa_ie_len;
3021 } else {
3022 wpa_ie = NULL;
3023 wpa_ie_len = 0;
3024 }
3025
3026#ifdef CONFIG_WPS
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003027 sta->flags &= ~(WLAN_STA_WPS | WLAN_STA_MAYBE_WPS | WLAN_STA_WPS2);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003028 if (hapd->conf->wps_state && elems.wps_ie) {
3029 wpa_printf(MSG_DEBUG, "STA included WPS IE in (Re)Association "
3030 "Request - assume WPS is used");
3031 sta->flags |= WLAN_STA_WPS;
3032 wpabuf_free(sta->wps_ie);
3033 sta->wps_ie = ieee802_11_vendor_ie_concat(ies, ies_len,
3034 WPS_IE_VENDOR_TYPE);
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003035 if (sta->wps_ie && wps_is_20(sta->wps_ie)) {
3036 wpa_printf(MSG_DEBUG, "WPS: STA supports WPS 2.0");
3037 sta->flags |= WLAN_STA_WPS2;
3038 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003039 wpa_ie = NULL;
3040 wpa_ie_len = 0;
3041 if (sta->wps_ie && wps_validate_assoc_req(sta->wps_ie) < 0) {
3042 wpa_printf(MSG_DEBUG, "WPS: Invalid WPS IE in "
3043 "(Re)Association Request - reject");
3044 return WLAN_STATUS_INVALID_IE;
3045 }
3046 } else if (hapd->conf->wps_state && wpa_ie == NULL) {
3047 wpa_printf(MSG_DEBUG, "STA did not include WPA/RSN IE in "
3048 "(Re)Association Request - possible WPS use");
3049 sta->flags |= WLAN_STA_MAYBE_WPS;
3050 } else
3051#endif /* CONFIG_WPS */
3052 if (hapd->conf->wpa && wpa_ie == NULL) {
3053 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
3054 HOSTAPD_LEVEL_INFO,
3055 "No WPA/RSN IE in association request");
3056 return WLAN_STATUS_INVALID_IE;
3057 }
3058
3059 if (hapd->conf->wpa && wpa_ie) {
3060 int res;
3061 wpa_ie -= 2;
3062 wpa_ie_len += 2;
3063 if (sta->wpa_sm == NULL)
3064 sta->wpa_sm = wpa_auth_sta_init(hapd->wpa_auth,
Dmitry Shmidt391c59f2013-09-03 12:16:28 -07003065 sta->addr,
3066 p2p_dev_addr);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003067 if (sta->wpa_sm == NULL) {
3068 wpa_printf(MSG_WARNING, "Failed to initialize WPA "
3069 "state machine");
3070 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3071 }
Hai Shalom021b0b52019-04-10 11:17:58 -07003072 wpa_auth_set_auth_alg(sta->wpa_sm, sta->auth_alg);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003073 res = wpa_validate_wpa_ie(hapd->wpa_auth, sta->wpa_sm,
Hai Shalom021b0b52019-04-10 11:17:58 -07003074 hapd->iface->freq,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003075 wpa_ie, wpa_ie_len,
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003076 elems.mdie, elems.mdie_len,
3077 elems.owe_dh, elems.owe_dh_len);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003078 resp = wpa_res_to_status_code(res);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003079 if (resp != WLAN_STATUS_SUCCESS)
3080 return resp;
3081#ifdef CONFIG_IEEE80211W
Hai Shalom39ba6fc2019-01-22 12:40:38 -08003082 if ((sta->flags & (WLAN_STA_ASSOC | WLAN_STA_MFP)) ==
3083 (WLAN_STA_ASSOC | WLAN_STA_MFP) &&
3084 !sta->sa_query_timed_out &&
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003085 sta->sa_query_count > 0)
3086 ap_check_sa_query_timeout(hapd, sta);
Hai Shalom39ba6fc2019-01-22 12:40:38 -08003087 if ((sta->flags & (WLAN_STA_ASSOC | WLAN_STA_MFP)) ==
3088 (WLAN_STA_ASSOC | WLAN_STA_MFP) &&
3089 !sta->sa_query_timed_out &&
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003090 (!reassoc || sta->auth_alg != WLAN_AUTH_FT)) {
3091 /*
3092 * STA has already been associated with MFP and SA
3093 * Query timeout has not been reached. Reject the
3094 * association attempt temporarily and start SA Query,
3095 * if one is not pending.
3096 */
3097
3098 if (sta->sa_query_count == 0)
3099 ap_sta_start_sa_query(hapd, sta);
3100
3101 return WLAN_STATUS_ASSOC_REJECTED_TEMPORARILY;
3102 }
3103
3104 if (wpa_auth_uses_mfp(sta->wpa_sm))
3105 sta->flags |= WLAN_STA_MFP;
3106 else
3107 sta->flags &= ~WLAN_STA_MFP;
3108#endif /* CONFIG_IEEE80211W */
3109
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003110#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003111 if (sta->auth_alg == WLAN_AUTH_FT) {
3112 if (!reassoc) {
3113 wpa_printf(MSG_DEBUG, "FT: " MACSTR " tried "
3114 "to use association (not "
3115 "re-association) with FT auth_alg",
3116 MAC2STR(sta->addr));
3117 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3118 }
3119
3120 resp = wpa_ft_validate_reassoc(sta->wpa_sm, ies,
3121 ies_len);
3122 if (resp != WLAN_STATUS_SUCCESS)
3123 return resp;
3124 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003125#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003126
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003127#ifdef CONFIG_SAE
Roshan Pius3a1667e2018-07-03 15:17:14 -07003128 if (wpa_auth_uses_sae(sta->wpa_sm) && sta->sae &&
3129 sta->sae->state == SAE_ACCEPTED)
3130 wpa_auth_add_sae_pmkid(sta->wpa_sm, sta->sae->pmkid);
3131
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003132 if (wpa_auth_uses_sae(sta->wpa_sm) &&
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003133 sta->auth_alg == WLAN_AUTH_OPEN) {
3134 struct rsn_pmksa_cache_entry *sa;
3135 sa = wpa_auth_sta_get_pmksa(sta->wpa_sm);
3136 if (!sa || sa->akmp != WPA_KEY_MGMT_SAE) {
3137 wpa_printf(MSG_DEBUG,
3138 "SAE: No PMKSA cache entry found for "
3139 MACSTR, MAC2STR(sta->addr));
3140 return WLAN_STATUS_INVALID_PMKID;
3141 }
3142 wpa_printf(MSG_DEBUG, "SAE: " MACSTR
3143 " using PMKSA caching", MAC2STR(sta->addr));
3144 } else if (wpa_auth_uses_sae(sta->wpa_sm) &&
3145 sta->auth_alg != WLAN_AUTH_SAE &&
3146 !(sta->auth_alg == WLAN_AUTH_FT &&
3147 wpa_auth_uses_ft_sae(sta->wpa_sm))) {
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003148 wpa_printf(MSG_DEBUG, "SAE: " MACSTR " tried to use "
3149 "SAE AKM after non-SAE auth_alg %u",
3150 MAC2STR(sta->addr), sta->auth_alg);
3151 return WLAN_STATUS_NOT_SUPPORTED_AUTH_ALG;
3152 }
3153#endif /* CONFIG_SAE */
3154
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003155#ifdef CONFIG_OWE
3156 if ((hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_OWE) &&
3157 wpa_auth_sta_key_mgmt(sta->wpa_sm) == WPA_KEY_MGMT_OWE &&
3158 elems.owe_dh) {
3159 resp = owe_process_assoc_req(hapd, sta, elems.owe_dh,
3160 elems.owe_dh_len);
3161 if (resp != WLAN_STATUS_SUCCESS)
3162 return resp;
3163 }
3164#endif /* CONFIG_OWE */
3165
Hai Shalom021b0b52019-04-10 11:17:58 -07003166#ifdef CONFIG_DPP2
3167 dpp_pfs_free(sta->dpp_pfs);
3168 sta->dpp_pfs = NULL;
3169
3170 if ((hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_DPP) &&
3171 hapd->conf->dpp_netaccesskey && sta->wpa_sm &&
3172 wpa_auth_sta_key_mgmt(sta->wpa_sm) == WPA_KEY_MGMT_DPP &&
3173 elems.owe_dh) {
3174 sta->dpp_pfs = dpp_pfs_init(
3175 wpabuf_head(hapd->conf->dpp_netaccesskey),
3176 wpabuf_len(hapd->conf->dpp_netaccesskey));
3177 if (!sta->dpp_pfs) {
3178 wpa_printf(MSG_DEBUG,
3179 "DPP: Could not initialize PFS");
3180 /* Try to continue without PFS */
3181 goto pfs_fail;
3182 }
3183
3184 if (dpp_pfs_process(sta->dpp_pfs, elems.owe_dh,
3185 elems.owe_dh_len) < 0) {
3186 dpp_pfs_free(sta->dpp_pfs);
3187 sta->dpp_pfs = NULL;
3188 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3189 }
3190 }
3191
3192 wpa_auth_set_dpp_z(sta->wpa_sm, sta->dpp_pfs ?
3193 sta->dpp_pfs->secret : NULL);
3194 pfs_fail:
3195#endif /* CONFIG_DPP2 */
3196
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003197#ifdef CONFIG_IEEE80211N
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003198 if ((sta->flags & (WLAN_STA_HT | WLAN_STA_VHT)) &&
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003199 wpa_auth_get_pairwise(sta->wpa_sm) == WPA_CIPHER_TKIP) {
3200 hostapd_logger(hapd, sta->addr,
3201 HOSTAPD_MODULE_IEEE80211,
3202 HOSTAPD_LEVEL_INFO,
3203 "Station tried to use TKIP with HT "
3204 "association");
3205 return WLAN_STATUS_CIPHER_REJECTED_PER_POLICY;
3206 }
3207#endif /* CONFIG_IEEE80211N */
Dmitry Shmidtf21452a2014-02-26 10:55:25 -08003208#ifdef CONFIG_HS20
3209 } else if (hapd->conf->osen) {
3210 if (elems.osen == NULL) {
3211 hostapd_logger(
3212 hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
3213 HOSTAPD_LEVEL_INFO,
3214 "No HS 2.0 OSEN element in association request");
3215 return WLAN_STATUS_INVALID_IE;
3216 }
3217
3218 wpa_printf(MSG_DEBUG, "HS 2.0: OSEN association");
3219 if (sta->wpa_sm == NULL)
3220 sta->wpa_sm = wpa_auth_sta_init(hapd->wpa_auth,
3221 sta->addr, NULL);
3222 if (sta->wpa_sm == NULL) {
3223 wpa_printf(MSG_WARNING, "Failed to initialize WPA "
3224 "state machine");
3225 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3226 }
3227 if (wpa_validate_osen(hapd->wpa_auth, sta->wpa_sm,
3228 elems.osen - 2, elems.osen_len + 2) < 0)
3229 return WLAN_STATUS_INVALID_IE;
3230#endif /* CONFIG_HS20 */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003231 } else
3232 wpa_auth_sta_no_wpa(sta->wpa_sm);
3233
3234#ifdef CONFIG_P2P
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003235 p2p_group_notif_assoc(hapd->p2p_group, sta->addr, ies, ies_len);
3236#endif /* CONFIG_P2P */
3237
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003238#ifdef CONFIG_HS20
3239 wpabuf_free(sta->hs20_ie);
3240 if (elems.hs20 && elems.hs20_len > 4) {
Hai Shalom74f70d42019-02-11 14:42:39 -08003241 int release;
3242
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003243 sta->hs20_ie = wpabuf_alloc_copy(elems.hs20 + 4,
3244 elems.hs20_len - 4);
Hai Shalom74f70d42019-02-11 14:42:39 -08003245 release = ((elems.hs20[4] >> 4) & 0x0f) + 1;
3246 if (release >= 2 && !wpa_auth_uses_mfp(sta->wpa_sm)) {
3247 wpa_printf(MSG_DEBUG,
3248 "HS 2.0: PMF not negotiated by release %d station "
3249 MACSTR, release, MAC2STR(sta->addr));
3250 return WLAN_STATUS_ROBUST_MGMT_FRAME_POLICY_VIOLATION;
3251 }
3252 } else {
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003253 sta->hs20_ie = NULL;
Hai Shalom74f70d42019-02-11 14:42:39 -08003254 }
Roshan Pius3a1667e2018-07-03 15:17:14 -07003255
3256 wpabuf_free(sta->roaming_consortium);
3257 if (elems.roaming_cons_sel)
3258 sta->roaming_consortium = wpabuf_alloc_copy(
3259 elems.roaming_cons_sel + 4,
3260 elems.roaming_cons_sel_len - 4);
3261 else
3262 sta->roaming_consortium = NULL;
Dmitry Shmidtd5e49232012-12-03 15:08:10 -08003263#endif /* CONFIG_HS20 */
3264
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08003265#ifdef CONFIG_FST
3266 wpabuf_free(sta->mb_ies);
3267 if (hapd->iface->fst)
3268 sta->mb_ies = mb_ies_by_info(&elems.mb_ies);
3269 else
3270 sta->mb_ies = NULL;
3271#endif /* CONFIG_FST */
3272
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003273#ifdef CONFIG_MBO
3274 mbo_ap_check_sta_assoc(hapd, sta, &elems);
3275
3276 if (hapd->conf->mbo_enabled && (hapd->conf->wpa & 2) &&
3277 elems.mbo && sta->cell_capa && !(sta->flags & WLAN_STA_MFP) &&
3278 hapd->conf->ieee80211w != NO_MGMT_FRAME_PROTECTION) {
3279 wpa_printf(MSG_INFO,
3280 "MBO: Reject WPA2 association without PMF");
3281 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3282 }
3283#endif /* CONFIG_MBO */
3284
Hai Shalom74f70d42019-02-11 14:42:39 -08003285#if defined(CONFIG_FILS) && defined(CONFIG_OCV)
3286 if (wpa_auth_uses_ocv(sta->wpa_sm) &&
3287 (sta->auth_alg == WLAN_AUTH_FILS_SK ||
3288 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
3289 sta->auth_alg == WLAN_AUTH_FILS_PK)) {
3290 struct wpa_channel_info ci;
3291 int tx_chanwidth;
3292 int tx_seg1_idx;
3293
3294 if (hostapd_drv_channel_info(hapd, &ci) != 0) {
3295 wpa_printf(MSG_WARNING,
3296 "Failed to get channel info to validate received OCI in FILS (Re)Association Request frame");
3297 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3298 }
3299
3300 if (get_sta_tx_parameters(sta->wpa_sm,
3301 channel_width_to_int(ci.chanwidth),
3302 ci.seg1_idx, &tx_chanwidth,
3303 &tx_seg1_idx) < 0)
3304 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3305
3306 if (ocv_verify_tx_params(elems.oci, elems.oci_len, &ci,
3307 tx_chanwidth, tx_seg1_idx) != 0) {
3308 wpa_printf(MSG_WARNING, "FILS: %s", ocv_errorstr);
3309 return WLAN_STATUS_UNSPECIFIED_FAILURE;
3310 }
3311 }
3312#endif /* CONFIG_FILS && CONFIG_OCV */
3313
Dmitry Shmidt9c175262016-03-03 10:20:07 -08003314 ap_copy_sta_supp_op_classes(sta, elems.supp_op_classes,
3315 elems.supp_op_classes_len);
3316
Dmitry Shmidt849734c2016-05-27 09:59:01 -07003317 if ((sta->capability & WLAN_CAPABILITY_RADIO_MEASUREMENT) &&
3318 elems.rrm_enabled &&
3319 elems.rrm_enabled_len >= sizeof(sta->rrm_enabled_capa))
3320 os_memcpy(sta->rrm_enabled_capa, elems.rrm_enabled,
3321 sizeof(sta->rrm_enabled_capa));
3322
Roshan Pius3a1667e2018-07-03 15:17:14 -07003323 if (elems.power_capab) {
3324 sta->min_tx_power = elems.power_capab[0];
3325 sta->max_tx_power = elems.power_capab[1];
3326 sta->power_capab = 1;
3327 } else {
3328 sta->power_capab = 0;
3329 }
3330
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003331 return WLAN_STATUS_SUCCESS;
3332}
3333
3334
3335static void send_deauth(struct hostapd_data *hapd, const u8 *addr,
3336 u16 reason_code)
3337{
3338 int send_len;
3339 struct ieee80211_mgmt reply;
3340
3341 os_memset(&reply, 0, sizeof(reply));
3342 reply.frame_control =
3343 IEEE80211_FC(WLAN_FC_TYPE_MGMT, WLAN_FC_STYPE_DEAUTH);
3344 os_memcpy(reply.da, addr, ETH_ALEN);
3345 os_memcpy(reply.sa, hapd->own_addr, ETH_ALEN);
3346 os_memcpy(reply.bssid, hapd->own_addr, ETH_ALEN);
3347
3348 send_len = IEEE80211_HDRLEN + sizeof(reply.u.deauth);
3349 reply.u.deauth.reason_code = host_to_le16(reason_code);
3350
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08003351 if (hostapd_drv_send_mlme(hapd, &reply, send_len, 0) < 0)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003352 wpa_printf(MSG_INFO, "Failed to send deauth: %s",
3353 strerror(errno));
3354}
3355
3356
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003357static int add_associated_sta(struct hostapd_data *hapd,
Hai Shalom74f70d42019-02-11 14:42:39 -08003358 struct sta_info *sta, int reassoc)
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003359{
3360 struct ieee80211_ht_capabilities ht_cap;
3361 struct ieee80211_vht_capabilities vht_cap;
Hai Shalom81f62d82019-07-22 12:10:00 -07003362 struct ieee80211_he_capabilities he_cap;
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003363 int set = 1;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003364
3365 /*
3366 * Remove the STA entry to ensure the STA PS state gets cleared and
3367 * configuration gets updated. This is relevant for cases, such as
3368 * FT-over-the-DS, where a station re-associates back to the same AP but
3369 * skips the authentication flow, or if working with a driver that
3370 * does not support full AP client state.
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003371 *
3372 * Skip this if the STA has already completed FT reassociation and the
3373 * TK has been configured since the TX/RX PN must not be reset to 0 for
3374 * the same key.
Hai Shalom74f70d42019-02-11 14:42:39 -08003375 *
3376 * FT-over-the-DS has a special case where the STA entry (and as such,
3377 * the TK) has not yet been configured to the driver depending on which
3378 * driver interface is used. For that case, allow add-STA operation to
3379 * be used (instead of set-STA). This is needed to allow mac80211-based
3380 * drivers to accept the STA parameter configuration. Since this is
3381 * after a new FT-over-DS exchange, a new TK has been derived, so key
3382 * reinstallation is not a concern for this case.
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003383 */
Hai Shalom74f70d42019-02-11 14:42:39 -08003384 wpa_printf(MSG_DEBUG, "Add associated STA " MACSTR
3385 " (added_unassoc=%d auth_alg=%u ft_over_ds=%u reassoc=%d authorized=%d ft_tk=%d fils_tk=%d)",
3386 MAC2STR(sta->addr), sta->added_unassoc, sta->auth_alg,
3387 sta->ft_over_ds, reassoc,
3388 !!(sta->flags & WLAN_STA_AUTHORIZED),
3389 wpa_auth_sta_ft_tk_already_set(sta->wpa_sm),
3390 wpa_auth_sta_fils_tk_already_set(sta->wpa_sm));
3391
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003392 if (!sta->added_unassoc &&
3393 (!(sta->flags & WLAN_STA_AUTHORIZED) ||
Hai Shalom74f70d42019-02-11 14:42:39 -08003394 (reassoc && sta->ft_over_ds && sta->auth_alg == WLAN_AUTH_FT) ||
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003395 (!wpa_auth_sta_ft_tk_already_set(sta->wpa_sm) &&
3396 !wpa_auth_sta_fils_tk_already_set(sta->wpa_sm)))) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003397 hostapd_drv_sta_remove(hapd, sta->addr);
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003398 wpa_auth_sm_event(sta->wpa_sm, WPA_DRV_STA_REMOVED);
3399 set = 0;
Hai Shalom74f70d42019-02-11 14:42:39 -08003400
3401 /* Do not allow the FT-over-DS exception to be used more than
3402 * once per authentication exchange to guarantee a new TK is
3403 * used here */
3404 sta->ft_over_ds = 0;
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003405 }
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003406
3407#ifdef CONFIG_IEEE80211N
3408 if (sta->flags & WLAN_STA_HT)
3409 hostapd_get_ht_capab(hapd, sta->ht_capabilities, &ht_cap);
3410#endif /* CONFIG_IEEE80211N */
3411#ifdef CONFIG_IEEE80211AC
3412 if (sta->flags & WLAN_STA_VHT)
3413 hostapd_get_vht_capab(hapd, sta->vht_capabilities, &vht_cap);
3414#endif /* CONFIG_IEEE80211AC */
Hai Shalom81f62d82019-07-22 12:10:00 -07003415#ifdef CONFIG_IEEE80211AX
3416 if (sta->flags & WLAN_STA_HE) {
3417 hostapd_get_he_capab(hapd, sta->he_capab, &he_cap,
3418 sta->he_capab_len);
3419 }
3420#endif /* CONFIG_IEEE80211AX */
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003421
3422 /*
3423 * Add the station with forced WLAN_STA_ASSOC flag. The sta->flags
3424 * will be set when the ACK frame for the (Re)Association Response frame
3425 * is processed (TX status driver event).
3426 */
3427 if (hostapd_sta_add(hapd, sta->addr, sta->aid, sta->capability,
3428 sta->supported_rates, sta->supported_rates_len,
3429 sta->listen_interval,
3430 sta->flags & WLAN_STA_HT ? &ht_cap : NULL,
3431 sta->flags & WLAN_STA_VHT ? &vht_cap : NULL,
Hai Shalom81f62d82019-07-22 12:10:00 -07003432 sta->flags & WLAN_STA_HE ? &he_cap : NULL,
3433 sta->flags & WLAN_STA_HE ? sta->he_capab_len : 0,
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003434 sta->flags | WLAN_STA_ASSOC, sta->qosinfo,
Dmitry Shmidt849734c2016-05-27 09:59:01 -07003435 sta->vht_opmode, sta->p2p_ie ? 1 : 0,
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003436 set)) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003437 hostapd_logger(hapd, sta->addr,
3438 HOSTAPD_MODULE_IEEE80211, HOSTAPD_LEVEL_NOTICE,
3439 "Could not %s STA to kernel driver",
Mathy Vanhoeff6e1f662017-07-14 15:15:35 +02003440 set ? "set" : "add");
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003441
3442 if (sta->added_unassoc) {
3443 hostapd_drv_sta_remove(hapd, sta->addr);
3444 sta->added_unassoc = 0;
3445 }
3446
3447 return -1;
3448 }
3449
3450 sta->added_unassoc = 0;
3451
3452 return 0;
3453}
3454
3455
3456static u16 send_assoc_resp(struct hostapd_data *hapd, struct sta_info *sta,
Dmitry Shmidt29333592017-01-09 12:27:11 -08003457 const u8 *addr, u16 status_code, int reassoc,
Hai Shalom74f70d42019-02-11 14:42:39 -08003458 const u8 *ies, size_t ies_len, int rssi)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003459{
3460 int send_len;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003461 u8 *buf;
3462 size_t buflen;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003463 struct ieee80211_mgmt *reply;
3464 u8 *p;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003465 u16 res = WLAN_STATUS_SUCCESS;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003466
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003467 buflen = sizeof(struct ieee80211_mgmt) + 1024;
3468#ifdef CONFIG_FILS
3469 if (sta && sta->fils_hlp_resp)
3470 buflen += wpabuf_len(sta->fils_hlp_resp);
Hai Shalom81f62d82019-07-22 12:10:00 -07003471 if (sta)
3472 buflen += 150;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003473#endif /* CONFIG_FILS */
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003474#ifdef CONFIG_OWE
3475 if (sta && (hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_OWE))
3476 buflen += 150;
3477#endif /* CONFIG_OWE */
Hai Shalom021b0b52019-04-10 11:17:58 -07003478#ifdef CONFIG_DPP2
3479 if (sta && sta->dpp_pfs)
3480 buflen += 5 + sta->dpp_pfs->curve->prime_len;
3481#endif /* CONFIG_DPP2 */
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003482 buf = os_zalloc(buflen);
3483 if (!buf) {
3484 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
3485 goto done;
3486 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003487 reply = (struct ieee80211_mgmt *) buf;
3488 reply->frame_control =
3489 IEEE80211_FC(WLAN_FC_TYPE_MGMT,
3490 (reassoc ? WLAN_FC_STYPE_REASSOC_RESP :
3491 WLAN_FC_STYPE_ASSOC_RESP));
Dmitry Shmidt29333592017-01-09 12:27:11 -08003492 os_memcpy(reply->da, addr, ETH_ALEN);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003493 os_memcpy(reply->sa, hapd->own_addr, ETH_ALEN);
3494 os_memcpy(reply->bssid, hapd->own_addr, ETH_ALEN);
3495
3496 send_len = IEEE80211_HDRLEN;
3497 send_len += sizeof(reply->u.assoc_resp);
3498 reply->u.assoc_resp.capab_info =
Dmitry Shmidt9d9e6022015-04-23 10:34:55 -07003499 host_to_le16(hostapd_own_capab_info(hapd));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003500 reply->u.assoc_resp.status_code = host_to_le16(status_code);
Dmitry Shmidt29333592017-01-09 12:27:11 -08003501
3502 reply->u.assoc_resp.aid = host_to_le16((sta ? sta->aid : 0) |
3503 BIT(14) | BIT(15));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003504 /* Supported rates */
3505 p = hostapd_eid_supp_rates(hapd, reply->u.assoc_resp.variable);
3506 /* Extended supported rates */
3507 p = hostapd_eid_ext_supp_rates(hapd, p);
3508
Hai Shalom74f70d42019-02-11 14:42:39 -08003509#ifdef CONFIG_MBO
3510 if (status_code == WLAN_STATUS_DENIED_POOR_CHANNEL_CONDITIONS &&
3511 rssi != 0) {
3512 int delta = hapd->iconf->rssi_reject_assoc_rssi - rssi;
3513
3514 p = hostapd_eid_mbo_rssi_assoc_rej(hapd, p, buf + buflen - p,
3515 delta);
3516 }
3517#endif /* CONFIG_MBO */
3518
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003519#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt29333592017-01-09 12:27:11 -08003520 if (sta && status_code == WLAN_STATUS_SUCCESS) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003521 /* IEEE 802.11r: Mobility Domain Information, Fast BSS
3522 * Transition Information, RSN, [RIC Response] */
3523 p = wpa_sm_write_assoc_resp_ies(sta->wpa_sm, p,
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003524 buf + buflen - p,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003525 sta->auth_alg, ies, ies_len);
Roshan Pius3a1667e2018-07-03 15:17:14 -07003526 if (!p) {
3527 wpa_printf(MSG_DEBUG,
3528 "FT: Failed to write AssocResp IEs");
3529 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
3530 goto done;
3531 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003532 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003533#endif /* CONFIG_IEEE80211R_AP */
Hai Shalom81f62d82019-07-22 12:10:00 -07003534#ifdef CONFIG_FILS
3535 if (sta && status_code == WLAN_STATUS_SUCCESS &&
3536 (sta->auth_alg == WLAN_AUTH_FILS_SK ||
3537 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
3538 sta->auth_alg == WLAN_AUTH_FILS_PK))
3539 p = wpa_auth_write_assoc_resp_fils(sta->wpa_sm, p,
3540 buf + buflen - p,
3541 ies, ies_len);
3542#endif /* CONFIG_FILS */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003543
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003544#ifdef CONFIG_OWE
Hai Shalom74f70d42019-02-11 14:42:39 -08003545 if (sta && status_code == WLAN_STATUS_SUCCESS &&
3546 (hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_OWE))
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003547 p = wpa_auth_write_assoc_resp_owe(sta->wpa_sm, p,
3548 buf + buflen - p,
3549 ies, ies_len);
3550#endif /* CONFIG_OWE */
3551
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003552#ifdef CONFIG_IEEE80211W
Dmitry Shmidt29333592017-01-09 12:27:11 -08003553 if (sta && status_code == WLAN_STATUS_ASSOC_REJECTED_TEMPORARILY)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003554 p = hostapd_eid_assoc_comeback_time(hapd, sta, p);
3555#endif /* CONFIG_IEEE80211W */
3556
3557#ifdef CONFIG_IEEE80211N
3558 p = hostapd_eid_ht_capabilities(hapd, p);
3559 p = hostapd_eid_ht_operation(hapd, p);
3560#endif /* CONFIG_IEEE80211N */
3561
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003562#ifdef CONFIG_IEEE80211AC
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08003563 if (hapd->iconf->ieee80211ac && !hapd->conf->disable_11ac) {
Dmitry Shmidt7d175302016-09-06 13:11:34 -07003564 u32 nsts = 0, sta_nsts;
3565
Dmitry Shmidt29333592017-01-09 12:27:11 -08003566 if (sta && hapd->conf->use_sta_nsts && sta->vht_capabilities) {
Dmitry Shmidt7d175302016-09-06 13:11:34 -07003567 struct ieee80211_vht_capabilities *capa;
3568
3569 nsts = (hapd->iface->conf->vht_capab >>
3570 VHT_CAP_BEAMFORMEE_STS_OFFSET) & 7;
3571 capa = sta->vht_capabilities;
3572 sta_nsts = (le_to_host32(capa->vht_capabilities_info) >>
3573 VHT_CAP_BEAMFORMEE_STS_OFFSET) & 7;
3574
3575 if (nsts < sta_nsts)
3576 nsts = 0;
3577 else
3578 nsts = sta_nsts;
3579 }
3580 p = hostapd_eid_vht_capabilities(hapd, p, nsts);
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08003581 p = hostapd_eid_vht_operation(hapd, p);
3582 }
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07003583#endif /* CONFIG_IEEE80211AC */
3584
Hai Shalom81f62d82019-07-22 12:10:00 -07003585#ifdef CONFIG_IEEE80211AX
3586 if (hapd->iconf->ieee80211ax) {
3587 p = hostapd_eid_he_capab(hapd, p, IEEE80211_MODE_AP);
3588 p = hostapd_eid_he_operation(hapd, p);
3589 p = hostapd_eid_spatial_reuse(hapd, p);
3590 p = hostapd_eid_he_mu_edca_parameter_set(hapd, p);
3591 }
3592#endif /* CONFIG_IEEE80211AX */
3593
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003594 p = hostapd_eid_ext_capab(hapd, p);
Dmitry Shmidt04949592012-07-19 12:16:46 -07003595 p = hostapd_eid_bss_max_idle_period(hapd, p);
Dmitry Shmidt29333592017-01-09 12:27:11 -08003596 if (sta && sta->qos_map_enabled)
Dmitry Shmidt051af732013-10-22 13:52:46 -07003597 p = hostapd_eid_qos_map_set(hapd, p);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003598
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08003599#ifdef CONFIG_FST
3600 if (hapd->iface->fst_ies) {
3601 os_memcpy(p, wpabuf_head(hapd->iface->fst_ies),
3602 wpabuf_len(hapd->iface->fst_ies));
3603 p += wpabuf_len(hapd->iface->fst_ies);
3604 }
3605#endif /* CONFIG_FST */
3606
Hai Shalom021b0b52019-04-10 11:17:58 -07003607#ifdef CONFIG_OWE
3608 if ((hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_OWE) &&
3609 sta && sta->owe_ecdh && status_code == WLAN_STATUS_SUCCESS &&
3610 wpa_auth_sta_key_mgmt(sta->wpa_sm) == WPA_KEY_MGMT_OWE) {
3611 struct wpabuf *pub;
3612
3613 pub = crypto_ecdh_get_pubkey(sta->owe_ecdh, 0);
3614 if (!pub) {
3615 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
3616 goto done;
3617 }
3618 /* OWE Diffie-Hellman Parameter element */
3619 *p++ = WLAN_EID_EXTENSION; /* Element ID */
3620 *p++ = 1 + 2 + wpabuf_len(pub); /* Length */
3621 *p++ = WLAN_EID_EXT_OWE_DH_PARAM; /* Element ID Extension */
3622 WPA_PUT_LE16(p, sta->owe_group);
3623 p += 2;
3624 os_memcpy(p, wpabuf_head(pub), wpabuf_len(pub));
3625 p += wpabuf_len(pub);
3626 wpabuf_free(pub);
3627 }
3628#endif /* CONFIG_OWE */
3629
3630#ifdef CONFIG_DPP2
3631 if ((hapd->conf->wpa_key_mgmt & WPA_KEY_MGMT_DPP) &&
3632 sta && sta->dpp_pfs && status_code == WLAN_STATUS_SUCCESS &&
3633 wpa_auth_sta_key_mgmt(sta->wpa_sm) == WPA_KEY_MGMT_DPP) {
3634 os_memcpy(p, wpabuf_head(sta->dpp_pfs->ie),
3635 wpabuf_len(sta->dpp_pfs->ie));
3636 p += wpabuf_len(sta->dpp_pfs->ie);
3637 }
3638#endif /* CONFIG_DPP2 */
3639
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08003640#ifdef CONFIG_IEEE80211AC
Dmitry Shmidt29333592017-01-09 12:27:11 -08003641 if (sta && hapd->conf->vendor_vht && (sta->flags & WLAN_STA_VENDOR_VHT))
Dmitry Shmidt2f74e362015-01-21 13:19:05 -08003642 p = hostapd_eid_vendor_vht(hapd, p);
3643#endif /* CONFIG_IEEE80211AC */
3644
Dmitry Shmidt29333592017-01-09 12:27:11 -08003645 if (sta && (sta->flags & WLAN_STA_WMM))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003646 p = hostapd_eid_wmm(hapd, p);
3647
3648#ifdef CONFIG_WPS
Dmitry Shmidt29333592017-01-09 12:27:11 -08003649 if (sta &&
3650 ((sta->flags & WLAN_STA_WPS) ||
3651 ((sta->flags & WLAN_STA_MAYBE_WPS) && hapd->conf->wpa))) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003652 struct wpabuf *wps = wps_build_assoc_resp_ie();
3653 if (wps) {
3654 os_memcpy(p, wpabuf_head(wps), wpabuf_len(wps));
3655 p += wpabuf_len(wps);
3656 wpabuf_free(wps);
3657 }
3658 }
3659#endif /* CONFIG_WPS */
3660
Hai Shalom74f70d42019-02-11 14:42:39 -08003661 if (sta && (sta->flags & WLAN_STA_MULTI_AP))
3662 p = hostapd_eid_multi_ap(hapd, p);
3663
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003664#ifdef CONFIG_P2P
Dmitry Shmidt29333592017-01-09 12:27:11 -08003665 if (sta && sta->p2p_ie && hapd->p2p_group) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003666 struct wpabuf *p2p_resp_ie;
3667 enum p2p_status_code status;
3668 switch (status_code) {
3669 case WLAN_STATUS_SUCCESS:
3670 status = P2P_SC_SUCCESS;
3671 break;
3672 case WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA:
3673 status = P2P_SC_FAIL_LIMIT_REACHED;
3674 break;
3675 default:
3676 status = P2P_SC_FAIL_INVALID_PARAMS;
3677 break;
3678 }
3679 p2p_resp_ie = p2p_group_assoc_resp_ie(hapd->p2p_group, status);
3680 if (p2p_resp_ie) {
3681 os_memcpy(p, wpabuf_head(p2p_resp_ie),
3682 wpabuf_len(p2p_resp_ie));
3683 p += wpabuf_len(p2p_resp_ie);
3684 wpabuf_free(p2p_resp_ie);
3685 }
3686 }
3687#endif /* CONFIG_P2P */
3688
3689#ifdef CONFIG_P2P_MANAGER
3690 if (hapd->conf->p2p & P2P_MANAGE)
3691 p = hostapd_eid_p2p_manage(hapd, p);
3692#endif /* CONFIG_P2P_MANAGER */
3693
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003694 p = hostapd_eid_mbo(hapd, p, buf + buflen - p);
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003695
Dmitry Shmidt849734c2016-05-27 09:59:01 -07003696 if (hapd->conf->assocresp_elements &&
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003697 (size_t) (buf + buflen - p) >=
Dmitry Shmidt849734c2016-05-27 09:59:01 -07003698 wpabuf_len(hapd->conf->assocresp_elements)) {
3699 os_memcpy(p, wpabuf_head(hapd->conf->assocresp_elements),
3700 wpabuf_len(hapd->conf->assocresp_elements));
3701 p += wpabuf_len(hapd->conf->assocresp_elements);
3702 }
3703
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003704 send_len += p - reply->u.assoc_resp.variable;
3705
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003706#ifdef CONFIG_FILS
Dmitry Shmidt29333592017-01-09 12:27:11 -08003707 if (sta &&
3708 (sta->auth_alg == WLAN_AUTH_FILS_SK ||
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003709 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
3710 sta->auth_alg == WLAN_AUTH_FILS_PK) &&
3711 status_code == WLAN_STATUS_SUCCESS) {
3712 struct ieee802_11_elems elems;
3713
3714 if (ieee802_11_parse_elems(ies, ies_len, &elems, 0) ==
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003715 ParseFailed || !elems.fils_session) {
3716 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
3717 goto done;
3718 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003719
3720 /* FILS Session */
3721 *p++ = WLAN_EID_EXTENSION; /* Element ID */
3722 *p++ = 1 + FILS_SESSION_LEN; /* Length */
3723 *p++ = WLAN_EID_EXT_FILS_SESSION; /* Element ID Extension */
3724 os_memcpy(p, elems.fils_session, FILS_SESSION_LEN);
3725 send_len += 2 + 1 + FILS_SESSION_LEN;
3726
3727 send_len = fils_encrypt_assoc(sta->wpa_sm, buf, send_len,
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003728 buflen, sta->fils_hlp_resp);
3729 if (send_len < 0) {
3730 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
3731 goto done;
3732 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003733 }
3734#endif /* CONFIG_FILS */
3735
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003736 if (hostapd_drv_send_mlme(hapd, reply, send_len, 0) < 0) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003737 wpa_printf(MSG_INFO, "Failed to send assoc resp: %s",
3738 strerror(errno));
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003739 res = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003740 }
3741
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003742done:
3743 os_free(buf);
3744 return res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003745}
3746
3747
Roshan Pius3a1667e2018-07-03 15:17:14 -07003748#ifdef CONFIG_OWE
3749u8 * owe_assoc_req_process(struct hostapd_data *hapd, struct sta_info *sta,
3750 const u8 *owe_dh, u8 owe_dh_len,
3751 u8 *owe_buf, size_t owe_buf_len, u16 *reason)
3752{
3753#ifdef CONFIG_TESTING_OPTIONS
3754 if (hapd->conf->own_ie_override) {
3755 wpa_printf(MSG_DEBUG, "OWE: Using IE override");
3756 *reason = WLAN_STATUS_SUCCESS;
3757 return wpa_auth_write_assoc_resp_owe(sta->wpa_sm, owe_buf,
3758 owe_buf_len, NULL, 0);
3759 }
3760#endif /* CONFIG_TESTING_OPTIONS */
3761
3762 if (wpa_auth_sta_get_pmksa(sta->wpa_sm)) {
3763 wpa_printf(MSG_DEBUG, "OWE: Using PMKSA caching");
3764 owe_buf = wpa_auth_write_assoc_resp_owe(sta->wpa_sm, owe_buf,
3765 owe_buf_len, NULL, 0);
3766 *reason = WLAN_STATUS_SUCCESS;
3767 return owe_buf;
3768 }
3769
Hai Shalom81f62d82019-07-22 12:10:00 -07003770 if (sta->owe_pmk && sta->external_dh_updated) {
3771 wpa_printf(MSG_DEBUG, "OWE: Using previously derived PMK");
3772 *reason = WLAN_STATUS_SUCCESS;
3773 return owe_buf;
3774 }
3775
Roshan Pius3a1667e2018-07-03 15:17:14 -07003776 *reason = owe_process_assoc_req(hapd, sta, owe_dh, owe_dh_len);
3777 if (*reason != WLAN_STATUS_SUCCESS)
3778 return NULL;
3779
3780 owe_buf = wpa_auth_write_assoc_resp_owe(sta->wpa_sm, owe_buf,
3781 owe_buf_len, NULL, 0);
3782
3783 if (sta->owe_ecdh && owe_buf) {
3784 struct wpabuf *pub;
3785
3786 pub = crypto_ecdh_get_pubkey(sta->owe_ecdh, 0);
3787 if (!pub) {
3788 *reason = WLAN_STATUS_UNSPECIFIED_FAILURE;
3789 return owe_buf;
3790 }
3791
3792 /* OWE Diffie-Hellman Parameter element */
3793 *owe_buf++ = WLAN_EID_EXTENSION; /* Element ID */
3794 *owe_buf++ = 1 + 2 + wpabuf_len(pub); /* Length */
3795 *owe_buf++ = WLAN_EID_EXT_OWE_DH_PARAM; /* Element ID Extension
3796 */
3797 WPA_PUT_LE16(owe_buf, sta->owe_group);
3798 owe_buf += 2;
3799 os_memcpy(owe_buf, wpabuf_head(pub), wpabuf_len(pub));
3800 owe_buf += wpabuf_len(pub);
3801 wpabuf_free(pub);
3802 }
3803
3804 return owe_buf;
3805}
3806#endif /* CONFIG_OWE */
3807
3808
Paul Stewart092955c2017-02-06 09:13:09 -08003809#ifdef CONFIG_FILS
3810
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003811void fils_hlp_finish_assoc(struct hostapd_data *hapd, struct sta_info *sta)
Paul Stewart092955c2017-02-06 09:13:09 -08003812{
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003813 u16 reply_res;
Paul Stewart092955c2017-02-06 09:13:09 -08003814
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003815 wpa_printf(MSG_DEBUG, "FILS: Finish association with " MACSTR,
3816 MAC2STR(sta->addr));
3817 eloop_cancel_timeout(fils_hlp_timeout, hapd, sta);
3818 if (!sta->fils_pending_assoc_req)
Paul Stewart092955c2017-02-06 09:13:09 -08003819 return;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003820 reply_res = send_assoc_resp(hapd, sta, sta->addr, WLAN_STATUS_SUCCESS,
3821 sta->fils_pending_assoc_is_reassoc,
3822 sta->fils_pending_assoc_req,
Hai Shalom74f70d42019-02-11 14:42:39 -08003823 sta->fils_pending_assoc_req_len, 0);
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003824 os_free(sta->fils_pending_assoc_req);
3825 sta->fils_pending_assoc_req = NULL;
3826 sta->fils_pending_assoc_req_len = 0;
3827 wpabuf_free(sta->fils_hlp_resp);
3828 sta->fils_hlp_resp = NULL;
3829 wpabuf_free(sta->hlp_dhcp_discover);
3830 sta->hlp_dhcp_discover = NULL;
Paul Stewart092955c2017-02-06 09:13:09 -08003831
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003832 /*
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003833 * Remove the station in case transmission of a success response fails.
3834 * At this point the station was already added associated to the driver.
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003835 */
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003836 if (reply_res != WLAN_STATUS_SUCCESS)
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003837 hostapd_drv_sta_remove(hapd, sta->addr);
Paul Stewart092955c2017-02-06 09:13:09 -08003838}
3839
3840
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003841void fils_hlp_timeout(void *eloop_ctx, void *eloop_data)
Paul Stewart092955c2017-02-06 09:13:09 -08003842{
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003843 struct hostapd_data *hapd = eloop_ctx;
3844 struct sta_info *sta = eloop_data;
Paul Stewart092955c2017-02-06 09:13:09 -08003845
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003846 wpa_printf(MSG_DEBUG,
3847 "FILS: HLP response timeout - continue with association response for "
3848 MACSTR, MAC2STR(sta->addr));
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07003849 if (sta->fils_drv_assoc_finish)
3850 hostapd_notify_assoc_fils_finish(hapd, sta);
3851 else
3852 fils_hlp_finish_assoc(hapd, sta);
Paul Stewart092955c2017-02-06 09:13:09 -08003853}
3854
3855#endif /* CONFIG_FILS */
3856
3857
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003858static void handle_assoc(struct hostapd_data *hapd,
3859 const struct ieee80211_mgmt *mgmt, size_t len,
Hai Shalom74f70d42019-02-11 14:42:39 -08003860 int reassoc, int rssi)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003861{
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003862 u16 capab_info, listen_interval, seq_ctrl, fc;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003863 u16 resp = WLAN_STATUS_SUCCESS, reply_res;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003864 const u8 *pos;
3865 int left, i;
3866 struct sta_info *sta;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003867 u8 *tmp = NULL;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003868 struct hostapd_sta_wpa_psk_short *psk = NULL;
3869 char *identity = NULL;
3870 char *radius_cui = NULL;
3871#ifdef CONFIG_FILS
3872 int delay_assoc = 0;
3873#endif /* CONFIG_FILS */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003874
3875 if (len < IEEE80211_HDRLEN + (reassoc ? sizeof(mgmt->u.reassoc_req) :
3876 sizeof(mgmt->u.assoc_req))) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08003877 wpa_printf(MSG_INFO, "handle_assoc(reassoc=%d) - too short payload (len=%lu)",
3878 reassoc, (unsigned long) len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003879 return;
3880 }
3881
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07003882#ifdef CONFIG_TESTING_OPTIONS
3883 if (reassoc) {
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07003884 if (hapd->iconf->ignore_reassoc_probability > 0.0 &&
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07003885 drand48() < hapd->iconf->ignore_reassoc_probability) {
3886 wpa_printf(MSG_INFO,
3887 "TESTING: ignoring reassoc request from "
3888 MACSTR, MAC2STR(mgmt->sa));
3889 return;
3890 }
3891 } else {
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07003892 if (hapd->iconf->ignore_assoc_probability > 0.0 &&
Dmitry Shmidt8da800a2013-04-24 12:57:01 -07003893 drand48() < hapd->iconf->ignore_assoc_probability) {
3894 wpa_printf(MSG_INFO,
3895 "TESTING: ignoring assoc request from "
3896 MACSTR, MAC2STR(mgmt->sa));
3897 return;
3898 }
3899 }
3900#endif /* CONFIG_TESTING_OPTIONS */
3901
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003902 fc = le_to_host16(mgmt->frame_control);
3903 seq_ctrl = le_to_host16(mgmt->seq_ctrl);
3904
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003905 if (reassoc) {
3906 capab_info = le_to_host16(mgmt->u.reassoc_req.capab_info);
3907 listen_interval = le_to_host16(
3908 mgmt->u.reassoc_req.listen_interval);
3909 wpa_printf(MSG_DEBUG, "reassociation request: STA=" MACSTR
3910 " capab_info=0x%02x listen_interval=%d current_ap="
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003911 MACSTR " seq_ctrl=0x%x%s",
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003912 MAC2STR(mgmt->sa), capab_info, listen_interval,
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003913 MAC2STR(mgmt->u.reassoc_req.current_ap),
3914 seq_ctrl, (fc & WLAN_FC_RETRY) ? " retry" : "");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003915 left = len - (IEEE80211_HDRLEN + sizeof(mgmt->u.reassoc_req));
3916 pos = mgmt->u.reassoc_req.variable;
3917 } else {
3918 capab_info = le_to_host16(mgmt->u.assoc_req.capab_info);
3919 listen_interval = le_to_host16(
3920 mgmt->u.assoc_req.listen_interval);
3921 wpa_printf(MSG_DEBUG, "association request: STA=" MACSTR
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08003922 " capab_info=0x%02x listen_interval=%d "
3923 "seq_ctrl=0x%x%s",
3924 MAC2STR(mgmt->sa), capab_info, listen_interval,
3925 seq_ctrl, (fc & WLAN_FC_RETRY) ? " retry" : "");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003926 left = len - (IEEE80211_HDRLEN + sizeof(mgmt->u.assoc_req));
3927 pos = mgmt->u.assoc_req.variable;
3928 }
3929
3930 sta = ap_get_sta(hapd, mgmt->sa);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003931#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003932 if (sta && sta->auth_alg == WLAN_AUTH_FT &&
3933 (sta->flags & WLAN_STA_AUTH) == 0) {
3934 wpa_printf(MSG_DEBUG, "FT: Allow STA " MACSTR " to associate "
3935 "prior to authentication since it is using "
3936 "over-the-DS FT", MAC2STR(mgmt->sa));
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08003937
3938 /*
3939 * Mark station as authenticated, to avoid adding station
3940 * entry in the driver as associated and not authenticated
3941 */
3942 sta->flags |= WLAN_STA_AUTH;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003943 } else
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08003944#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07003945 if (sta == NULL || (sta->flags & WLAN_STA_AUTH) == 0) {
Dmitry Shmidt29333592017-01-09 12:27:11 -08003946 if (hapd->iface->current_mode &&
3947 hapd->iface->current_mode->mode ==
3948 HOSTAPD_MODE_IEEE80211AD) {
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003949 int acl_res;
3950 u32 session_timeout, acct_interim_interval;
3951 struct vlan_description vlan_id;
3952
3953 acl_res = ieee802_11_allowed_address(
3954 hapd, mgmt->sa, (const u8 *) mgmt, len,
3955 &session_timeout, &acct_interim_interval,
Roshan Pius3a1667e2018-07-03 15:17:14 -07003956 &vlan_id, &psk, &identity, &radius_cui, 0);
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003957 if (acl_res == HOSTAPD_ACL_REJECT) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07003958 wpa_msg(hapd->msg_ctx, MSG_DEBUG,
3959 "Ignore Association Request frame from "
3960 MACSTR " due to ACL reject",
3961 MAC2STR(mgmt->sa));
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003962 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
3963 goto fail;
3964 }
3965 if (acl_res == HOSTAPD_ACL_PENDING)
3966 return;
3967
Dmitry Shmidt29333592017-01-09 12:27:11 -08003968 /* DMG/IEEE 802.11ad does not use authentication.
3969 * Allocate sta entry upon association. */
3970 sta = ap_sta_add(hapd, mgmt->sa);
3971 if (!sta) {
3972 hostapd_logger(hapd, mgmt->sa,
3973 HOSTAPD_MODULE_IEEE80211,
3974 HOSTAPD_LEVEL_INFO,
3975 "Failed to add STA");
3976 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
3977 goto fail;
3978 }
3979
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08003980 acl_res = ieee802_11_set_radius_info(
3981 hapd, sta, acl_res, session_timeout,
3982 acct_interim_interval, &vlan_id, &psk,
3983 &identity, &radius_cui);
3984 if (acl_res) {
3985 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
3986 goto fail;
3987 }
3988
Dmitry Shmidt29333592017-01-09 12:27:11 -08003989 hostapd_logger(hapd, sta->addr,
3990 HOSTAPD_MODULE_IEEE80211,
3991 HOSTAPD_LEVEL_DEBUG,
3992 "Skip authentication for DMG/IEEE 802.11ad");
3993 sta->flags |= WLAN_STA_AUTH;
3994 wpa_auth_sm_event(sta->wpa_sm, WPA_AUTH);
3995 sta->auth_alg = WLAN_AUTH_OPEN;
3996 } else {
3997 hostapd_logger(hapd, mgmt->sa,
3998 HOSTAPD_MODULE_IEEE80211,
3999 HOSTAPD_LEVEL_INFO,
4000 "Station tried to associate before authentication (aid=%d flags=0x%x)",
4001 sta ? sta->aid : -1,
4002 sta ? sta->flags : 0);
4003 send_deauth(hapd, mgmt->sa,
4004 WLAN_REASON_CLASS2_FRAME_FROM_NONAUTH_STA);
4005 return;
4006 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004007 }
4008
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004009 if ((fc & WLAN_FC_RETRY) &&
4010 sta->last_seq_ctrl != WLAN_INVALID_MGMT_SEQ &&
4011 sta->last_seq_ctrl == seq_ctrl &&
Paul Stewart092955c2017-02-06 09:13:09 -08004012 sta->last_subtype == (reassoc ? WLAN_FC_STYPE_REASSOC_REQ :
4013 WLAN_FC_STYPE_ASSOC_REQ)) {
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004014 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4015 HOSTAPD_LEVEL_DEBUG,
4016 "Drop repeated association frame seq_ctrl=0x%x",
4017 seq_ctrl);
4018 return;
4019 }
4020 sta->last_seq_ctrl = seq_ctrl;
4021 sta->last_subtype = reassoc ? WLAN_FC_STYPE_REASSOC_REQ :
4022 WLAN_FC_STYPE_ASSOC_REQ;
4023
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004024 if (hapd->tkip_countermeasures) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07004025 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004026 goto fail;
4027 }
4028
4029 if (listen_interval > hapd->conf->max_listen_interval) {
4030 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4031 HOSTAPD_LEVEL_DEBUG,
4032 "Too large Listen Interval (%d)",
4033 listen_interval);
4034 resp = WLAN_STATUS_ASSOC_DENIED_LISTEN_INT_TOO_LARGE;
4035 goto fail;
4036 }
4037
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004038#ifdef CONFIG_MBO
4039 if (hapd->conf->mbo_enabled && hapd->mbo_assoc_disallow) {
4040 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
4041 goto fail;
4042 }
Hai Shalom74f70d42019-02-11 14:42:39 -08004043
4044 if (hapd->iconf->rssi_reject_assoc_rssi && rssi &&
4045 rssi < hapd->iconf->rssi_reject_assoc_rssi &&
4046 (sta->auth_rssi == 0 ||
4047 sta->auth_rssi < hapd->iconf->rssi_reject_assoc_rssi)) {
4048 resp = WLAN_STATUS_DENIED_POOR_CHANNEL_CONDITIONS;
4049 goto fail;
4050 }
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004051#endif /* CONFIG_MBO */
4052
Dmitry Shmidt849734c2016-05-27 09:59:01 -07004053 /*
4054 * sta->capability is used in check_assoc_ies() for RRM enabled
4055 * capability element.
4056 */
4057 sta->capability = capab_info;
4058
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004059#ifdef CONFIG_FILS
4060 if (sta->auth_alg == WLAN_AUTH_FILS_SK ||
4061 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
4062 sta->auth_alg == WLAN_AUTH_FILS_PK) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07004063 int res;
4064
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004065 /* The end of the payload is encrypted. Need to decrypt it
4066 * before parsing. */
4067
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004068 tmp = os_memdup(pos, left);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004069 if (!tmp) {
4070 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
4071 goto fail;
4072 }
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004073
Roshan Pius3a1667e2018-07-03 15:17:14 -07004074 res = fils_decrypt_assoc(sta->wpa_sm, sta->fils_session, mgmt,
4075 len, tmp, left);
4076 if (res < 0) {
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004077 resp = WLAN_STATUS_UNSPECIFIED_FAILURE;
4078 goto fail;
4079 }
4080 pos = tmp;
Roshan Pius3a1667e2018-07-03 15:17:14 -07004081 left = res;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004082 }
4083#endif /* CONFIG_FILS */
4084
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004085 /* followed by SSID and Supported rates; and HT capabilities if 802.11n
4086 * is used */
4087 resp = check_assoc_ies(hapd, sta, pos, left, reassoc);
4088 if (resp != WLAN_STATUS_SUCCESS)
4089 goto fail;
4090
4091 if (hostapd_get_aid(hapd, sta) < 0) {
4092 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4093 HOSTAPD_LEVEL_INFO, "No room for more AIDs");
4094 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
4095 goto fail;
4096 }
4097
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004098 sta->listen_interval = listen_interval;
4099
Roshan Pius3a1667e2018-07-03 15:17:14 -07004100 if (hapd->iface->current_mode &&
4101 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211G)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004102 sta->flags |= WLAN_STA_NONERP;
4103 for (i = 0; i < sta->supported_rates_len; i++) {
4104 if ((sta->supported_rates[i] & 0x7f) > 22) {
4105 sta->flags &= ~WLAN_STA_NONERP;
4106 break;
4107 }
4108 }
4109 if (sta->flags & WLAN_STA_NONERP && !sta->nonerp_set) {
4110 sta->nonerp_set = 1;
4111 hapd->iface->num_sta_non_erp++;
4112 if (hapd->iface->num_sta_non_erp == 1)
4113 ieee802_11_set_beacons(hapd->iface);
4114 }
4115
4116 if (!(sta->capability & WLAN_CAPABILITY_SHORT_SLOT_TIME) &&
4117 !sta->no_short_slot_time_set) {
4118 sta->no_short_slot_time_set = 1;
4119 hapd->iface->num_sta_no_short_slot_time++;
Roshan Pius3a1667e2018-07-03 15:17:14 -07004120 if (hapd->iface->current_mode &&
4121 hapd->iface->current_mode->mode ==
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004122 HOSTAPD_MODE_IEEE80211G &&
4123 hapd->iface->num_sta_no_short_slot_time == 1)
4124 ieee802_11_set_beacons(hapd->iface);
4125 }
4126
4127 if (sta->capability & WLAN_CAPABILITY_SHORT_PREAMBLE)
4128 sta->flags |= WLAN_STA_SHORT_PREAMBLE;
4129 else
4130 sta->flags &= ~WLAN_STA_SHORT_PREAMBLE;
4131
4132 if (!(sta->capability & WLAN_CAPABILITY_SHORT_PREAMBLE) &&
4133 !sta->no_short_preamble_set) {
4134 sta->no_short_preamble_set = 1;
4135 hapd->iface->num_sta_no_short_preamble++;
Roshan Pius3a1667e2018-07-03 15:17:14 -07004136 if (hapd->iface->current_mode &&
4137 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211G
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004138 && hapd->iface->num_sta_no_short_preamble == 1)
4139 ieee802_11_set_beacons(hapd->iface);
4140 }
4141
4142#ifdef CONFIG_IEEE80211N
4143 update_ht_state(hapd, sta);
4144#endif /* CONFIG_IEEE80211N */
4145
4146 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4147 HOSTAPD_LEVEL_DEBUG,
4148 "association OK (aid %d)", sta->aid);
4149 /* Station will be marked associated, after it acknowledges AssocResp
4150 */
4151 sta->flags |= WLAN_STA_ASSOC_REQ_OK;
4152
4153#ifdef CONFIG_IEEE80211W
4154 if ((sta->flags & WLAN_STA_MFP) && sta->sa_query_timed_out) {
4155 wpa_printf(MSG_DEBUG, "Allowing %sassociation after timed out "
4156 "SA Query procedure", reassoc ? "re" : "");
4157 /* TODO: Send a protected Disassociate frame to the STA using
4158 * the old key and Reason Code "Previous Authentication no
4159 * longer valid". Make sure this is only sent protected since
4160 * unprotected frame would be received by the STA that is now
4161 * trying to associate.
4162 */
4163 }
4164#endif /* CONFIG_IEEE80211W */
4165
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004166 /* Make sure that the previously registered inactivity timer will not
4167 * remove the STA immediately. */
4168 sta->timeout_next = STA_NULLFUNC;
4169
Dmitry Shmidtaca489e2016-09-28 15:44:14 -07004170#ifdef CONFIG_TAXONOMY
4171 taxonomy_sta_info_assoc_req(hapd, sta, pos, left);
4172#endif /* CONFIG_TAXONOMY */
4173
Dmitry Shmidt29333592017-01-09 12:27:11 -08004174 sta->pending_wds_enable = 0;
4175
Paul Stewart092955c2017-02-06 09:13:09 -08004176#ifdef CONFIG_FILS
4177 if (sta->auth_alg == WLAN_AUTH_FILS_SK ||
4178 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004179 sta->auth_alg == WLAN_AUTH_FILS_PK) {
4180 if (fils_process_hlp(hapd, sta, pos, left) > 0)
4181 delay_assoc = 1;
4182 }
Paul Stewart092955c2017-02-06 09:13:09 -08004183#endif /* CONFIG_FILS */
4184
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004185 fail:
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004186 os_free(identity);
4187 os_free(radius_cui);
4188 hostapd_free_psk_list(psk);
4189
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004190 /*
4191 * In case of a successful response, add the station to the driver.
4192 * Otherwise, the kernel may ignore Data frames before we process the
4193 * ACK frame (TX status). In case of a failure, this station will be
4194 * removed.
4195 *
4196 * Note that this is not compliant with the IEEE 802.11 standard that
4197 * states that a non-AP station should transition into the
4198 * authenticated/associated state only after the station acknowledges
4199 * the (Re)Association Response frame. However, still do this as:
4200 *
4201 * 1. In case the station does not acknowledge the (Re)Association
4202 * Response frame, it will be removed.
4203 * 2. Data frames will be dropped in the kernel until the station is
4204 * set into authorized state, and there are no significant known
4205 * issues with processing other non-Data Class 3 frames during this
4206 * window.
4207 */
Hai Shalom74f70d42019-02-11 14:42:39 -08004208 if (resp == WLAN_STATUS_SUCCESS && sta &&
4209 add_associated_sta(hapd, sta, reassoc))
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004210 resp = WLAN_STATUS_AP_UNABLE_TO_HANDLE_NEW_STA;
4211
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004212#ifdef CONFIG_FILS
Hai Shalom74f70d42019-02-11 14:42:39 -08004213 if (sta && delay_assoc && resp == WLAN_STATUS_SUCCESS &&
4214 eloop_is_timeout_registered(fils_hlp_timeout, hapd, sta) &&
4215 sta->fils_pending_assoc_req) {
4216 /* Do not reschedule fils_hlp_timeout in case the station
4217 * retransmits (Re)Association Request frame while waiting for
4218 * the previously started FILS HLP wait, so that the timeout can
4219 * be determined from the first pending attempt. */
4220 wpa_printf(MSG_DEBUG,
4221 "FILS: Continue waiting for HLP processing before sending (Re)Association Response frame to "
4222 MACSTR, MAC2STR(sta->addr));
4223 os_free(tmp);
4224 return;
4225 }
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004226 if (sta) {
4227 eloop_cancel_timeout(fils_hlp_timeout, hapd, sta);
4228 os_free(sta->fils_pending_assoc_req);
4229 sta->fils_pending_assoc_req = NULL;
4230 sta->fils_pending_assoc_req_len = 0;
4231 wpabuf_free(sta->fils_hlp_resp);
4232 sta->fils_hlp_resp = NULL;
4233 }
4234 if (sta && delay_assoc && resp == WLAN_STATUS_SUCCESS) {
4235 sta->fils_pending_assoc_req = tmp;
4236 sta->fils_pending_assoc_req_len = left;
4237 sta->fils_pending_assoc_is_reassoc = reassoc;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004238 sta->fils_drv_assoc_finish = 0;
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004239 wpa_printf(MSG_DEBUG,
4240 "FILS: Waiting for HLP processing before sending (Re)Association Response frame to "
4241 MACSTR, MAC2STR(sta->addr));
4242 eloop_cancel_timeout(fils_hlp_timeout, hapd, sta);
4243 eloop_register_timeout(0, hapd->conf->fils_hlp_wait_time * 1024,
4244 fils_hlp_timeout, hapd, sta);
4245 return;
4246 }
4247#endif /* CONFIG_FILS */
4248
Dmitry Shmidt29333592017-01-09 12:27:11 -08004249 reply_res = send_assoc_resp(hapd, sta, mgmt->sa, resp, reassoc, pos,
Hai Shalom74f70d42019-02-11 14:42:39 -08004250 left, rssi);
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004251 os_free(tmp);
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004252
4253 /*
4254 * Remove the station in case tranmission of a success response fails
4255 * (the STA was added associated to the driver) or if the station was
4256 * previously added unassociated.
4257 */
Dmitry Shmidt29333592017-01-09 12:27:11 -08004258 if (sta && ((reply_res != WLAN_STATUS_SUCCESS &&
4259 resp == WLAN_STATUS_SUCCESS) || sta->added_unassoc)) {
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004260 hostapd_drv_sta_remove(hapd, sta->addr);
4261 sta->added_unassoc = 0;
4262 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004263}
4264
4265
4266static void handle_disassoc(struct hostapd_data *hapd,
4267 const struct ieee80211_mgmt *mgmt, size_t len)
4268{
4269 struct sta_info *sta;
4270
4271 if (len < IEEE80211_HDRLEN + sizeof(mgmt->u.disassoc)) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08004272 wpa_printf(MSG_INFO, "handle_disassoc - too short payload (len=%lu)",
4273 (unsigned long) len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004274 return;
4275 }
4276
4277 wpa_printf(MSG_DEBUG, "disassocation: STA=" MACSTR " reason_code=%d",
4278 MAC2STR(mgmt->sa),
4279 le_to_host16(mgmt->u.disassoc.reason_code));
4280
4281 sta = ap_get_sta(hapd, mgmt->sa);
4282 if (sta == NULL) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08004283 wpa_printf(MSG_INFO, "Station " MACSTR " trying to disassociate, but it is not associated",
4284 MAC2STR(mgmt->sa));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004285 return;
4286 }
4287
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004288 ap_sta_set_authorized(hapd, sta, 0);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004289 sta->last_seq_ctrl = WLAN_INVALID_MGMT_SEQ;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004290 sta->flags &= ~(WLAN_STA_ASSOC | WLAN_STA_ASSOC_REQ_OK);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004291 wpa_auth_sm_event(sta->wpa_sm, WPA_DISASSOC);
4292 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4293 HOSTAPD_LEVEL_INFO, "disassociated");
4294 sta->acct_terminate_cause = RADIUS_ACCT_TERMINATE_CAUSE_USER_REQUEST;
4295 ieee802_1x_notify_port_enabled(sta->eapol_sm, 0);
4296 /* Stop Accounting and IEEE 802.1X sessions, but leave the STA
4297 * authenticated. */
4298 accounting_sta_stop(hapd, sta);
Dmitry Shmidtde47be72016-01-07 12:52:55 -08004299 ieee802_1x_free_station(hapd, sta);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004300 if (sta->ipaddr)
4301 hostapd_drv_br_delete_ip_neigh(hapd, 4, (u8 *) &sta->ipaddr);
4302 ap_sta_ip6addr_del(hapd, sta);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004303 hostapd_drv_sta_remove(hapd, sta->addr);
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004304 sta->added_unassoc = 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004305
4306 if (sta->timeout_next == STA_NULLFUNC ||
4307 sta->timeout_next == STA_DISASSOC) {
4308 sta->timeout_next = STA_DEAUTH;
4309 eloop_cancel_timeout(ap_handle_timer, hapd, sta);
4310 eloop_register_timeout(AP_DEAUTH_DELAY, 0, ap_handle_timer,
4311 hapd, sta);
4312 }
4313
4314 mlme_disassociate_indication(
4315 hapd, sta, le_to_host16(mgmt->u.disassoc.reason_code));
Dmitry Shmidt29333592017-01-09 12:27:11 -08004316
4317 /* DMG/IEEE 802.11ad does not use deauthication. Deallocate sta upon
4318 * disassociation. */
4319 if (hapd->iface->current_mode &&
4320 hapd->iface->current_mode->mode == HOSTAPD_MODE_IEEE80211AD) {
4321 sta->flags &= ~WLAN_STA_AUTH;
4322 wpa_auth_sm_event(sta->wpa_sm, WPA_DEAUTH);
4323 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4324 HOSTAPD_LEVEL_DEBUG, "deauthenticated");
4325 ap_free_sta(hapd, sta);
4326 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004327}
4328
4329
4330static void handle_deauth(struct hostapd_data *hapd,
4331 const struct ieee80211_mgmt *mgmt, size_t len)
4332{
4333 struct sta_info *sta;
4334
4335 if (len < IEEE80211_HDRLEN + sizeof(mgmt->u.deauth)) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004336 wpa_msg(hapd->msg_ctx, MSG_DEBUG, "handle_deauth - too short "
4337 "payload (len=%lu)", (unsigned long) len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004338 return;
4339 }
4340
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004341 wpa_msg(hapd->msg_ctx, MSG_DEBUG, "deauthentication: STA=" MACSTR
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004342 " reason_code=%d",
4343 MAC2STR(mgmt->sa), le_to_host16(mgmt->u.deauth.reason_code));
4344
4345 sta = ap_get_sta(hapd, mgmt->sa);
4346 if (sta == NULL) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004347 wpa_msg(hapd->msg_ctx, MSG_DEBUG, "Station " MACSTR " trying "
4348 "to deauthenticate, but it is not authenticated",
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004349 MAC2STR(mgmt->sa));
4350 return;
4351 }
4352
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004353 ap_sta_set_authorized(hapd, sta, 0);
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004354 sta->last_seq_ctrl = WLAN_INVALID_MGMT_SEQ;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004355 sta->flags &= ~(WLAN_STA_AUTH | WLAN_STA_ASSOC |
4356 WLAN_STA_ASSOC_REQ_OK);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004357 wpa_auth_sm_event(sta->wpa_sm, WPA_DEAUTH);
4358 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4359 HOSTAPD_LEVEL_DEBUG, "deauthenticated");
4360 mlme_deauthenticate_indication(
4361 hapd, sta, le_to_host16(mgmt->u.deauth.reason_code));
4362 sta->acct_terminate_cause = RADIUS_ACCT_TERMINATE_CAUSE_USER_REQUEST;
4363 ieee802_1x_notify_port_enabled(sta->eapol_sm, 0);
4364 ap_free_sta(hapd, sta);
4365}
4366
4367
4368static void handle_beacon(struct hostapd_data *hapd,
4369 const struct ieee80211_mgmt *mgmt, size_t len,
4370 struct hostapd_frame_info *fi)
4371{
4372 struct ieee802_11_elems elems;
4373
4374 if (len < IEEE80211_HDRLEN + sizeof(mgmt->u.beacon)) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08004375 wpa_printf(MSG_INFO, "handle_beacon - too short payload (len=%lu)",
4376 (unsigned long) len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004377 return;
4378 }
4379
4380 (void) ieee802_11_parse_elems(mgmt->u.beacon.variable,
4381 len - (IEEE80211_HDRLEN +
4382 sizeof(mgmt->u.beacon)), &elems,
4383 0);
4384
4385 ap_list_process_beacon(hapd->iface, mgmt, &elems, fi);
4386}
4387
4388
4389#ifdef CONFIG_IEEE80211W
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004390static int robust_action_frame(u8 category)
4391{
4392 return category != WLAN_ACTION_PUBLIC &&
4393 category != WLAN_ACTION_HT;
4394}
4395#endif /* CONFIG_IEEE80211W */
4396
4397
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004398static int handle_action(struct hostapd_data *hapd,
Roshan Pius3a1667e2018-07-03 15:17:14 -07004399 const struct ieee80211_mgmt *mgmt, size_t len,
4400 unsigned int freq)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004401{
4402 struct sta_info *sta;
Hai Shalom74f70d42019-02-11 14:42:39 -08004403 u8 *action __maybe_unused;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004404
Hai Shalom74f70d42019-02-11 14:42:39 -08004405 if (len < IEEE80211_HDRLEN + 2 + 1) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004406 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4407 HOSTAPD_LEVEL_DEBUG,
4408 "handle_action - too short payload (len=%lu)",
4409 (unsigned long) len);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004410 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004411 }
4412
Hai Shalom74f70d42019-02-11 14:42:39 -08004413 action = (u8 *) &mgmt->u.action.u;
4414 wpa_printf(MSG_DEBUG, "RX_ACTION category %u action %u sa " MACSTR
4415 " da " MACSTR " len %d freq %u",
4416 mgmt->u.action.category, *action,
4417 MAC2STR(mgmt->sa), MAC2STR(mgmt->da), (int) len, freq);
4418
4419 sta = ap_get_sta(hapd, mgmt->sa);
4420
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08004421 if (mgmt->u.action.category != WLAN_ACTION_PUBLIC &&
4422 (sta == NULL || !(sta->flags & WLAN_STA_ASSOC))) {
4423 wpa_printf(MSG_DEBUG, "IEEE 802.11: Ignored Action "
4424 "frame (category=%u) from unassociated STA " MACSTR,
Dmitry Shmidtebd93af2017-02-21 13:40:44 -08004425 mgmt->u.action.category, MAC2STR(mgmt->sa));
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004426 return 0;
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08004427 }
4428
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004429#ifdef CONFIG_IEEE80211W
4430 if (sta && (sta->flags & WLAN_STA_MFP) &&
Dmitry Shmidt18463232014-01-24 12:29:41 -08004431 !(mgmt->frame_control & host_to_le16(WLAN_FC_ISWEP)) &&
4432 robust_action_frame(mgmt->u.action.category)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004433 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4434 HOSTAPD_LEVEL_DEBUG,
4435 "Dropped unprotected Robust Action frame from "
4436 "an MFP STA");
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004437 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004438 }
4439#endif /* CONFIG_IEEE80211W */
4440
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004441 if (sta) {
4442 u16 fc = le_to_host16(mgmt->frame_control);
4443 u16 seq_ctrl = le_to_host16(mgmt->seq_ctrl);
4444
4445 if ((fc & WLAN_FC_RETRY) &&
4446 sta->last_seq_ctrl != WLAN_INVALID_MGMT_SEQ &&
4447 sta->last_seq_ctrl == seq_ctrl &&
4448 sta->last_subtype == WLAN_FC_STYPE_ACTION) {
4449 hostapd_logger(hapd, sta->addr,
4450 HOSTAPD_MODULE_IEEE80211,
4451 HOSTAPD_LEVEL_DEBUG,
4452 "Drop repeated action frame seq_ctrl=0x%x",
4453 seq_ctrl);
4454 return 1;
4455 }
4456
4457 sta->last_seq_ctrl = seq_ctrl;
4458 sta->last_subtype = WLAN_FC_STYPE_ACTION;
4459 }
4460
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004461 switch (mgmt->u.action.category) {
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004462#ifdef CONFIG_IEEE80211R_AP
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004463 case WLAN_ACTION_FT:
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07004464 if (!sta ||
4465 wpa_ft_action_rx(sta->wpa_sm, (u8 *) &mgmt->u.action,
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004466 len - IEEE80211_HDRLEN))
4467 break;
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004468 return 1;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004469#endif /* CONFIG_IEEE80211R_AP */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004470 case WLAN_ACTION_WMM:
4471 hostapd_wmm_action(hapd, mgmt, len);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004472 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004473#ifdef CONFIG_IEEE80211W
4474 case WLAN_ACTION_SA_QUERY:
Hai Shalom021b0b52019-04-10 11:17:58 -07004475 ieee802_11_sa_query_action(hapd, mgmt, len);
4476 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004477#endif /* CONFIG_IEEE80211W */
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004478#ifdef CONFIG_WNM_AP
Dmitry Shmidta54fa5f2013-01-15 13:53:35 -08004479 case WLAN_ACTION_WNM:
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004480 ieee802_11_rx_wnm_action_ap(hapd, mgmt, len);
4481 return 1;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004482#endif /* CONFIG_WNM_AP */
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08004483#ifdef CONFIG_FST
4484 case WLAN_ACTION_FST:
4485 if (hapd->iface->fst)
4486 fst_rx_action(hapd->iface->fst, mgmt, len);
4487 else
4488 wpa_printf(MSG_DEBUG,
4489 "FST: Ignore FST Action frame - no FST attached");
4490 return 1;
4491#endif /* CONFIG_FST */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004492 case WLAN_ACTION_PUBLIC:
Dmitry Shmidt18463232014-01-24 12:29:41 -08004493 case WLAN_ACTION_PROTECTED_DUAL:
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07004494#ifdef CONFIG_IEEE80211N
Dmitry Shmidtcc00d5d2015-05-04 10:34:12 -07004495 if (len >= IEEE80211_HDRLEN + 2 &&
4496 mgmt->u.action.u.public_action.action ==
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07004497 WLAN_PA_20_40_BSS_COEX) {
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07004498 hostapd_2040_coex_action(hapd, mgmt, len);
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004499 return 1;
Dmitry Shmidt7832adb2014-04-29 10:53:02 -07004500 }
4501#endif /* CONFIG_IEEE80211N */
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004502#ifdef CONFIG_DPP
4503 if (len >= IEEE80211_HDRLEN + 6 &&
4504 mgmt->u.action.u.vs_public_action.action ==
4505 WLAN_PA_VENDOR_SPECIFIC &&
4506 WPA_GET_BE24(mgmt->u.action.u.vs_public_action.oui) ==
4507 OUI_WFA &&
4508 mgmt->u.action.u.vs_public_action.variable[0] ==
4509 DPP_OUI_TYPE) {
4510 const u8 *pos, *end;
4511
4512 pos = mgmt->u.action.u.vs_public_action.oui;
4513 end = ((const u8 *) mgmt) + len;
4514 hostapd_dpp_rx_action(hapd, mgmt->sa, pos, end - pos,
Roshan Pius3a1667e2018-07-03 15:17:14 -07004515 freq);
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004516 return 1;
4517 }
4518 if (len >= IEEE80211_HDRLEN + 2 &&
4519 (mgmt->u.action.u.public_action.action ==
4520 WLAN_PA_GAS_INITIAL_RESP ||
4521 mgmt->u.action.u.public_action.action ==
4522 WLAN_PA_GAS_COMEBACK_RESP)) {
4523 const u8 *pos, *end;
4524
4525 pos = &mgmt->u.action.u.public_action.action;
4526 end = ((const u8 *) mgmt) + len;
4527 gas_query_ap_rx(hapd->gas, mgmt->sa,
4528 mgmt->u.action.category,
4529 pos, end - pos, hapd->iface->freq);
4530 return 1;
4531 }
4532#endif /* CONFIG_DPP */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004533 if (hapd->public_action_cb) {
4534 hapd->public_action_cb(hapd->public_action_cb_ctx,
4535 (u8 *) mgmt, len,
4536 hapd->iface->freq);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004537 }
Dmitry Shmidt4b9d52f2013-02-05 17:44:43 -08004538 if (hapd->public_action_cb2) {
Dmitry Shmidtf8623282013-02-20 14:34:59 -08004539 hapd->public_action_cb2(hapd->public_action_cb2_ctx,
Dmitry Shmidt4b9d52f2013-02-05 17:44:43 -08004540 (u8 *) mgmt, len,
4541 hapd->iface->freq);
4542 }
4543 if (hapd->public_action_cb || hapd->public_action_cb2)
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004544 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004545 break;
4546 case WLAN_ACTION_VENDOR_SPECIFIC:
4547 if (hapd->vendor_action_cb) {
4548 if (hapd->vendor_action_cb(hapd->vendor_action_cb_ctx,
4549 (u8 *) mgmt, len,
4550 hapd->iface->freq) == 0)
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004551 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004552 }
4553 break;
Dmitry Shmidt849734c2016-05-27 09:59:01 -07004554 case WLAN_ACTION_RADIO_MEASUREMENT:
4555 hostapd_handle_radio_measurement(hapd, (const u8 *) mgmt, len);
4556 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004557 }
4558
4559 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4560 HOSTAPD_LEVEL_DEBUG,
4561 "handle_action - unknown action category %d or invalid "
4562 "frame",
4563 mgmt->u.action.category);
Dmitry Shmidtd13095b2016-08-22 14:02:19 -07004564 if (!is_multicast_ether_addr(mgmt->da) &&
4565 !(mgmt->u.action.category & 0x80) &&
4566 !is_multicast_ether_addr(mgmt->sa)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004567 struct ieee80211_mgmt *resp;
4568
4569 /*
4570 * IEEE 802.11-REVma/D9.0 - 7.3.1.11
4571 * Return the Action frame to the source without change
4572 * except that MSB of the Category set to 1.
4573 */
4574 wpa_printf(MSG_DEBUG, "IEEE 802.11: Return unknown Action "
4575 "frame back to sender");
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004576 resp = os_memdup(mgmt, len);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004577 if (resp == NULL)
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004578 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004579 os_memcpy(resp->da, resp->sa, ETH_ALEN);
4580 os_memcpy(resp->sa, hapd->own_addr, ETH_ALEN);
4581 os_memcpy(resp->bssid, hapd->own_addr, ETH_ALEN);
4582 resp->u.action.category |= 0x80;
4583
Dmitry Shmidt61d9df32012-08-29 16:22:06 -07004584 if (hostapd_drv_send_mlme(hapd, resp, len, 0) < 0) {
4585 wpa_printf(MSG_ERROR, "IEEE 802.11: Failed to send "
4586 "Action frame");
4587 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004588 os_free(resp);
4589 }
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004590
4591 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004592}
4593
4594
4595/**
4596 * ieee802_11_mgmt - process incoming IEEE 802.11 management frames
4597 * @hapd: hostapd BSS data structure (the BSS to which the management frame was
4598 * sent to)
4599 * @buf: management frame data (starting from IEEE 802.11 header)
4600 * @len: length of frame data in octets
4601 * @fi: meta data about received frame (signal level, etc.)
4602 *
4603 * Process all incoming IEEE 802.11 management frames. This will be called for
4604 * each frame received from the kernel driver through wlan#ap interface. In
4605 * addition, it can be called to re-inserted pending frames (e.g., when using
4606 * external RADIUS server as an MAC ACL).
4607 */
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004608int ieee802_11_mgmt(struct hostapd_data *hapd, const u8 *buf, size_t len,
4609 struct hostapd_frame_info *fi)
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004610{
4611 struct ieee80211_mgmt *mgmt;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004612 u16 fc, stype;
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004613 int ret = 0;
Roshan Pius3a1667e2018-07-03 15:17:14 -07004614 unsigned int freq;
4615 int ssi_signal = fi ? fi->ssi_signal : 0;
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004616
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004617 if (len < 24)
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004618 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004619
Roshan Pius3a1667e2018-07-03 15:17:14 -07004620 if (fi && fi->freq)
4621 freq = fi->freq;
4622 else
4623 freq = hapd->iface->freq;
4624
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004625 mgmt = (struct ieee80211_mgmt *) buf;
4626 fc = le_to_host16(mgmt->frame_control);
4627 stype = WLAN_FC_GET_STYPE(fc);
4628
4629 if (stype == WLAN_FC_STYPE_BEACON) {
4630 handle_beacon(hapd, mgmt, len, fi);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004631 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004632 }
4633
Dmitry Shmidt7f2c7532016-08-15 09:48:12 -07004634 if (!is_broadcast_ether_addr(mgmt->bssid) &&
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004635#ifdef CONFIG_P2P
4636 /* Invitation responses can be sent with the peer MAC as BSSID */
4637 !((hapd->conf->p2p & P2P_GROUP_OWNER) &&
4638 stype == WLAN_FC_STYPE_ACTION) &&
4639#endif /* CONFIG_P2P */
Dmitry Shmidt6c0da2b2015-01-05 13:08:17 -08004640#ifdef CONFIG_MESH
4641 !(hapd->conf->mesh & MESH_ENABLED) &&
4642#endif /* CONFIG_MESH */
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004643 os_memcmp(mgmt->bssid, hapd->own_addr, ETH_ALEN) != 0) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08004644 wpa_printf(MSG_INFO, "MGMT: BSSID=" MACSTR " not our address",
4645 MAC2STR(mgmt->bssid));
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004646 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004647 }
4648
4649
4650 if (stype == WLAN_FC_STYPE_PROBE_REQ) {
Roshan Pius3a1667e2018-07-03 15:17:14 -07004651 handle_probe_req(hapd, mgmt, len, ssi_signal);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004652 return 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004653 }
4654
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07004655 if ((!is_broadcast_ether_addr(mgmt->da) ||
4656 stype != WLAN_FC_STYPE_ACTION) &&
4657 os_memcmp(mgmt->da, hapd->own_addr, ETH_ALEN) != 0) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004658 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4659 HOSTAPD_LEVEL_DEBUG,
4660 "MGMT: DA=" MACSTR " not our address",
4661 MAC2STR(mgmt->da));
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004662 return 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004663 }
4664
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08004665 if (hapd->iconf->track_sta_max_num)
Roshan Pius3a1667e2018-07-03 15:17:14 -07004666 sta_track_add(hapd->iface, mgmt->sa, ssi_signal);
Dmitry Shmidtd80a4012015-11-05 16:35:40 -08004667
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004668 switch (stype) {
4669 case WLAN_FC_STYPE_AUTH:
4670 wpa_printf(MSG_DEBUG, "mgmt::auth");
Hai Shalom021b0b52019-04-10 11:17:58 -07004671 handle_auth(hapd, mgmt, len, ssi_signal, 0);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004672 ret = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004673 break;
4674 case WLAN_FC_STYPE_ASSOC_REQ:
4675 wpa_printf(MSG_DEBUG, "mgmt::assoc_req");
Hai Shalom74f70d42019-02-11 14:42:39 -08004676 handle_assoc(hapd, mgmt, len, 0, ssi_signal);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004677 ret = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004678 break;
4679 case WLAN_FC_STYPE_REASSOC_REQ:
4680 wpa_printf(MSG_DEBUG, "mgmt::reassoc_req");
Hai Shalom74f70d42019-02-11 14:42:39 -08004681 handle_assoc(hapd, mgmt, len, 1, ssi_signal);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004682 ret = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004683 break;
4684 case WLAN_FC_STYPE_DISASSOC:
4685 wpa_printf(MSG_DEBUG, "mgmt::disassoc");
4686 handle_disassoc(hapd, mgmt, len);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004687 ret = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004688 break;
4689 case WLAN_FC_STYPE_DEAUTH:
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004690 wpa_msg(hapd->msg_ctx, MSG_DEBUG, "mgmt::deauth");
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004691 handle_deauth(hapd, mgmt, len);
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004692 ret = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004693 break;
4694 case WLAN_FC_STYPE_ACTION:
4695 wpa_printf(MSG_DEBUG, "mgmt::action");
Roshan Pius3a1667e2018-07-03 15:17:14 -07004696 ret = handle_action(hapd, mgmt, len, freq);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004697 break;
4698 default:
4699 hostapd_logger(hapd, mgmt->sa, HOSTAPD_MODULE_IEEE80211,
4700 HOSTAPD_LEVEL_DEBUG,
4701 "unknown mgmt frame subtype %d", stype);
4702 break;
4703 }
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004704
4705 return ret;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004706}
4707
4708
4709static void handle_auth_cb(struct hostapd_data *hapd,
4710 const struct ieee80211_mgmt *mgmt,
4711 size_t len, int ok)
4712{
4713 u16 auth_alg, auth_transaction, status_code;
4714 struct sta_info *sta;
4715
Hai Shalome5e28bb2019-01-28 14:51:04 -08004716 if (len < IEEE80211_HDRLEN + sizeof(mgmt->u.auth)) {
4717 wpa_printf(MSG_INFO, "handle_auth_cb - too short payload (len=%lu)",
4718 (unsigned long) len);
4719
4720 /*
4721 * Initialize status_code here because we are not able to read
4722 * it from the short payload.
4723 */
4724 status_code = WLAN_STATUS_UNSPECIFIED_FAILURE;
4725 goto fail;
4726 }
4727
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004728 sta = ap_get_sta(hapd, mgmt->da);
4729 if (!sta) {
Hai Shalom39ba6fc2019-01-22 12:40:38 -08004730 wpa_printf(MSG_DEBUG, "handle_auth_cb: STA " MACSTR
4731 " not found",
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004732 MAC2STR(mgmt->da));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004733 return;
4734 }
4735
4736 auth_alg = le_to_host16(mgmt->u.auth.auth_alg);
4737 auth_transaction = le_to_host16(mgmt->u.auth.auth_transaction);
4738 status_code = le_to_host16(mgmt->u.auth.status_code);
4739
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004740 if (!ok) {
4741 hostapd_logger(hapd, mgmt->da, HOSTAPD_MODULE_IEEE80211,
4742 HOSTAPD_LEVEL_NOTICE,
4743 "did not acknowledge authentication response");
4744 goto fail;
4745 }
4746
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004747 if (status_code == WLAN_STATUS_SUCCESS &&
4748 ((auth_alg == WLAN_AUTH_OPEN && auth_transaction == 2) ||
4749 (auth_alg == WLAN_AUTH_SHARED_KEY && auth_transaction == 4))) {
4750 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4751 HOSTAPD_LEVEL_INFO, "authenticated");
4752 sta->flags |= WLAN_STA_AUTH;
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004753 if (sta->added_unassoc)
4754 hostapd_set_sta_flags(hapd, sta);
4755 return;
4756 }
4757
4758fail:
4759 if (status_code != WLAN_STATUS_SUCCESS && sta->added_unassoc) {
4760 hostapd_drv_sta_remove(hapd, sta->addr);
4761 sta->added_unassoc = 0;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004762 }
4763}
4764
4765
Dmitry Shmidtc2ebb4b2013-07-24 12:57:51 -07004766static void hostapd_set_wds_encryption(struct hostapd_data *hapd,
4767 struct sta_info *sta,
4768 char *ifname_wds)
4769{
4770 int i;
Dmitry Shmidt9d9e6022015-04-23 10:34:55 -07004771 struct hostapd_ssid *ssid = &hapd->conf->ssid;
Dmitry Shmidtc2ebb4b2013-07-24 12:57:51 -07004772
4773 if (hapd->conf->ieee802_1x || hapd->conf->wpa)
4774 return;
4775
4776 for (i = 0; i < 4; i++) {
4777 if (ssid->wep.key[i] &&
4778 hostapd_drv_set_key(ifname_wds, hapd, WPA_ALG_WEP, NULL, i,
4779 i == ssid->wep.idx, NULL, 0,
4780 ssid->wep.key[i], ssid->wep.len[i])) {
4781 wpa_printf(MSG_WARNING,
4782 "Could not set WEP keys for WDS interface; %s",
4783 ifname_wds);
4784 break;
4785 }
4786 }
4787}
4788
4789
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004790static void handle_assoc_cb(struct hostapd_data *hapd,
4791 const struct ieee80211_mgmt *mgmt,
4792 size_t len, int reassoc, int ok)
4793{
4794 u16 status;
4795 struct sta_info *sta;
4796 int new_assoc = 1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004797
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004798 sta = ap_get_sta(hapd, mgmt->da);
4799 if (!sta) {
Dmitry Shmidtcce06662013-11-04 18:44:24 -08004800 wpa_printf(MSG_INFO, "handle_assoc_cb: STA " MACSTR " not found",
4801 MAC2STR(mgmt->da));
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004802 return;
4803 }
4804
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004805 if (len < IEEE80211_HDRLEN + (reassoc ? sizeof(mgmt->u.reassoc_resp) :
4806 sizeof(mgmt->u.assoc_resp))) {
4807 wpa_printf(MSG_INFO,
4808 "handle_assoc_cb(reassoc=%d) - too short payload (len=%lu)",
4809 reassoc, (unsigned long) len);
4810 hostapd_drv_sta_remove(hapd, sta->addr);
Dmitry Shmidtaa532512012-09-24 10:35:31 -07004811 return;
4812 }
4813
4814 if (reassoc)
4815 status = le_to_host16(mgmt->u.reassoc_resp.status_code);
4816 else
4817 status = le_to_host16(mgmt->u.assoc_resp.status_code);
4818
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08004819 if (!ok) {
4820 hostapd_logger(hapd, mgmt->da, HOSTAPD_MODULE_IEEE80211,
4821 HOSTAPD_LEVEL_DEBUG,
4822 "did not acknowledge association response");
4823 sta->flags &= ~WLAN_STA_ASSOC_REQ_OK;
4824 /* The STA is added only in case of SUCCESS */
4825 if (status == WLAN_STATUS_SUCCESS)
4826 hostapd_drv_sta_remove(hapd, sta->addr);
4827
4828 return;
4829 }
4830
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004831 if (status != WLAN_STATUS_SUCCESS)
Dmitry Shmidtb36ed7c2014-03-17 10:57:26 -07004832 return;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004833
4834 /* Stop previous accounting session, if one is started, and allocate
4835 * new session id for the new session. */
4836 accounting_sta_stop(hapd, sta);
4837
4838 hostapd_logger(hapd, sta->addr, HOSTAPD_MODULE_IEEE80211,
4839 HOSTAPD_LEVEL_INFO,
4840 "associated (aid %d)",
4841 sta->aid);
4842
4843 if (sta->flags & WLAN_STA_ASSOC)
4844 new_assoc = 0;
4845 sta->flags |= WLAN_STA_ASSOC;
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08004846 sta->flags &= ~WLAN_STA_WNM_SLEEP_MODE;
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004847 if ((!hapd->conf->ieee802_1x && !hapd->conf->wpa &&
4848 !hapd->conf->osen) ||
4849 sta->auth_alg == WLAN_AUTH_FILS_SK ||
4850 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
4851 sta->auth_alg == WLAN_AUTH_FILS_PK ||
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004852 sta->auth_alg == WLAN_AUTH_FT) {
4853 /*
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004854 * Open, static WEP, FT protocol, or FILS; no separate
4855 * authorization step.
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004856 */
4857 ap_sta_set_authorized(hapd, sta, 1);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004858 }
4859
4860 if (reassoc)
4861 mlme_reassociate_indication(hapd, sta);
4862 else
4863 mlme_associate_indication(hapd, sta);
4864
4865#ifdef CONFIG_IEEE80211W
4866 sta->sa_query_timed_out = 0;
4867#endif /* CONFIG_IEEE80211W */
4868
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004869 if (sta->eapol_sm == NULL) {
4870 /*
4871 * This STA does not use RADIUS server for EAP authentication,
4872 * so bind it to the selected VLAN interface now, since the
4873 * interface selection is not going to change anymore.
4874 */
Dmitry Shmidt83474442015-04-15 13:47:09 -07004875 if (ap_sta_bind_vlan(hapd, sta) < 0)
Dmitry Shmidtb36ed7c2014-03-17 10:57:26 -07004876 return;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004877 } else if (sta->vlan_id) {
4878 /* VLAN ID already set (e.g., by PMKSA caching), so bind STA */
Dmitry Shmidt83474442015-04-15 13:47:09 -07004879 if (ap_sta_bind_vlan(hapd, sta) < 0)
Dmitry Shmidtb36ed7c2014-03-17 10:57:26 -07004880 return;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004881 }
4882
4883 hostapd_set_sta_flags(hapd, sta);
4884
Dmitry Shmidt29333592017-01-09 12:27:11 -08004885 if (!(sta->flags & WLAN_STA_WDS) && sta->pending_wds_enable) {
4886 wpa_printf(MSG_DEBUG, "Enable 4-address WDS mode for STA "
4887 MACSTR " based on pending request",
4888 MAC2STR(sta->addr));
4889 sta->pending_wds_enable = 0;
4890 sta->flags |= WLAN_STA_WDS;
4891 }
4892
Hai Shalom74f70d42019-02-11 14:42:39 -08004893 if (sta->flags & (WLAN_STA_WDS | WLAN_STA_MULTI_AP)) {
Dmitry Shmidtabb90a32016-12-05 15:34:39 -08004894 int ret;
4895 char ifname_wds[IFNAMSIZ + 1];
4896
4897 wpa_printf(MSG_DEBUG, "Reenable 4-address WDS mode for STA "
4898 MACSTR " (aid %u)",
4899 MAC2STR(sta->addr), sta->aid);
4900 ret = hostapd_set_wds_sta(hapd, ifname_wds, sta->addr,
4901 sta->aid, 1);
4902 if (!ret)
4903 hostapd_set_wds_encryption(hapd, sta, ifname_wds);
4904 }
4905
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004906 if (sta->auth_alg == WLAN_AUTH_FT)
4907 wpa_auth_sm_event(sta->wpa_sm, WPA_ASSOC_FT);
4908 else
4909 wpa_auth_sm_event(sta->wpa_sm, WPA_ASSOC);
4910 hapd->new_assoc_sta_cb(hapd, sta, !new_assoc);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004911 ieee802_1x_notify_port_enabled(sta->eapol_sm, 1);
Dmitry Shmidt31a29cc2016-03-09 15:58:17 -08004912
Dmitry Shmidt9839ecd2016-11-07 11:05:47 -08004913#ifdef CONFIG_FILS
4914 if ((sta->auth_alg == WLAN_AUTH_FILS_SK ||
4915 sta->auth_alg == WLAN_AUTH_FILS_SK_PFS ||
4916 sta->auth_alg == WLAN_AUTH_FILS_PK) &&
4917 fils_set_tk(sta->wpa_sm) < 0) {
4918 wpa_printf(MSG_DEBUG, "FILS: TK configuration failed");
4919 ap_sta_disconnect(hapd, sta, sta->addr,
4920 WLAN_REASON_UNSPECIFIED);
4921 return;
4922 }
4923#endif /* CONFIG_FILS */
4924
Dmitry Shmidt31a29cc2016-03-09 15:58:17 -08004925 if (sta->pending_eapol_rx) {
4926 struct os_reltime now, age;
4927
4928 os_get_reltime(&now);
4929 os_reltime_sub(&now, &sta->pending_eapol_rx->rx_time, &age);
4930 if (age.sec == 0 && age.usec < 200000) {
4931 wpa_printf(MSG_DEBUG,
4932 "Process pending EAPOL frame that was received from " MACSTR " just before association notification",
4933 MAC2STR(sta->addr));
4934 ieee802_1x_receive(
4935 hapd, mgmt->da,
4936 wpabuf_head(sta->pending_eapol_rx->buf),
4937 wpabuf_len(sta->pending_eapol_rx->buf));
4938 }
4939 wpabuf_free(sta->pending_eapol_rx->buf);
4940 os_free(sta->pending_eapol_rx);
4941 sta->pending_eapol_rx = NULL;
4942 }
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07004943}
4944
4945
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004946static void handle_deauth_cb(struct hostapd_data *hapd,
4947 const struct ieee80211_mgmt *mgmt,
4948 size_t len, int ok)
4949{
4950 struct sta_info *sta;
Dmitry Shmidtd13095b2016-08-22 14:02:19 -07004951 if (is_multicast_ether_addr(mgmt->da))
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004952 return;
4953 sta = ap_get_sta(hapd, mgmt->da);
4954 if (!sta) {
4955 wpa_printf(MSG_DEBUG, "handle_deauth_cb: STA " MACSTR
4956 " not found", MAC2STR(mgmt->da));
4957 return;
4958 }
4959 if (ok)
4960 wpa_printf(MSG_DEBUG, "STA " MACSTR " acknowledged deauth",
4961 MAC2STR(sta->addr));
4962 else
4963 wpa_printf(MSG_DEBUG, "STA " MACSTR " did not acknowledge "
4964 "deauth", MAC2STR(sta->addr));
4965
4966 ap_sta_deauth_cb(hapd, sta);
4967}
4968
4969
4970static void handle_disassoc_cb(struct hostapd_data *hapd,
4971 const struct ieee80211_mgmt *mgmt,
4972 size_t len, int ok)
4973{
4974 struct sta_info *sta;
Dmitry Shmidtd13095b2016-08-22 14:02:19 -07004975 if (is_multicast_ether_addr(mgmt->da))
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08004976 return;
4977 sta = ap_get_sta(hapd, mgmt->da);
4978 if (!sta) {
4979 wpa_printf(MSG_DEBUG, "handle_disassoc_cb: STA " MACSTR
4980 " not found", MAC2STR(mgmt->da));
4981 return;
4982 }
4983 if (ok)
4984 wpa_printf(MSG_DEBUG, "STA " MACSTR " acknowledged disassoc",
4985 MAC2STR(sta->addr));
4986 else
4987 wpa_printf(MSG_DEBUG, "STA " MACSTR " did not acknowledge "
4988 "disassoc", MAC2STR(sta->addr));
4989
4990 ap_sta_disassoc_cb(hapd, sta);
4991}
4992
4993
Dmitry Shmidt29333592017-01-09 12:27:11 -08004994static void handle_action_cb(struct hostapd_data *hapd,
4995 const struct ieee80211_mgmt *mgmt,
4996 size_t len, int ok)
4997{
4998 struct sta_info *sta;
Paul Stewart092955c2017-02-06 09:13:09 -08004999 const struct rrm_measurement_report_element *report;
Dmitry Shmidt29333592017-01-09 12:27:11 -08005000
5001 if (is_multicast_ether_addr(mgmt->da))
5002 return;
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07005003#ifdef CONFIG_DPP
5004 if (len >= IEEE80211_HDRLEN + 6 &&
5005 mgmt->u.action.category == WLAN_ACTION_PUBLIC &&
5006 mgmt->u.action.u.vs_public_action.action ==
5007 WLAN_PA_VENDOR_SPECIFIC &&
5008 WPA_GET_BE24(mgmt->u.action.u.vs_public_action.oui) ==
5009 OUI_WFA &&
5010 mgmt->u.action.u.vs_public_action.variable[0] ==
5011 DPP_OUI_TYPE) {
5012 const u8 *pos, *end;
5013
5014 pos = &mgmt->u.action.u.vs_public_action.variable[1];
5015 end = ((const u8 *) mgmt) + len;
5016 hostapd_dpp_tx_status(hapd, mgmt->da, pos, end - pos, ok);
5017 return;
5018 }
5019 if (len >= IEEE80211_HDRLEN + 2 &&
5020 mgmt->u.action.category == WLAN_ACTION_PUBLIC &&
5021 (mgmt->u.action.u.public_action.action ==
5022 WLAN_PA_GAS_INITIAL_REQ ||
5023 mgmt->u.action.u.public_action.action ==
5024 WLAN_PA_GAS_COMEBACK_REQ)) {
5025 const u8 *pos, *end;
5026
5027 pos = mgmt->u.action.u.public_action.variable;
5028 end = ((const u8 *) mgmt) + len;
5029 gas_query_ap_tx_status(hapd->gas, mgmt->da, pos, end - pos, ok);
5030 return;
5031 }
5032#endif /* CONFIG_DPP */
Dmitry Shmidt29333592017-01-09 12:27:11 -08005033 sta = ap_get_sta(hapd, mgmt->da);
5034 if (!sta) {
5035 wpa_printf(MSG_DEBUG, "handle_action_cb: STA " MACSTR
5036 " not found", MAC2STR(mgmt->da));
5037 return;
5038 }
5039
Paul Stewart092955c2017-02-06 09:13:09 -08005040 if (len < 24 + 5 + sizeof(*report))
Dmitry Shmidt29333592017-01-09 12:27:11 -08005041 return;
Paul Stewart092955c2017-02-06 09:13:09 -08005042 report = (const struct rrm_measurement_report_element *)
5043 &mgmt->u.action.u.rrm.variable[2];
Dmitry Shmidt29333592017-01-09 12:27:11 -08005044 if (mgmt->u.action.category == WLAN_ACTION_RADIO_MEASUREMENT &&
Paul Stewart092955c2017-02-06 09:13:09 -08005045 mgmt->u.action.u.rrm.action == WLAN_RRM_RADIO_MEASUREMENT_REQUEST &&
5046 report->eid == WLAN_EID_MEASURE_REQUEST &&
5047 report->len >= 3 &&
5048 report->type == MEASURE_TYPE_BEACON)
Dmitry Shmidt29333592017-01-09 12:27:11 -08005049 hostapd_rrm_beacon_req_tx_status(hapd, mgmt, len, ok);
5050}
5051
5052
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005053/**
5054 * ieee802_11_mgmt_cb - Process management frame TX status callback
5055 * @hapd: hostapd BSS data structure (the BSS from which the management frame
5056 * was sent from)
5057 * @buf: management frame data (starting from IEEE 802.11 header)
5058 * @len: length of frame data in octets
5059 * @stype: management frame subtype from frame control field
5060 * @ok: Whether the frame was ACK'ed
5061 */
5062void ieee802_11_mgmt_cb(struct hostapd_data *hapd, const u8 *buf, size_t len,
5063 u16 stype, int ok)
5064{
5065 const struct ieee80211_mgmt *mgmt;
5066 mgmt = (const struct ieee80211_mgmt *) buf;
5067
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08005068#ifdef CONFIG_TESTING_OPTIONS
5069 if (hapd->ext_mgmt_frame_handling) {
Dmitry Shmidtd2986c22017-10-23 14:22:09 -07005070 size_t hex_len = 2 * len + 1;
5071 char *hex = os_malloc(hex_len);
5072
5073 if (hex) {
5074 wpa_snprintf_hex(hex, hex_len, buf, len);
5075 wpa_msg(hapd->msg_ctx, MSG_INFO,
5076 "MGMT-TX-STATUS stype=%u ok=%d buf=%s",
5077 stype, ok, hex);
5078 os_free(hex);
5079 }
Dmitry Shmidtfb79edc2014-01-10 10:45:54 -08005080 return;
5081 }
5082#endif /* CONFIG_TESTING_OPTIONS */
5083
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005084 switch (stype) {
5085 case WLAN_FC_STYPE_AUTH:
5086 wpa_printf(MSG_DEBUG, "mgmt::auth cb");
5087 handle_auth_cb(hapd, mgmt, len, ok);
5088 break;
5089 case WLAN_FC_STYPE_ASSOC_RESP:
5090 wpa_printf(MSG_DEBUG, "mgmt::assoc_resp cb");
5091 handle_assoc_cb(hapd, mgmt, len, 0, ok);
5092 break;
5093 case WLAN_FC_STYPE_REASSOC_RESP:
5094 wpa_printf(MSG_DEBUG, "mgmt::reassoc_resp cb");
5095 handle_assoc_cb(hapd, mgmt, len, 1, ok);
5096 break;
5097 case WLAN_FC_STYPE_PROBE_RESP:
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08005098 wpa_printf(MSG_EXCESSIVE, "mgmt::proberesp cb ok=%d", ok);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005099 break;
5100 case WLAN_FC_STYPE_DEAUTH:
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08005101 wpa_printf(MSG_DEBUG, "mgmt::deauth cb");
5102 handle_deauth_cb(hapd, mgmt, len, ok);
5103 break;
5104 case WLAN_FC_STYPE_DISASSOC:
5105 wpa_printf(MSG_DEBUG, "mgmt::disassoc cb");
5106 handle_disassoc_cb(hapd, mgmt, len, ok);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005107 break;
5108 case WLAN_FC_STYPE_ACTION:
Dmitry Shmidt57c2d392016-02-23 13:40:19 -08005109 wpa_printf(MSG_DEBUG, "mgmt::action cb ok=%d", ok);
Dmitry Shmidt29333592017-01-09 12:27:11 -08005110 handle_action_cb(hapd, mgmt, len, ok);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005111 break;
5112 default:
Dmitry Shmidtcce06662013-11-04 18:44:24 -08005113 wpa_printf(MSG_INFO, "unknown mgmt cb frame subtype %d", stype);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005114 break;
5115 }
5116}
5117
5118
5119int ieee802_11_get_mib(struct hostapd_data *hapd, char *buf, size_t buflen)
5120{
5121 /* TODO */
5122 return 0;
5123}
5124
5125
5126int ieee802_11_get_mib_sta(struct hostapd_data *hapd, struct sta_info *sta,
5127 char *buf, size_t buflen)
5128{
5129 /* TODO */
5130 return 0;
5131}
5132
5133
5134void hostapd_tx_status(struct hostapd_data *hapd, const u8 *addr,
5135 const u8 *buf, size_t len, int ack)
5136{
5137 struct sta_info *sta;
5138 struct hostapd_iface *iface = hapd->iface;
5139
5140 sta = ap_get_sta(hapd, addr);
5141 if (sta == NULL && iface->num_bss > 1) {
5142 size_t j;
5143 for (j = 0; j < iface->num_bss; j++) {
5144 hapd = iface->bss[j];
5145 sta = ap_get_sta(hapd, addr);
5146 if (sta)
5147 break;
5148 }
5149 }
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005150 if (sta == NULL || !(sta->flags & WLAN_STA_ASSOC))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005151 return;
5152 if (sta->flags & WLAN_STA_PENDING_POLL) {
5153 wpa_printf(MSG_DEBUG, "STA " MACSTR " %s pending "
5154 "activity poll", MAC2STR(sta->addr),
5155 ack ? "ACKed" : "did not ACK");
5156 if (ack)
5157 sta->flags &= ~WLAN_STA_PENDING_POLL;
5158 }
5159
5160 ieee802_1x_tx_status(hapd, sta, buf, len, ack);
5161}
5162
5163
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08005164void hostapd_eapol_tx_status(struct hostapd_data *hapd, const u8 *dst,
5165 const u8 *data, size_t len, int ack)
5166{
5167 struct sta_info *sta;
5168 struct hostapd_iface *iface = hapd->iface;
5169
5170 sta = ap_get_sta(hapd, dst);
5171 if (sta == NULL && iface->num_bss > 1) {
5172 size_t j;
5173 for (j = 0; j < iface->num_bss; j++) {
5174 hapd = iface->bss[j];
5175 sta = ap_get_sta(hapd, dst);
5176 if (sta)
5177 break;
5178 }
5179 }
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005180 if (sta == NULL || !(sta->flags & WLAN_STA_ASSOC)) {
5181 wpa_printf(MSG_DEBUG, "Ignore TX status for Data frame to STA "
5182 MACSTR " that is not currently associated",
5183 MAC2STR(dst));
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08005184 return;
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005185 }
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08005186
5187 ieee802_1x_eapol_tx_status(hapd, sta, data, len, ack);
5188}
5189
5190
5191void hostapd_client_poll_ok(struct hostapd_data *hapd, const u8 *addr)
5192{
5193 struct sta_info *sta;
5194 struct hostapd_iface *iface = hapd->iface;
5195
5196 sta = ap_get_sta(hapd, addr);
5197 if (sta == NULL && iface->num_bss > 1) {
5198 size_t j;
5199 for (j = 0; j < iface->num_bss; j++) {
5200 hapd = iface->bss[j];
5201 sta = ap_get_sta(hapd, addr);
5202 if (sta)
5203 break;
5204 }
5205 }
5206 if (sta == NULL)
5207 return;
Dmitry Shmidt849734c2016-05-27 09:59:01 -07005208 wpa_msg(hapd->msg_ctx, MSG_INFO, AP_STA_POLL_OK MACSTR,
5209 MAC2STR(sta->addr));
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -08005210 if (!(sta->flags & WLAN_STA_PENDING_POLL))
5211 return;
5212
5213 wpa_printf(MSG_DEBUG, "STA " MACSTR " ACKed pending "
5214 "activity poll", MAC2STR(sta->addr));
5215 sta->flags &= ~WLAN_STA_PENDING_POLL;
5216}
5217
5218
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005219void ieee802_11_rx_from_unknown(struct hostapd_data *hapd, const u8 *src,
5220 int wds)
5221{
5222 struct sta_info *sta;
5223
5224 sta = ap_get_sta(hapd, src);
Dmitry Shmidt29333592017-01-09 12:27:11 -08005225 if (sta &&
5226 ((sta->flags & WLAN_STA_ASSOC) ||
5227 ((sta->flags & WLAN_STA_ASSOC_REQ_OK) && wds))) {
Dmitry Shmidtaa532512012-09-24 10:35:31 -07005228 if (!hapd->conf->wds_sta)
5229 return;
5230
Dmitry Shmidt29333592017-01-09 12:27:11 -08005231 if ((sta->flags & (WLAN_STA_ASSOC | WLAN_STA_ASSOC_REQ_OK)) ==
5232 WLAN_STA_ASSOC_REQ_OK) {
5233 wpa_printf(MSG_DEBUG,
5234 "Postpone 4-address WDS mode enabling for STA "
5235 MACSTR " since TX status for AssocResp is not yet known",
5236 MAC2STR(sta->addr));
5237 sta->pending_wds_enable = 1;
5238 return;
5239 }
5240
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005241 if (wds && !(sta->flags & WLAN_STA_WDS)) {
Dmitry Shmidtc2ebb4b2013-07-24 12:57:51 -07005242 int ret;
5243 char ifname_wds[IFNAMSIZ + 1];
5244
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005245 wpa_printf(MSG_DEBUG, "Enable 4-address WDS mode for "
5246 "STA " MACSTR " (aid %u)",
5247 MAC2STR(sta->addr), sta->aid);
5248 sta->flags |= WLAN_STA_WDS;
Dmitry Shmidtc2ebb4b2013-07-24 12:57:51 -07005249 ret = hostapd_set_wds_sta(hapd, ifname_wds,
5250 sta->addr, sta->aid, 1);
5251 if (!ret)
5252 hostapd_set_wds_encryption(hapd, sta,
5253 ifname_wds);
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005254 }
5255 return;
5256 }
5257
5258 wpa_printf(MSG_DEBUG, "Data/PS-poll frame from not associated STA "
5259 MACSTR, MAC2STR(src));
Dmitry Shmidtd13095b2016-08-22 14:02:19 -07005260 if (is_multicast_ether_addr(src)) {
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07005261 /* Broadcast bit set in SA?! Ignore the frame silently. */
5262 return;
5263 }
5264
5265 if (sta && (sta->flags & WLAN_STA_ASSOC_REQ_OK)) {
5266 wpa_printf(MSG_DEBUG, "Association Response to the STA has "
5267 "already been sent, but no TX status yet known - "
5268 "ignore Class 3 frame issue with " MACSTR,
5269 MAC2STR(src));
5270 return;
5271 }
5272
5273 if (sta && (sta->flags & WLAN_STA_AUTH))
5274 hostapd_drv_sta_disassoc(
5275 hapd, src,
5276 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA);
5277 else
5278 hostapd_drv_sta_deauth(
5279 hapd, src,
5280 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA);
5281}
5282
5283
5284#endif /* CONFIG_NATIVE_WINDOWS */