2 * Wi-Fi Protected Setup - Enrollee
3 * Copyright (c) 2008, Jouni Malinen <j@w1.fi>
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
12 #include "crypto/crypto.h"
13 #include "crypto/sha256.h"
14 #include "crypto/random.h"
16 #include "wps_dev_attr.h"
19 static int wps_build_wps_state(struct wps_data
*wps
, struct wpabuf
*msg
)
23 state
= wps
->wps
->wps_state
;
25 state
= WPS_STATE_NOT_CONFIGURED
;
26 wpa_printf(MSG_DEBUG
, "WPS: * Wi-Fi Protected Setup State (%d)",
28 wpabuf_put_be16(msg
, ATTR_WPS_STATE
);
29 wpabuf_put_be16(msg
, 1);
30 wpabuf_put_u8(msg
, state
);
35 static int wps_build_e_hash(struct wps_data
*wps
, struct wpabuf
*msg
)
41 if (random_get_bytes(wps
->snonce
, 2 * WPS_SECRET_NONCE_LEN
) < 0)
43 wpa_hexdump(MSG_DEBUG
, "WPS: E-S1", wps
->snonce
, WPS_SECRET_NONCE_LEN
);
44 wpa_hexdump(MSG_DEBUG
, "WPS: E-S2",
45 wps
->snonce
+ WPS_SECRET_NONCE_LEN
, WPS_SECRET_NONCE_LEN
);
47 if (wps
->dh_pubkey_e
== NULL
|| wps
->dh_pubkey_r
== NULL
) {
48 wpa_printf(MSG_DEBUG
, "WPS: DH public keys not available for "
53 wpa_printf(MSG_DEBUG
, "WPS: * E-Hash1");
54 wpabuf_put_be16(msg
, ATTR_E_HASH1
);
55 wpabuf_put_be16(msg
, SHA256_MAC_LEN
);
56 hash
= wpabuf_put(msg
, SHA256_MAC_LEN
);
57 /* E-Hash1 = HMAC_AuthKey(E-S1 || PSK1 || PK_E || PK_R) */
58 addr
[0] = wps
->snonce
;
59 len
[0] = WPS_SECRET_NONCE_LEN
;
62 addr
[2] = wpabuf_head(wps
->dh_pubkey_e
);
63 len
[2] = wpabuf_len(wps
->dh_pubkey_e
);
64 addr
[3] = wpabuf_head(wps
->dh_pubkey_r
);
65 len
[3] = wpabuf_len(wps
->dh_pubkey_r
);
66 hmac_sha256_vector(wps
->authkey
, WPS_AUTHKEY_LEN
, 4, addr
, len
, hash
);
67 wpa_hexdump(MSG_DEBUG
, "WPS: E-Hash1", hash
, SHA256_MAC_LEN
);
69 wpa_printf(MSG_DEBUG
, "WPS: * E-Hash2");
70 wpabuf_put_be16(msg
, ATTR_E_HASH2
);
71 wpabuf_put_be16(msg
, SHA256_MAC_LEN
);
72 hash
= wpabuf_put(msg
, SHA256_MAC_LEN
);
73 /* E-Hash2 = HMAC_AuthKey(E-S2 || PSK2 || PK_E || PK_R) */
74 addr
[0] = wps
->snonce
+ WPS_SECRET_NONCE_LEN
;
76 hmac_sha256_vector(wps
->authkey
, WPS_AUTHKEY_LEN
, 4, addr
, len
, hash
);
77 wpa_hexdump(MSG_DEBUG
, "WPS: E-Hash2", hash
, SHA256_MAC_LEN
);
83 static int wps_build_e_snonce1(struct wps_data
*wps
, struct wpabuf
*msg
)
85 wpa_printf(MSG_DEBUG
, "WPS: * E-SNonce1");
86 wpabuf_put_be16(msg
, ATTR_E_SNONCE1
);
87 wpabuf_put_be16(msg
, WPS_SECRET_NONCE_LEN
);
88 wpabuf_put_data(msg
, wps
->snonce
, WPS_SECRET_NONCE_LEN
);
93 static int wps_build_e_snonce2(struct wps_data
*wps
, struct wpabuf
*msg
)
95 wpa_printf(MSG_DEBUG
, "WPS: * E-SNonce2");
96 wpabuf_put_be16(msg
, ATTR_E_SNONCE2
);
97 wpabuf_put_be16(msg
, WPS_SECRET_NONCE_LEN
);
98 wpabuf_put_data(msg
, wps
->snonce
+ WPS_SECRET_NONCE_LEN
,
99 WPS_SECRET_NONCE_LEN
);
104 static struct wpabuf
* wps_build_m1(struct wps_data
*wps
)
109 if (random_get_bytes(wps
->nonce_e
, WPS_NONCE_LEN
) < 0)
111 wpa_hexdump(MSG_DEBUG
, "WPS: Enrollee Nonce",
112 wps
->nonce_e
, WPS_NONCE_LEN
);
114 wpa_printf(MSG_DEBUG
, "WPS: Building Message M1");
115 msg
= wpabuf_alloc(1000);
119 config_methods
= wps
->wps
->config_methods
;
120 if (wps
->wps
->ap
&& !wps
->pbc_in_m1
&&
121 (wps
->dev_password_len
!= 0 ||
122 (config_methods
& WPS_CONFIG_DISPLAY
))) {
124 * These are the methods that the AP supports as an Enrollee
125 * for adding external Registrars, so remove PushButton.
127 * As a workaround for Windows 7 mechanism for probing WPS
128 * capabilities from M1, leave PushButton option if no PIN
129 * method is available or if WPS configuration enables PBC
132 config_methods
&= ~WPS_CONFIG_PUSHBUTTON
;
134 config_methods
&= ~(WPS_CONFIG_VIRT_PUSHBUTTON
|
135 WPS_CONFIG_PHY_PUSHBUTTON
);
136 #endif /* CONFIG_WPS2 */
139 if (wps_build_version(msg
) ||
140 wps_build_msg_type(msg
, WPS_M1
) ||
141 wps_build_uuid_e(msg
, wps
->uuid_e
) ||
142 wps_build_mac_addr(msg
, wps
->mac_addr_e
) ||
143 wps_build_enrollee_nonce(wps
, msg
) ||
144 wps_build_public_key(wps
, msg
) ||
145 wps_build_auth_type_flags(wps
, msg
) ||
146 wps_build_encr_type_flags(wps
, msg
) ||
147 wps_build_conn_type_flags(wps
, msg
) ||
148 wps_build_config_methods(msg
, config_methods
) ||
149 wps_build_wps_state(wps
, msg
) ||
150 wps_build_device_attrs(&wps
->wps
->dev
, msg
) ||
151 wps_build_rf_bands(&wps
->wps
->dev
, msg
,
152 wps
->wps
->rf_band_cb(wps
->wps
->cb_ctx
)) ||
153 wps_build_assoc_state(wps
, msg
) ||
154 wps_build_dev_password_id(msg
, wps
->dev_pw_id
) ||
155 wps_build_config_error(msg
, WPS_CFG_NO_ERROR
) ||
156 wps_build_os_version(&wps
->wps
->dev
, msg
) ||
157 wps_build_wfa_ext(msg
, 0, NULL
, 0) ||
158 wps_build_vendor_ext_m1(&wps
->wps
->dev
, msg
)) {
163 wps
->state
= RECV_M2
;
168 static struct wpabuf
* wps_build_m3(struct wps_data
*wps
)
172 wpa_printf(MSG_DEBUG
, "WPS: Building Message M3");
174 if (wps
->dev_password
== NULL
) {
175 wpa_printf(MSG_DEBUG
, "WPS: No Device Password available");
178 wps_derive_psk(wps
, wps
->dev_password
, wps
->dev_password_len
);
180 msg
= wpabuf_alloc(1000);
184 if (wps_build_version(msg
) ||
185 wps_build_msg_type(msg
, WPS_M3
) ||
186 wps_build_registrar_nonce(wps
, msg
) ||
187 wps_build_e_hash(wps
, msg
) ||
188 wps_build_wfa_ext(msg
, 0, NULL
, 0) ||
189 wps_build_authenticator(wps
, msg
)) {
194 wps
->state
= RECV_M4
;
199 static struct wpabuf
* wps_build_m5(struct wps_data
*wps
)
201 struct wpabuf
*msg
, *plain
;
203 wpa_printf(MSG_DEBUG
, "WPS: Building Message M5");
205 plain
= wpabuf_alloc(200);
209 msg
= wpabuf_alloc(1000);
215 if (wps_build_version(msg
) ||
216 wps_build_msg_type(msg
, WPS_M5
) ||
217 wps_build_registrar_nonce(wps
, msg
) ||
218 wps_build_e_snonce1(wps
, plain
) ||
219 wps_build_key_wrap_auth(wps
, plain
) ||
220 wps_build_encr_settings(wps
, msg
, plain
) ||
221 wps_build_wfa_ext(msg
, 0, NULL
, 0) ||
222 wps_build_authenticator(wps
, msg
)) {
229 wps
->state
= RECV_M6
;
234 static int wps_build_cred_ssid(struct wps_data
*wps
, struct wpabuf
*msg
)
236 wpa_printf(MSG_DEBUG
, "WPS: * SSID");
237 wpabuf_put_be16(msg
, ATTR_SSID
);
238 wpabuf_put_be16(msg
, wps
->wps
->ssid_len
);
239 wpabuf_put_data(msg
, wps
->wps
->ssid
, wps
->wps
->ssid_len
);
244 static int wps_build_cred_auth_type(struct wps_data
*wps
, struct wpabuf
*msg
)
246 u16 auth_type
= wps
->wps
->auth_types
;
248 /* Select the best authentication type */
249 if (auth_type
& WPS_AUTH_WPA2PSK
)
250 auth_type
= WPS_AUTH_WPA2PSK
;
251 else if (auth_type
& WPS_AUTH_WPAPSK
)
252 auth_type
= WPS_AUTH_WPAPSK
;
253 else if (auth_type
& WPS_AUTH_OPEN
)
254 auth_type
= WPS_AUTH_OPEN
;
255 else if (auth_type
& WPS_AUTH_SHARED
)
256 auth_type
= WPS_AUTH_SHARED
;
258 wpa_printf(MSG_DEBUG
, "WPS: * Authentication Type (0x%x)", auth_type
);
259 wpabuf_put_be16(msg
, ATTR_AUTH_TYPE
);
260 wpabuf_put_be16(msg
, 2);
261 wpabuf_put_be16(msg
, auth_type
);
266 static int wps_build_cred_encr_type(struct wps_data
*wps
, struct wpabuf
*msg
)
268 u16 encr_type
= wps
->wps
->encr_types
;
270 /* Select the best encryption type */
271 if (wps
->wps
->auth_types
& (WPS_AUTH_WPA2PSK
| WPS_AUTH_WPAPSK
)) {
272 if (encr_type
& WPS_ENCR_AES
)
273 encr_type
= WPS_ENCR_AES
;
274 else if (encr_type
& WPS_ENCR_TKIP
)
275 encr_type
= WPS_ENCR_TKIP
;
277 if (encr_type
& WPS_ENCR_WEP
)
278 encr_type
= WPS_ENCR_WEP
;
279 else if (encr_type
& WPS_ENCR_NONE
)
280 encr_type
= WPS_ENCR_NONE
;
283 wpa_printf(MSG_DEBUG
, "WPS: * Encryption Type (0x%x)", encr_type
);
284 wpabuf_put_be16(msg
, ATTR_ENCR_TYPE
);
285 wpabuf_put_be16(msg
, 2);
286 wpabuf_put_be16(msg
, encr_type
);
291 static int wps_build_cred_network_key(struct wps_data
*wps
, struct wpabuf
*msg
)
293 wpa_printf(MSG_DEBUG
, "WPS: * Network Key");
294 wpabuf_put_be16(msg
, ATTR_NETWORK_KEY
);
295 wpabuf_put_be16(msg
, wps
->wps
->network_key_len
);
296 wpabuf_put_data(msg
, wps
->wps
->network_key
, wps
->wps
->network_key_len
);
301 static int wps_build_cred_mac_addr(struct wps_data
*wps
, struct wpabuf
*msg
)
303 wpa_printf(MSG_DEBUG
, "WPS: * MAC Address (AP BSSID)");
304 wpabuf_put_be16(msg
, ATTR_MAC_ADDR
);
305 wpabuf_put_be16(msg
, ETH_ALEN
);
306 wpabuf_put_data(msg
, wps
->wps
->dev
.mac_addr
, ETH_ALEN
);
311 static int wps_build_ap_settings(struct wps_data
*wps
, struct wpabuf
*plain
)
313 if (wps
->wps
->ap_settings
) {
314 wpa_printf(MSG_DEBUG
, "WPS: * AP Settings (pre-configured)");
315 wpabuf_put_data(plain
, wps
->wps
->ap_settings
,
316 wps
->wps
->ap_settings_len
);
320 return wps_build_cred_ssid(wps
, plain
) ||
321 wps_build_cred_mac_addr(wps
, plain
) ||
322 wps_build_cred_auth_type(wps
, plain
) ||
323 wps_build_cred_encr_type(wps
, plain
) ||
324 wps_build_cred_network_key(wps
, plain
);
328 static struct wpabuf
* wps_build_m7(struct wps_data
*wps
)
330 struct wpabuf
*msg
, *plain
;
332 wpa_printf(MSG_DEBUG
, "WPS: Building Message M7");
334 plain
= wpabuf_alloc(500 + wps
->wps
->ap_settings_len
);
338 msg
= wpabuf_alloc(1000 + wps
->wps
->ap_settings_len
);
344 if (wps_build_version(msg
) ||
345 wps_build_msg_type(msg
, WPS_M7
) ||
346 wps_build_registrar_nonce(wps
, msg
) ||
347 wps_build_e_snonce2(wps
, plain
) ||
348 (wps
->wps
->ap
&& wps_build_ap_settings(wps
, plain
)) ||
349 wps_build_key_wrap_auth(wps
, plain
) ||
350 wps_build_encr_settings(wps
, msg
, plain
) ||
351 wps_build_wfa_ext(msg
, 0, NULL
, 0) ||
352 wps_build_authenticator(wps
, msg
)) {
359 if (wps
->wps
->ap
&& wps
->wps
->registrar
) {
361 * If the Registrar is only learning our current configuration,
362 * it may not continue protocol run to successful completion.
363 * Store information here to make sure it remains available.
365 wps_device_store(wps
->wps
->registrar
, &wps
->peer_dev
,
369 wps
->state
= RECV_M8
;
374 static struct wpabuf
* wps_build_wsc_done(struct wps_data
*wps
)
378 wpa_printf(MSG_DEBUG
, "WPS: Building Message WSC_Done");
380 msg
= wpabuf_alloc(1000);
384 if (wps_build_version(msg
) ||
385 wps_build_msg_type(msg
, WPS_WSC_DONE
) ||
386 wps_build_enrollee_nonce(wps
, msg
) ||
387 wps_build_registrar_nonce(wps
, msg
) ||
388 wps_build_wfa_ext(msg
, 0, NULL
, 0)) {
394 wps
->state
= RECV_ACK
;
396 wps_success_event(wps
->wps
, wps
->peer_dev
.mac_addr
);
397 wps
->state
= WPS_FINISHED
;
403 struct wpabuf
* wps_enrollee_get_msg(struct wps_data
*wps
,
404 enum wsc_op_code
*op_code
)
408 switch (wps
->state
) {
410 msg
= wps_build_m1(wps
);
414 msg
= wps_build_m3(wps
);
418 msg
= wps_build_m5(wps
);
422 msg
= wps_build_m7(wps
);
427 msg
= wps_build_wsc_nack(wps
);
431 msg
= wps_build_wsc_ack(wps
);
434 /* Another M2/M2D may be received */
435 wps
->state
= RECV_M2
;
439 msg
= wps_build_wsc_nack(wps
);
443 msg
= wps_build_wsc_done(wps
);
447 wpa_printf(MSG_DEBUG
, "WPS: Unsupported state %d for building "
448 "a message", wps
->state
);
453 if (*op_code
== WSC_MSG
&& msg
) {
454 /* Save a copy of the last message for Authenticator derivation
456 wpabuf_free(wps
->last_msg
);
457 wps
->last_msg
= wpabuf_dup(msg
);
464 static int wps_process_registrar_nonce(struct wps_data
*wps
, const u8
*r_nonce
)
466 if (r_nonce
== NULL
) {
467 wpa_printf(MSG_DEBUG
, "WPS: No Registrar Nonce received");
471 os_memcpy(wps
->nonce_r
, r_nonce
, WPS_NONCE_LEN
);
472 wpa_hexdump(MSG_DEBUG
, "WPS: Registrar Nonce",
473 wps
->nonce_r
, WPS_NONCE_LEN
);
479 static int wps_process_enrollee_nonce(struct wps_data
*wps
, const u8
*e_nonce
)
481 if (e_nonce
== NULL
) {
482 wpa_printf(MSG_DEBUG
, "WPS: No Enrollee Nonce received");
486 if (os_memcmp(wps
->nonce_e
, e_nonce
, WPS_NONCE_LEN
) != 0) {
487 wpa_printf(MSG_DEBUG
, "WPS: Invalid Enrollee Nonce received");
495 static int wps_process_uuid_r(struct wps_data
*wps
, const u8
*uuid_r
)
497 if (uuid_r
== NULL
) {
498 wpa_printf(MSG_DEBUG
, "WPS: No UUID-R received");
502 os_memcpy(wps
->uuid_r
, uuid_r
, WPS_UUID_LEN
);
503 wpa_hexdump(MSG_DEBUG
, "WPS: UUID-R", wps
->uuid_r
, WPS_UUID_LEN
);
509 static int wps_process_pubkey(struct wps_data
*wps
, const u8
*pk
,
512 if (pk
== NULL
|| pk_len
== 0) {
513 wpa_printf(MSG_DEBUG
, "WPS: No Public Key received");
517 wpabuf_free(wps
->dh_pubkey_r
);
518 wps
->dh_pubkey_r
= wpabuf_alloc_copy(pk
, pk_len
);
519 if (wps
->dh_pubkey_r
== NULL
)
522 if (wps_derive_keys(wps
) < 0)
529 static int wps_process_r_hash1(struct wps_data
*wps
, const u8
*r_hash1
)
531 if (r_hash1
== NULL
) {
532 wpa_printf(MSG_DEBUG
, "WPS: No R-Hash1 received");
536 os_memcpy(wps
->peer_hash1
, r_hash1
, WPS_HASH_LEN
);
537 wpa_hexdump(MSG_DEBUG
, "WPS: R-Hash1", wps
->peer_hash1
, WPS_HASH_LEN
);
543 static int wps_process_r_hash2(struct wps_data
*wps
, const u8
*r_hash2
)
545 if (r_hash2
== NULL
) {
546 wpa_printf(MSG_DEBUG
, "WPS: No R-Hash2 received");
550 os_memcpy(wps
->peer_hash2
, r_hash2
, WPS_HASH_LEN
);
551 wpa_hexdump(MSG_DEBUG
, "WPS: R-Hash2", wps
->peer_hash2
, WPS_HASH_LEN
);
557 static int wps_process_r_snonce1(struct wps_data
*wps
, const u8
*r_snonce1
)
559 u8 hash
[SHA256_MAC_LEN
];
563 if (r_snonce1
== NULL
) {
564 wpa_printf(MSG_DEBUG
, "WPS: No R-SNonce1 received");
568 wpa_hexdump_key(MSG_DEBUG
, "WPS: R-SNonce1", r_snonce1
,
569 WPS_SECRET_NONCE_LEN
);
571 /* R-Hash1 = HMAC_AuthKey(R-S1 || PSK1 || PK_E || PK_R) */
573 len
[0] = WPS_SECRET_NONCE_LEN
;
575 len
[1] = WPS_PSK_LEN
;
576 addr
[2] = wpabuf_head(wps
->dh_pubkey_e
);
577 len
[2] = wpabuf_len(wps
->dh_pubkey_e
);
578 addr
[3] = wpabuf_head(wps
->dh_pubkey_r
);
579 len
[3] = wpabuf_len(wps
->dh_pubkey_r
);
580 hmac_sha256_vector(wps
->authkey
, WPS_AUTHKEY_LEN
, 4, addr
, len
, hash
);
582 if (os_memcmp(wps
->peer_hash1
, hash
, WPS_HASH_LEN
) != 0) {
583 wpa_printf(MSG_DEBUG
, "WPS: R-Hash1 derived from R-S1 does "
584 "not match with the pre-committed value");
585 wps
->config_error
= WPS_CFG_DEV_PASSWORD_AUTH_FAILURE
;
586 wps_pwd_auth_fail_event(wps
->wps
, 1, 1, wps
->peer_dev
.mac_addr
);
590 wpa_printf(MSG_DEBUG
, "WPS: Registrar proved knowledge of the first "
591 "half of the device password");
597 static int wps_process_r_snonce2(struct wps_data
*wps
, const u8
*r_snonce2
)
599 u8 hash
[SHA256_MAC_LEN
];
603 if (r_snonce2
== NULL
) {
604 wpa_printf(MSG_DEBUG
, "WPS: No R-SNonce2 received");
608 wpa_hexdump_key(MSG_DEBUG
, "WPS: R-SNonce2", r_snonce2
,
609 WPS_SECRET_NONCE_LEN
);
611 /* R-Hash2 = HMAC_AuthKey(R-S2 || PSK2 || PK_E || PK_R) */
613 len
[0] = WPS_SECRET_NONCE_LEN
;
615 len
[1] = WPS_PSK_LEN
;
616 addr
[2] = wpabuf_head(wps
->dh_pubkey_e
);
617 len
[2] = wpabuf_len(wps
->dh_pubkey_e
);
618 addr
[3] = wpabuf_head(wps
->dh_pubkey_r
);
619 len
[3] = wpabuf_len(wps
->dh_pubkey_r
);
620 hmac_sha256_vector(wps
->authkey
, WPS_AUTHKEY_LEN
, 4, addr
, len
, hash
);
622 if (os_memcmp(wps
->peer_hash2
, hash
, WPS_HASH_LEN
) != 0) {
623 wpa_printf(MSG_DEBUG
, "WPS: R-Hash2 derived from R-S2 does "
624 "not match with the pre-committed value");
625 wps
->config_error
= WPS_CFG_DEV_PASSWORD_AUTH_FAILURE
;
626 wps_pwd_auth_fail_event(wps
->wps
, 1, 2, wps
->peer_dev
.mac_addr
);
630 wpa_printf(MSG_DEBUG
, "WPS: Registrar proved knowledge of the second "
631 "half of the device password");
637 static int wps_process_cred_e(struct wps_data
*wps
, const u8
*cred
,
638 size_t cred_len
, int wps2
)
640 struct wps_parse_attr attr
;
644 wpa_printf(MSG_DEBUG
, "WPS: Received Credential");
645 os_memset(&wps
->cred
, 0, sizeof(wps
->cred
));
646 wpabuf_set(&msg
, cred
, cred_len
);
647 if (wps_parse_msg(&msg
, &attr
) < 0 ||
648 wps_process_cred(&attr
, &wps
->cred
))
651 if (os_memcmp(wps
->cred
.mac_addr
, wps
->wps
->dev
.mac_addr
, ETH_ALEN
) !=
653 wpa_printf(MSG_DEBUG
, "WPS: MAC Address in the Credential ("
654 MACSTR
") does not match with own address (" MACSTR
655 ")", MAC2STR(wps
->cred
.mac_addr
),
656 MAC2STR(wps
->wps
->dev
.mac_addr
));
658 * In theory, this could be consider fatal error, but there are
659 * number of deployed implementations using other address here
660 * due to unclarity in the specification. For interoperability
661 * reasons, allow this to be processed since we do not really
662 * use the MAC Address information for anything.
664 #ifdef CONFIG_WPS_STRICT
666 wpa_printf(MSG_INFO
, "WPS: Do not accept incorrect "
667 "MAC Address in AP Settings");
670 #endif /* CONFIG_WPS_STRICT */
674 if (!(wps
->cred
.encr_type
&
675 (WPS_ENCR_NONE
| WPS_ENCR_TKIP
| WPS_ENCR_AES
))) {
676 if (wps
->cred
.encr_type
& WPS_ENCR_WEP
) {
677 wpa_printf(MSG_INFO
, "WPS: Reject Credential "
678 "due to WEP configuration");
679 wps
->error_indication
= WPS_EI_SECURITY_WEP_PROHIBITED
;
683 wpa_printf(MSG_INFO
, "WPS: Reject Credential due to "
684 "invalid encr_type 0x%x", wps
->cred
.encr_type
);
687 #endif /* CONFIG_WPS2 */
689 if (wps
->wps
->cred_cb
) {
690 wps
->cred
.cred_attr
= cred
- 4;
691 wps
->cred
.cred_attr_len
= cred_len
+ 4;
692 ret
= wps
->wps
->cred_cb(wps
->wps
->cb_ctx
, &wps
->cred
);
693 wps
->cred
.cred_attr
= NULL
;
694 wps
->cred
.cred_attr_len
= 0;
701 static int wps_process_creds(struct wps_data
*wps
, const u8
*cred
[],
702 size_t cred_len
[], size_t num_cred
, int wps2
)
711 wpa_printf(MSG_DEBUG
, "WPS: No Credential attributes "
716 for (i
= 0; i
< num_cred
; i
++) {
718 res
= wps_process_cred_e(wps
, cred
[i
], cred_len
[i
], wps2
);
722 wpa_printf(MSG_DEBUG
, "WPS: WEP credential skipped");
728 wpa_printf(MSG_DEBUG
, "WPS: No valid Credential attribute "
737 static int wps_process_ap_settings_e(struct wps_data
*wps
,
738 struct wps_parse_attr
*attr
,
739 struct wpabuf
*attrs
, int wps2
)
741 struct wps_credential cred
;
746 if (wps_process_ap_settings(attr
, &cred
) < 0)
749 wpa_printf(MSG_INFO
, "WPS: Received new AP configuration from "
752 if (os_memcmp(cred
.mac_addr
, wps
->wps
->dev
.mac_addr
, ETH_ALEN
) !=
754 wpa_printf(MSG_DEBUG
, "WPS: MAC Address in the AP Settings ("
755 MACSTR
") does not match with own address (" MACSTR
756 ")", MAC2STR(cred
.mac_addr
),
757 MAC2STR(wps
->wps
->dev
.mac_addr
));
759 * In theory, this could be consider fatal error, but there are
760 * number of deployed implementations using other address here
761 * due to unclarity in the specification. For interoperability
762 * reasons, allow this to be processed since we do not really
763 * use the MAC Address information for anything.
765 #ifdef CONFIG_WPS_STRICT
767 wpa_printf(MSG_INFO
, "WPS: Do not accept incorrect "
768 "MAC Address in AP Settings");
771 #endif /* CONFIG_WPS_STRICT */
775 if (!(cred
.encr_type
& (WPS_ENCR_NONE
| WPS_ENCR_TKIP
| WPS_ENCR_AES
)))
777 if (cred
.encr_type
& WPS_ENCR_WEP
) {
778 wpa_printf(MSG_INFO
, "WPS: Reject new AP settings "
779 "due to WEP configuration");
780 wps
->error_indication
= WPS_EI_SECURITY_WEP_PROHIBITED
;
784 wpa_printf(MSG_INFO
, "WPS: Reject new AP settings due to "
785 "invalid encr_type 0x%x", cred
.encr_type
);
788 #endif /* CONFIG_WPS2 */
790 #ifdef CONFIG_WPS_STRICT
792 if ((cred
.encr_type
& (WPS_ENCR_TKIP
| WPS_ENCR_AES
)) ==
794 (cred
.auth_type
& (WPS_AUTH_WPAPSK
| WPS_AUTH_WPA2PSK
)) ==
796 wpa_printf(MSG_INFO
, "WPS-STRICT: Invalid WSC 2.0 "
797 "AP Settings: WPA-Personal/TKIP only");
798 wps
->error_indication
=
799 WPS_EI_SECURITY_TKIP_ONLY_PROHIBITED
;
803 #endif /* CONFIG_WPS_STRICT */
806 if ((cred
.encr_type
& (WPS_ENCR_TKIP
| WPS_ENCR_AES
)) == WPS_ENCR_TKIP
)
808 wpa_printf(MSG_DEBUG
, "WPS: Upgrade encr_type TKIP -> "
810 cred
.encr_type
|= WPS_ENCR_AES
;
813 if ((cred
.auth_type
& (WPS_AUTH_WPAPSK
| WPS_AUTH_WPA2PSK
)) ==
815 wpa_printf(MSG_DEBUG
, "WPS: Upgrade auth_type WPAPSK -> "
817 cred
.auth_type
|= WPS_AUTH_WPA2PSK
;
819 #endif /* CONFIG_WPS2 */
821 if (wps
->wps
->cred_cb
) {
822 cred
.cred_attr
= wpabuf_head(attrs
);
823 cred
.cred_attr_len
= wpabuf_len(attrs
);
824 wps
->wps
->cred_cb(wps
->wps
->cb_ctx
, &cred
);
831 static int wps_process_dev_pw_id(struct wps_data
*wps
, const u8
*dev_pw_id
)
835 if (dev_pw_id
== NULL
) {
836 wpa_printf(MSG_DEBUG
, "WPS: Device Password ID");
840 id
= WPA_GET_BE16(dev_pw_id
);
841 if (wps
->dev_pw_id
== id
) {
842 wpa_printf(MSG_DEBUG
, "WPS: Device Password ID %u", id
);
847 if ((id
== DEV_PW_DEFAULT
&&
848 wps
->dev_pw_id
== DEV_PW_REGISTRAR_SPECIFIED
) ||
849 (id
== DEV_PW_REGISTRAR_SPECIFIED
&&
850 wps
->dev_pw_id
== DEV_PW_DEFAULT
)) {
852 * Common P2P use cases indicate whether the PIN is from the
853 * client or GO using Device Password Id in M1/M2 in a way that
854 * does not look fully compliant with WSC specification. Anyway,
855 * this is deployed and needs to be allowed, so ignore changes
856 * between Registrar-Specified and Default PIN.
858 wpa_printf(MSG_DEBUG
, "WPS: Allow PIN Device Password ID "
862 #endif /* CONFIG_P2P */
864 wpa_printf(MSG_DEBUG
, "WPS: Registrar trying to change Device Password "
865 "ID from %u to %u", wps
->dev_pw_id
, id
);
867 if (wps
->dev_pw_id
== DEV_PW_PUSHBUTTON
&& id
== DEV_PW_DEFAULT
) {
868 wpa_printf(MSG_DEBUG
,
869 "WPS: Workaround - ignore PBC-to-PIN change");
873 if (wps
->alt_dev_password
&& wps
->alt_dev_pw_id
== id
) {
874 wpa_printf(MSG_DEBUG
, "WPS: Found a matching Device Password");
875 os_free(wps
->dev_password
);
876 wps
->dev_pw_id
= wps
->alt_dev_pw_id
;
877 wps
->dev_password
= wps
->alt_dev_password
;
878 wps
->dev_password_len
= wps
->alt_dev_password_len
;
879 wps
->alt_dev_password
= NULL
;
880 wps
->alt_dev_password_len
= 0;
888 static enum wps_process_res
wps_process_m2(struct wps_data
*wps
,
889 const struct wpabuf
*msg
,
890 struct wps_parse_attr
*attr
)
892 wpa_printf(MSG_DEBUG
, "WPS: Received M2");
894 if (wps
->state
!= RECV_M2
) {
895 wpa_printf(MSG_DEBUG
, "WPS: Unexpected state (%d) for "
896 "receiving M2", wps
->state
);
897 wps
->state
= SEND_WSC_NACK
;
901 if (wps_process_registrar_nonce(wps
, attr
->registrar_nonce
) ||
902 wps_process_enrollee_nonce(wps
, attr
->enrollee_nonce
) ||
903 wps_process_uuid_r(wps
, attr
->uuid_r
) ||
904 wps_process_dev_pw_id(wps
, attr
->dev_password_id
)) {
905 wps
->state
= SEND_WSC_NACK
;
910 * Stop here on an AP as an Enrollee if AP Setup is locked unless the
911 * special locked mode is used to allow protocol run up to M7 in order
912 * to support external Registrars that only learn the current AP
913 * configuration without changing it.
916 ((wps
->wps
->ap_setup_locked
&& wps
->wps
->ap_setup_locked
!= 2) ||
917 wps
->dev_password
== NULL
)) {
918 wpa_printf(MSG_DEBUG
, "WPS: AP Setup is locked - refuse "
919 "registration of a new Registrar");
920 wps
->config_error
= WPS_CFG_SETUP_LOCKED
;
921 wps
->state
= SEND_WSC_NACK
;
925 if (wps_process_pubkey(wps
, attr
->public_key
, attr
->public_key_len
) ||
926 wps_process_authenticator(wps
, attr
->authenticator
, msg
) ||
927 wps_process_device_attrs(&wps
->peer_dev
, attr
)) {
928 wps
->state
= SEND_WSC_NACK
;
932 wps
->state
= SEND_M3
;
937 static enum wps_process_res
wps_process_m2d(struct wps_data
*wps
,
938 struct wps_parse_attr
*attr
)
940 wpa_printf(MSG_DEBUG
, "WPS: Received M2D");
942 if (wps
->state
!= RECV_M2
) {
943 wpa_printf(MSG_DEBUG
, "WPS: Unexpected state (%d) for "
944 "receiving M2D", wps
->state
);
945 wps
->state
= SEND_WSC_NACK
;
949 wpa_hexdump_ascii(MSG_DEBUG
, "WPS: Manufacturer",
950 attr
->manufacturer
, attr
->manufacturer_len
);
951 wpa_hexdump_ascii(MSG_DEBUG
, "WPS: Model Name",
952 attr
->model_name
, attr
->model_name_len
);
953 wpa_hexdump_ascii(MSG_DEBUG
, "WPS: Model Number",
954 attr
->model_number
, attr
->model_number_len
);
955 wpa_hexdump_ascii(MSG_DEBUG
, "WPS: Serial Number",
956 attr
->serial_number
, attr
->serial_number_len
);
957 wpa_hexdump_ascii(MSG_DEBUG
, "WPS: Device Name",
958 attr
->dev_name
, attr
->dev_name_len
);
960 if (wps
->wps
->event_cb
) {
961 union wps_event_data data
;
962 struct wps_event_m2d
*m2d
= &data
.m2d
;
963 os_memset(&data
, 0, sizeof(data
));
964 if (attr
->config_methods
)
965 m2d
->config_methods
=
966 WPA_GET_BE16(attr
->config_methods
);
967 m2d
->manufacturer
= attr
->manufacturer
;
968 m2d
->manufacturer_len
= attr
->manufacturer_len
;
969 m2d
->model_name
= attr
->model_name
;
970 m2d
->model_name_len
= attr
->model_name_len
;
971 m2d
->model_number
= attr
->model_number
;
972 m2d
->model_number_len
= attr
->model_number_len
;
973 m2d
->serial_number
= attr
->serial_number
;
974 m2d
->serial_number_len
= attr
->serial_number_len
;
975 m2d
->dev_name
= attr
->dev_name
;
976 m2d
->dev_name_len
= attr
->dev_name_len
;
977 m2d
->primary_dev_type
= attr
->primary_dev_type
;
978 if (attr
->config_error
)
980 WPA_GET_BE16(attr
->config_error
);
981 if (attr
->dev_password_id
)
982 m2d
->dev_password_id
=
983 WPA_GET_BE16(attr
->dev_password_id
);
984 wps
->wps
->event_cb(wps
->wps
->cb_ctx
, WPS_EV_M2D
, &data
);
987 wps
->state
= RECEIVED_M2D
;
992 static enum wps_process_res
wps_process_m4(struct wps_data
*wps
,
993 const struct wpabuf
*msg
,
994 struct wps_parse_attr
*attr
)
996 struct wpabuf
*decrypted
;
997 struct wps_parse_attr eattr
;
999 wpa_printf(MSG_DEBUG
, "WPS: Received M4");
1001 if (wps
->state
!= RECV_M4
) {
1002 wpa_printf(MSG_DEBUG
, "WPS: Unexpected state (%d) for "
1003 "receiving M4", wps
->state
);
1004 wps
->state
= SEND_WSC_NACK
;
1005 return WPS_CONTINUE
;
1008 if (wps_process_enrollee_nonce(wps
, attr
->enrollee_nonce
) ||
1009 wps_process_authenticator(wps
, attr
->authenticator
, msg
) ||
1010 wps_process_r_hash1(wps
, attr
->r_hash1
) ||
1011 wps_process_r_hash2(wps
, attr
->r_hash2
)) {
1012 wps
->state
= SEND_WSC_NACK
;
1013 return WPS_CONTINUE
;
1016 decrypted
= wps_decrypt_encr_settings(wps
, attr
->encr_settings
,
1017 attr
->encr_settings_len
);
1018 if (decrypted
== NULL
) {
1019 wpa_printf(MSG_DEBUG
, "WPS: Failed to decrypted Encrypted "
1020 "Settings attribute");
1021 wps
->state
= SEND_WSC_NACK
;
1022 return WPS_CONTINUE
;
1025 if (wps_validate_m4_encr(decrypted
, attr
->version2
!= NULL
) < 0) {
1026 wpabuf_free(decrypted
);
1027 wps
->state
= SEND_WSC_NACK
;
1028 return WPS_CONTINUE
;
1031 wpa_printf(MSG_DEBUG
, "WPS: Processing decrypted Encrypted Settings "
1033 if (wps_parse_msg(decrypted
, &eattr
) < 0 ||
1034 wps_process_key_wrap_auth(wps
, decrypted
, eattr
.key_wrap_auth
) ||
1035 wps_process_r_snonce1(wps
, eattr
.r_snonce1
)) {
1036 wpabuf_free(decrypted
);
1037 wps
->state
= SEND_WSC_NACK
;
1038 return WPS_CONTINUE
;
1040 wpabuf_free(decrypted
);
1042 wps
->state
= SEND_M5
;
1043 return WPS_CONTINUE
;
1047 static enum wps_process_res
wps_process_m6(struct wps_data
*wps
,
1048 const struct wpabuf
*msg
,
1049 struct wps_parse_attr
*attr
)
1051 struct wpabuf
*decrypted
;
1052 struct wps_parse_attr eattr
;
1054 wpa_printf(MSG_DEBUG
, "WPS: Received M6");
1056 if (wps
->state
!= RECV_M6
) {
1057 wpa_printf(MSG_DEBUG
, "WPS: Unexpected state (%d) for "
1058 "receiving M6", wps
->state
);
1059 wps
->state
= SEND_WSC_NACK
;
1060 return WPS_CONTINUE
;
1063 if (wps_process_enrollee_nonce(wps
, attr
->enrollee_nonce
) ||
1064 wps_process_authenticator(wps
, attr
->authenticator
, msg
)) {
1065 wps
->state
= SEND_WSC_NACK
;
1066 return WPS_CONTINUE
;
1069 decrypted
= wps_decrypt_encr_settings(wps
, attr
->encr_settings
,
1070 attr
->encr_settings_len
);
1071 if (decrypted
== NULL
) {
1072 wpa_printf(MSG_DEBUG
, "WPS: Failed to decrypted Encrypted "
1073 "Settings attribute");
1074 wps
->state
= SEND_WSC_NACK
;
1075 return WPS_CONTINUE
;
1078 if (wps_validate_m6_encr(decrypted
, attr
->version2
!= NULL
) < 0) {
1079 wpabuf_free(decrypted
);
1080 wps
->state
= SEND_WSC_NACK
;
1081 return WPS_CONTINUE
;
1084 wpa_printf(MSG_DEBUG
, "WPS: Processing decrypted Encrypted Settings "
1086 if (wps_parse_msg(decrypted
, &eattr
) < 0 ||
1087 wps_process_key_wrap_auth(wps
, decrypted
, eattr
.key_wrap_auth
) ||
1088 wps_process_r_snonce2(wps
, eattr
.r_snonce2
)) {
1089 wpabuf_free(decrypted
);
1090 wps
->state
= SEND_WSC_NACK
;
1091 return WPS_CONTINUE
;
1093 wpabuf_free(decrypted
);
1096 wps
->wps
->event_cb(wps
->wps
->cb_ctx
, WPS_EV_AP_PIN_SUCCESS
,
1099 wps
->state
= SEND_M7
;
1100 return WPS_CONTINUE
;
1104 static enum wps_process_res
wps_process_m8(struct wps_data
*wps
,
1105 const struct wpabuf
*msg
,
1106 struct wps_parse_attr
*attr
)
1108 struct wpabuf
*decrypted
;
1109 struct wps_parse_attr eattr
;
1111 wpa_printf(MSG_DEBUG
, "WPS: Received M8");
1113 if (wps
->state
!= RECV_M8
) {
1114 wpa_printf(MSG_DEBUG
, "WPS: Unexpected state (%d) for "
1115 "receiving M8", wps
->state
);
1116 wps
->state
= SEND_WSC_NACK
;
1117 return WPS_CONTINUE
;
1120 if (wps_process_enrollee_nonce(wps
, attr
->enrollee_nonce
) ||
1121 wps_process_authenticator(wps
, attr
->authenticator
, msg
)) {
1122 wps
->state
= SEND_WSC_NACK
;
1123 return WPS_CONTINUE
;
1126 if (wps
->wps
->ap
&& wps
->wps
->ap_setup_locked
) {
1128 * Stop here if special ap_setup_locked == 2 mode allowed the
1129 * protocol to continue beyond M2. This allows ER to learn the
1130 * current AP settings without changing them.
1132 wpa_printf(MSG_DEBUG
, "WPS: AP Setup is locked - refuse "
1133 "registration of a new Registrar");
1134 wps
->config_error
= WPS_CFG_SETUP_LOCKED
;
1135 wps
->state
= SEND_WSC_NACK
;
1136 return WPS_CONTINUE
;
1139 decrypted
= wps_decrypt_encr_settings(wps
, attr
->encr_settings
,
1140 attr
->encr_settings_len
);
1141 if (decrypted
== NULL
) {
1142 wpa_printf(MSG_DEBUG
, "WPS: Failed to decrypted Encrypted "
1143 "Settings attribute");
1144 wps
->state
= SEND_WSC_NACK
;
1145 return WPS_CONTINUE
;
1148 if (wps_validate_m8_encr(decrypted
, wps
->wps
->ap
,
1149 attr
->version2
!= NULL
) < 0) {
1150 wpabuf_free(decrypted
);
1151 wps
->state
= SEND_WSC_NACK
;
1152 return WPS_CONTINUE
;
1155 wpa_printf(MSG_DEBUG
, "WPS: Processing decrypted Encrypted Settings "
1157 if (wps_parse_msg(decrypted
, &eattr
) < 0 ||
1158 wps_process_key_wrap_auth(wps
, decrypted
, eattr
.key_wrap_auth
) ||
1159 wps_process_creds(wps
, eattr
.cred
, eattr
.cred_len
,
1160 eattr
.num_cred
, attr
->version2
!= NULL
) ||
1161 wps_process_ap_settings_e(wps
, &eattr
, decrypted
,
1162 attr
->version2
!= NULL
)) {
1163 wpabuf_free(decrypted
);
1164 wps
->state
= SEND_WSC_NACK
;
1165 return WPS_CONTINUE
;
1167 wpabuf_free(decrypted
);
1169 wps
->state
= WPS_MSG_DONE
;
1170 return WPS_CONTINUE
;
1174 static enum wps_process_res
wps_process_wsc_msg(struct wps_data
*wps
,
1175 const struct wpabuf
*msg
)
1177 struct wps_parse_attr attr
;
1178 enum wps_process_res ret
= WPS_CONTINUE
;
1180 wpa_printf(MSG_DEBUG
, "WPS: Received WSC_MSG");
1182 if (wps_parse_msg(msg
, &attr
) < 0)
1185 if (attr
.enrollee_nonce
== NULL
||
1186 os_memcmp(wps
->nonce_e
, attr
.enrollee_nonce
, WPS_NONCE_LEN
) != 0) {
1187 wpa_printf(MSG_DEBUG
, "WPS: Mismatch in enrollee nonce");
1191 if (attr
.msg_type
== NULL
) {
1192 wpa_printf(MSG_DEBUG
, "WPS: No Message Type attribute");
1193 wps
->state
= SEND_WSC_NACK
;
1194 return WPS_CONTINUE
;
1197 switch (*attr
.msg_type
) {
1199 if (wps_validate_m2(msg
) < 0)
1201 ret
= wps_process_m2(wps
, msg
, &attr
);
1204 if (wps_validate_m2d(msg
) < 0)
1206 ret
= wps_process_m2d(wps
, &attr
);
1209 if (wps_validate_m4(msg
) < 0)
1211 ret
= wps_process_m4(wps
, msg
, &attr
);
1212 if (ret
== WPS_FAILURE
|| wps
->state
== SEND_WSC_NACK
)
1213 wps_fail_event(wps
->wps
, WPS_M4
, wps
->config_error
,
1214 wps
->error_indication
,
1215 wps
->peer_dev
.mac_addr
);
1218 if (wps_validate_m6(msg
) < 0)
1220 ret
= wps_process_m6(wps
, msg
, &attr
);
1221 if (ret
== WPS_FAILURE
|| wps
->state
== SEND_WSC_NACK
)
1222 wps_fail_event(wps
->wps
, WPS_M6
, wps
->config_error
,
1223 wps
->error_indication
,
1224 wps
->peer_dev
.mac_addr
);
1227 if (wps_validate_m8(msg
) < 0)
1229 ret
= wps_process_m8(wps
, msg
, &attr
);
1230 if (ret
== WPS_FAILURE
|| wps
->state
== SEND_WSC_NACK
)
1231 wps_fail_event(wps
->wps
, WPS_M8
, wps
->config_error
,
1232 wps
->error_indication
,
1233 wps
->peer_dev
.mac_addr
);
1236 wpa_printf(MSG_DEBUG
, "WPS: Unsupported Message Type %d",
1242 * Save a copy of the last message for Authenticator derivation if we
1243 * are continuing. However, skip M2D since it is not authenticated and
1244 * neither is the ACK/NACK response frame. This allows the possibly
1245 * following M2 to be processed correctly by using the previously sent
1246 * M1 in Authenticator derivation.
1248 if (ret
== WPS_CONTINUE
&& *attr
.msg_type
!= WPS_M2D
) {
1249 /* Save a copy of the last message for Authenticator derivation
1251 wpabuf_free(wps
->last_msg
);
1252 wps
->last_msg
= wpabuf_dup(msg
);
1259 static enum wps_process_res
wps_process_wsc_ack(struct wps_data
*wps
,
1260 const struct wpabuf
*msg
)
1262 struct wps_parse_attr attr
;
1264 wpa_printf(MSG_DEBUG
, "WPS: Received WSC_ACK");
1266 if (wps_parse_msg(msg
, &attr
) < 0)
1269 if (attr
.msg_type
== NULL
) {
1270 wpa_printf(MSG_DEBUG
, "WPS: No Message Type attribute");
1274 if (*attr
.msg_type
!= WPS_WSC_ACK
) {
1275 wpa_printf(MSG_DEBUG
, "WPS: Invalid Message Type %d",
1280 if (attr
.registrar_nonce
== NULL
||
1281 os_memcmp(wps
->nonce_r
, attr
.registrar_nonce
, WPS_NONCE_LEN
) != 0)
1283 wpa_printf(MSG_DEBUG
, "WPS: Mismatch in registrar nonce");
1287 if (attr
.enrollee_nonce
== NULL
||
1288 os_memcmp(wps
->nonce_e
, attr
.enrollee_nonce
, WPS_NONCE_LEN
) != 0) {
1289 wpa_printf(MSG_DEBUG
, "WPS: Mismatch in enrollee nonce");
1293 if (wps
->state
== RECV_ACK
&& wps
->wps
->ap
) {
1294 wpa_printf(MSG_DEBUG
, "WPS: External Registrar registration "
1295 "completed successfully");
1296 wps_success_event(wps
->wps
, wps
->peer_dev
.mac_addr
);
1297 wps
->state
= WPS_FINISHED
;
1305 static enum wps_process_res
wps_process_wsc_nack(struct wps_data
*wps
,
1306 const struct wpabuf
*msg
)
1308 struct wps_parse_attr attr
;
1311 wpa_printf(MSG_DEBUG
, "WPS: Received WSC_NACK");
1313 if (wps_parse_msg(msg
, &attr
) < 0)
1316 if (attr
.msg_type
== NULL
) {
1317 wpa_printf(MSG_DEBUG
, "WPS: No Message Type attribute");
1321 if (*attr
.msg_type
!= WPS_WSC_NACK
) {
1322 wpa_printf(MSG_DEBUG
, "WPS: Invalid Message Type %d",
1327 if (attr
.registrar_nonce
== NULL
||
1328 os_memcmp(wps
->nonce_r
, attr
.registrar_nonce
, WPS_NONCE_LEN
) != 0)
1330 wpa_printf(MSG_DEBUG
, "WPS: Mismatch in registrar nonce");
1331 wpa_hexdump(MSG_DEBUG
, "WPS: Received Registrar Nonce",
1332 attr
.registrar_nonce
, WPS_NONCE_LEN
);
1333 wpa_hexdump(MSG_DEBUG
, "WPS: Expected Registrar Nonce",
1334 wps
->nonce_r
, WPS_NONCE_LEN
);
1338 if (attr
.enrollee_nonce
== NULL
||
1339 os_memcmp(wps
->nonce_e
, attr
.enrollee_nonce
, WPS_NONCE_LEN
) != 0) {
1340 wpa_printf(MSG_DEBUG
, "WPS: Mismatch in enrollee nonce");
1341 wpa_hexdump(MSG_DEBUG
, "WPS: Received Enrollee Nonce",
1342 attr
.enrollee_nonce
, WPS_NONCE_LEN
);
1343 wpa_hexdump(MSG_DEBUG
, "WPS: Expected Enrollee Nonce",
1344 wps
->nonce_e
, WPS_NONCE_LEN
);
1348 if (attr
.config_error
== NULL
) {
1349 wpa_printf(MSG_DEBUG
, "WPS: No Configuration Error attribute "
1354 config_error
= WPA_GET_BE16(attr
.config_error
);
1355 wpa_printf(MSG_DEBUG
, "WPS: Registrar terminated negotiation with "
1356 "Configuration Error %d", config_error
);
1358 switch (wps
->state
) {
1360 wps_fail_event(wps
->wps
, WPS_M3
, config_error
,
1361 wps
->error_indication
, wps
->peer_dev
.mac_addr
);
1364 wps_fail_event(wps
->wps
, WPS_M5
, config_error
,
1365 wps
->error_indication
, wps
->peer_dev
.mac_addr
);
1368 wps_fail_event(wps
->wps
, WPS_M7
, config_error
,
1369 wps
->error_indication
, wps
->peer_dev
.mac_addr
);
1375 /* Followed by NACK if Enrollee is Supplicant or EAP-Failure if
1376 * Enrollee is Authenticator */
1377 wps
->state
= SEND_WSC_NACK
;
1383 enum wps_process_res
wps_enrollee_process_msg(struct wps_data
*wps
,
1384 enum wsc_op_code op_code
,
1385 const struct wpabuf
*msg
)
1388 wpa_printf(MSG_DEBUG
, "WPS: Processing received message (len=%lu "
1390 (unsigned long) wpabuf_len(msg
), op_code
);
1392 if (op_code
== WSC_UPnP
) {
1393 /* Determine the OpCode based on message type attribute */
1394 struct wps_parse_attr attr
;
1395 if (wps_parse_msg(msg
, &attr
) == 0 && attr
.msg_type
) {
1396 if (*attr
.msg_type
== WPS_WSC_ACK
)
1398 else if (*attr
.msg_type
== WPS_WSC_NACK
)
1406 return wps_process_wsc_msg(wps
, msg
);
1408 if (wps_validate_wsc_ack(msg
) < 0)
1410 return wps_process_wsc_ack(wps
, msg
);
1412 if (wps_validate_wsc_nack(msg
) < 0)
1414 return wps_process_wsc_nack(wps
, msg
);
1416 wpa_printf(MSG_DEBUG
, "WPS: Unsupported op_code %d", op_code
);