2 * Wi-Fi Protected Setup - attribute parsing
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.
13 #include "wps_attr_parse.h"
15 #ifndef CONFIG_WPS_STRICT
16 #define WPS_WORKAROUNDS
17 #endif /* CONFIG_WPS_STRICT */
20 static int wps_set_vendor_ext_wfa_subelem(struct wps_parse_attr *attr,
21 u8 id, u8 len, const u8 *pos)
23 wpa_printf(MSG_EXCESSIVE, "WPS: WFA subelement id=%u len=%u",
26 case WFA_ELEM_VERSION2:
28 wpa_printf(MSG_DEBUG, "WPS: Invalid Version2 length "
34 case WFA_ELEM_AUTHORIZEDMACS:
35 attr->authorized_macs = pos;
36 attr->authorized_macs_len = len;
38 case WFA_ELEM_NETWORK_KEY_SHAREABLE:
40 wpa_printf(MSG_DEBUG, "WPS: Invalid Network Key "
41 "Shareable length %u", len);
44 attr->network_key_shareable = pos;
46 case WFA_ELEM_REQUEST_TO_ENROLL:
48 wpa_printf(MSG_DEBUG, "WPS: Invalid Request to Enroll "
52 attr->request_to_enroll = pos;
54 case WFA_ELEM_SETTINGS_DELAY_TIME:
56 wpa_printf(MSG_DEBUG, "WPS: Invalid Settings Delay "
57 "Time length %u", len);
60 attr->settings_delay_time = pos;
63 wpa_printf(MSG_MSGDUMP, "WPS: Skipped unknown WFA Vendor "
64 "Extension subelement %u", id);
72 static int wps_parse_vendor_ext_wfa(struct wps_parse_attr *attr, const u8 *pos,
75 const u8 *end = pos + len;
78 while (pos + 2 < end) {
83 if (wps_set_vendor_ext_wfa_subelem(attr, id, elen, pos) < 0)
92 static int wps_parse_vendor_ext(struct wps_parse_attr *attr, const u8 *pos,
98 wpa_printf(MSG_DEBUG, "WPS: Skip invalid Vendor Extension");
102 vendor_id = WPA_GET_BE24(pos);
104 case WPS_VENDOR_ID_WFA:
105 return wps_parse_vendor_ext_wfa(attr, pos + 3, len - 3);
108 /* Handle unknown vendor extensions */
110 wpa_printf(MSG_MSGDUMP, "WPS: Unknown Vendor Extension (Vendor ID %u)",
113 if (len > WPS_MAX_VENDOR_EXT_LEN) {
114 wpa_printf(MSG_DEBUG, "WPS: Too long Vendor Extension (%u)",
119 if (attr->num_vendor_ext >= MAX_WPS_PARSE_VENDOR_EXT) {
120 wpa_printf(MSG_DEBUG, "WPS: Skipped Vendor Extension "
121 "attribute (max %d vendor extensions)",
122 MAX_WPS_PARSE_VENDOR_EXT);
125 attr->vendor_ext[attr->num_vendor_ext] = pos;
126 attr->vendor_ext_len[attr->num_vendor_ext] = len;
127 attr->num_vendor_ext++;
133 static int wps_set_attr(struct wps_parse_attr *attr, u16 type,
134 const u8 *pos, u16 len)
139 wpa_printf(MSG_DEBUG, "WPS: Invalid Version length %u",
147 wpa_printf(MSG_DEBUG, "WPS: Invalid Message Type "
151 attr->msg_type = pos;
153 case ATTR_ENROLLEE_NONCE:
154 if (len != WPS_NONCE_LEN) {
155 wpa_printf(MSG_DEBUG, "WPS: Invalid Enrollee Nonce "
159 attr->enrollee_nonce = pos;
161 case ATTR_REGISTRAR_NONCE:
162 if (len != WPS_NONCE_LEN) {
163 wpa_printf(MSG_DEBUG, "WPS: Invalid Registrar Nonce "
167 attr->registrar_nonce = pos;
170 if (len != WPS_UUID_LEN) {
171 wpa_printf(MSG_DEBUG, "WPS: Invalid UUID-E length %u",
178 if (len != WPS_UUID_LEN) {
179 wpa_printf(MSG_DEBUG, "WPS: Invalid UUID-R length %u",
185 case ATTR_AUTH_TYPE_FLAGS:
187 wpa_printf(MSG_DEBUG, "WPS: Invalid Authentication "
188 "Type Flags length %u", len);
191 attr->auth_type_flags = pos;
193 case ATTR_ENCR_TYPE_FLAGS:
195 wpa_printf(MSG_DEBUG, "WPS: Invalid Encryption Type "
196 "Flags length %u", len);
199 attr->encr_type_flags = pos;
201 case ATTR_CONN_TYPE_FLAGS:
203 wpa_printf(MSG_DEBUG, "WPS: Invalid Connection Type "
204 "Flags length %u", len);
207 attr->conn_type_flags = pos;
209 case ATTR_CONFIG_METHODS:
211 wpa_printf(MSG_DEBUG, "WPS: Invalid Config Methods "
215 attr->config_methods = pos;
217 case ATTR_SELECTED_REGISTRAR_CONFIG_METHODS:
219 wpa_printf(MSG_DEBUG, "WPS: Invalid Selected "
220 "Registrar Config Methods length %u", len);
223 attr->sel_reg_config_methods = pos;
225 case ATTR_PRIMARY_DEV_TYPE:
226 if (len != WPS_DEV_TYPE_LEN) {
227 wpa_printf(MSG_DEBUG, "WPS: Invalid Primary Device "
228 "Type length %u", len);
231 attr->primary_dev_type = pos;
235 wpa_printf(MSG_DEBUG, "WPS: Invalid RF Bands length "
239 attr->rf_bands = pos;
241 case ATTR_ASSOC_STATE:
243 wpa_printf(MSG_DEBUG, "WPS: Invalid Association State "
247 attr->assoc_state = pos;
249 case ATTR_CONFIG_ERROR:
251 wpa_printf(MSG_DEBUG, "WPS: Invalid Configuration "
252 "Error length %u", len);
255 attr->config_error = pos;
257 case ATTR_DEV_PASSWORD_ID:
259 wpa_printf(MSG_DEBUG, "WPS: Invalid Device Password "
260 "ID length %u", len);
263 attr->dev_password_id = pos;
265 case ATTR_OOB_DEVICE_PASSWORD:
266 if (len < WPS_OOB_PUBKEY_HASH_LEN + 2 +
267 WPS_OOB_DEVICE_PASSWORD_MIN_LEN ||
268 len > WPS_OOB_PUBKEY_HASH_LEN + 2 +
269 WPS_OOB_DEVICE_PASSWORD_LEN) {
270 wpa_printf(MSG_DEBUG, "WPS: Invalid OOB Device "
271 "Password length %u", len);
274 attr->oob_dev_password = pos;
275 attr->oob_dev_password_len = len;
277 case ATTR_OS_VERSION:
279 wpa_printf(MSG_DEBUG, "WPS: Invalid OS Version length "
283 attr->os_version = pos;
287 wpa_printf(MSG_DEBUG, "WPS: Invalid Wi-Fi Protected "
288 "Setup State length %u", len);
291 attr->wps_state = pos;
293 case ATTR_AUTHENTICATOR:
294 if (len != WPS_AUTHENTICATOR_LEN) {
295 wpa_printf(MSG_DEBUG, "WPS: Invalid Authenticator "
299 attr->authenticator = pos;
302 if (len != WPS_HASH_LEN) {
303 wpa_printf(MSG_DEBUG, "WPS: Invalid R-Hash1 length %u",
310 if (len != WPS_HASH_LEN) {
311 wpa_printf(MSG_DEBUG, "WPS: Invalid R-Hash2 length %u",
318 if (len != WPS_HASH_LEN) {
319 wpa_printf(MSG_DEBUG, "WPS: Invalid E-Hash1 length %u",
326 if (len != WPS_HASH_LEN) {
327 wpa_printf(MSG_DEBUG, "WPS: Invalid E-Hash2 length %u",
334 if (len != WPS_SECRET_NONCE_LEN) {
335 wpa_printf(MSG_DEBUG, "WPS: Invalid R-SNonce1 length "
339 attr->r_snonce1 = pos;
342 if (len != WPS_SECRET_NONCE_LEN) {
343 wpa_printf(MSG_DEBUG, "WPS: Invalid R-SNonce2 length "
347 attr->r_snonce2 = pos;
350 if (len != WPS_SECRET_NONCE_LEN) {
351 wpa_printf(MSG_DEBUG, "WPS: Invalid E-SNonce1 length "
355 attr->e_snonce1 = pos;
358 if (len != WPS_SECRET_NONCE_LEN) {
359 wpa_printf(MSG_DEBUG, "WPS: Invalid E-SNonce2 length "
363 attr->e_snonce2 = pos;
365 case ATTR_KEY_WRAP_AUTH:
366 if (len != WPS_KWA_LEN) {
367 wpa_printf(MSG_DEBUG, "WPS: Invalid Key Wrap "
368 "Authenticator length %u", len);
371 attr->key_wrap_auth = pos;
375 wpa_printf(MSG_DEBUG, "WPS: Invalid Authentication "
376 "Type length %u", len);
379 attr->auth_type = pos;
383 wpa_printf(MSG_DEBUG, "WPS: Invalid Encryption "
384 "Type length %u", len);
387 attr->encr_type = pos;
389 case ATTR_NETWORK_INDEX:
391 wpa_printf(MSG_DEBUG, "WPS: Invalid Network Index "
395 attr->network_idx = pos;
397 case ATTR_NETWORK_KEY_INDEX:
399 wpa_printf(MSG_DEBUG, "WPS: Invalid Network Key Index "
403 attr->network_key_idx = pos;
406 if (len != ETH_ALEN) {
407 wpa_printf(MSG_DEBUG, "WPS: Invalid MAC Address "
411 attr->mac_addr = pos;
413 case ATTR_KEY_PROVIDED_AUTO:
415 wpa_printf(MSG_DEBUG, "WPS: Invalid Key Provided "
416 "Automatically length %u", len);
419 attr->key_prov_auto = pos;
421 case ATTR_802_1X_ENABLED:
423 wpa_printf(MSG_DEBUG, "WPS: Invalid 802.1X Enabled "
427 attr->dot1x_enabled = pos;
429 case ATTR_SELECTED_REGISTRAR:
431 wpa_printf(MSG_DEBUG, "WPS: Invalid Selected Registrar"
435 attr->selected_registrar = pos;
437 case ATTR_REQUEST_TYPE:
439 wpa_printf(MSG_DEBUG, "WPS: Invalid Request Type "
443 attr->request_type = pos;
445 case ATTR_RESPONSE_TYPE:
447 wpa_printf(MSG_DEBUG, "WPS: Invalid Response Type "
451 attr->response_type = pos;
453 case ATTR_MANUFACTURER:
454 attr->manufacturer = pos;
455 attr->manufacturer_len = len;
457 case ATTR_MODEL_NAME:
458 attr->model_name = pos;
459 attr->model_name_len = len;
461 case ATTR_MODEL_NUMBER:
462 attr->model_number = pos;
463 attr->model_number_len = len;
465 case ATTR_SERIAL_NUMBER:
466 attr->serial_number = pos;
467 attr->serial_number_len = len;
470 attr->dev_name = pos;
471 attr->dev_name_len = len;
473 case ATTR_PUBLIC_KEY:
474 attr->public_key = pos;
475 attr->public_key_len = len;
477 case ATTR_ENCR_SETTINGS:
478 attr->encr_settings = pos;
479 attr->encr_settings_len = len;
482 if (attr->num_cred >= MAX_CRED_COUNT) {
483 wpa_printf(MSG_DEBUG, "WPS: Skipped Credential "
484 "attribute (max %d credentials)",
488 attr->cred[attr->num_cred] = pos;
489 attr->cred_len[attr->num_cred] = len;
494 attr->ssid_len = len;
496 case ATTR_NETWORK_KEY:
497 attr->network_key = pos;
498 attr->network_key_len = len;
501 attr->eap_type = pos;
502 attr->eap_type_len = len;
504 case ATTR_EAP_IDENTITY:
505 attr->eap_identity = pos;
506 attr->eap_identity_len = len;
508 case ATTR_AP_SETUP_LOCKED:
510 wpa_printf(MSG_DEBUG, "WPS: Invalid AP Setup Locked "
514 attr->ap_setup_locked = pos;
516 case ATTR_REQUESTED_DEV_TYPE:
517 if (len != WPS_DEV_TYPE_LEN) {
518 wpa_printf(MSG_DEBUG, "WPS: Invalid Requested Device "
519 "Type length %u", len);
522 if (attr->num_req_dev_type >= MAX_REQ_DEV_TYPE_COUNT) {
523 wpa_printf(MSG_DEBUG, "WPS: Skipped Requested Device "
524 "Type attribute (max %u types)",
525 MAX_REQ_DEV_TYPE_COUNT);
528 attr->req_dev_type[attr->num_req_dev_type] = pos;
529 attr->num_req_dev_type++;
531 case ATTR_SECONDARY_DEV_TYPE_LIST:
532 if (len > WPS_SEC_DEV_TYPE_MAX_LEN ||
533 (len % WPS_DEV_TYPE_LEN) > 0) {
534 wpa_printf(MSG_DEBUG, "WPS: Invalid Secondary Device "
535 "Type length %u", len);
538 attr->sec_dev_type_list = pos;
539 attr->sec_dev_type_list_len = len;
541 case ATTR_VENDOR_EXT:
542 if (wps_parse_vendor_ext(attr, pos, len) < 0)
545 case ATTR_AP_CHANNEL:
547 wpa_printf(MSG_DEBUG, "WPS: Invalid AP Channel "
551 attr->ap_channel = pos;
554 wpa_printf(MSG_DEBUG, "WPS: Unsupported attribute type 0x%x "
555 "len=%u", type, len);
563 int wps_parse_msg(const struct wpabuf *msg, struct wps_parse_attr *attr)
567 #ifdef WPS_WORKAROUNDS
569 #endif /* WPS_WORKAROUNDS */
571 os_memset(attr, 0, sizeof(*attr));
572 pos = wpabuf_head(msg);
573 end = pos + wpabuf_len(msg);
577 wpa_printf(MSG_DEBUG, "WPS: Invalid message - "
578 "%lu bytes remaining",
579 (unsigned long) (end - pos));
583 type = WPA_GET_BE16(pos);
585 len = WPA_GET_BE16(pos);
587 wpa_printf(MSG_EXCESSIVE, "WPS: attr type=0x%x len=%u",
589 if (len > end - pos) {
590 wpa_printf(MSG_DEBUG, "WPS: Attribute overflow");
591 wpa_hexdump_buf(MSG_MSGDUMP, "WPS: Message data", msg);
592 #ifdef WPS_WORKAROUNDS
594 * Some deployed APs seem to have a bug in encoding of
595 * Network Key attribute in the Credential attribute
596 * where they add an extra octet after the Network Key
597 * attribute at least when open network is being
600 if ((type & 0xff00) != 0x1000 &&
601 prev_type == ATTR_NETWORK_KEY) {
602 wpa_printf(MSG_DEBUG, "WPS: Workaround - try "
603 "to skip unexpected octet after "
608 #endif /* WPS_WORKAROUNDS */
612 #ifdef WPS_WORKAROUNDS
613 if (type == 0 && len == 0) {
615 * Mac OS X 10.6 seems to be adding 0x00 padding to the
616 * end of M1. Skip those to avoid interop issues.
619 for (i = 0; i < end - pos; i++) {
623 if (i == end - pos) {
624 wpa_printf(MSG_DEBUG, "WPS: Workaround - skip "
625 "unexpected message padding");
629 #endif /* WPS_WORKAROUNDS */
631 if (wps_set_attr(attr, type, pos, len) < 0)
634 #ifdef WPS_WORKAROUNDS
636 #endif /* WPS_WORKAROUNDS */