Updated through tag hostap_2_5 from git://w1.fi/hostap.git
[mech_eap.git] / libeap / src / wps / wps_dev_attr.c
index 823c7ef..0d01211 100644 (file)
@@ -2,14 +2,8 @@
  * Wi-Fi Protected Setup - device attributes
  * Copyright (c) 2008, Jouni Malinen <j@w1.fi>
  *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- *
- * Alternatively, this software may be distributed under the terms of BSD
- * license.
- *
- * See README and COPYING for more details.
+ * This software may be distributed under the terms of the BSD license.
+ * See README for more details.
  */
 
 #include "includes.h"
@@ -91,8 +85,7 @@ int wps_build_model_number(struct wps_device_data *dev, struct wpabuf *msg)
 }
 
 
-static int wps_build_serial_number(struct wps_device_data *dev,
-                                  struct wpabuf *msg)
+int wps_build_serial_number(struct wps_device_data *dev, struct wpabuf *msg)
 {
        size_t len;
        wpa_printf(MSG_DEBUG, "WPS:  * Serial Number");
@@ -126,6 +119,42 @@ int wps_build_primary_dev_type(struct wps_device_data *dev, struct wpabuf *msg)
 }
 
 
+int wps_build_secondary_dev_type(struct wps_device_data *dev,
+                                 struct wpabuf *msg)
+{
+       if (!dev->num_sec_dev_types)
+               return 0;
+
+       wpa_printf(MSG_DEBUG, "WPS:  * Secondary Device Type");
+       wpabuf_put_be16(msg, ATTR_SECONDARY_DEV_TYPE_LIST);
+       wpabuf_put_be16(msg, WPS_DEV_TYPE_LEN * dev->num_sec_dev_types);
+       wpabuf_put_data(msg, dev->sec_dev_type,
+                       WPS_DEV_TYPE_LEN * dev->num_sec_dev_types);
+
+       return 0;
+}
+
+
+int wps_build_req_dev_type(struct wps_device_data *dev, struct wpabuf *msg,
+                          unsigned int num_req_dev_types,
+                          const u8 *req_dev_types)
+{
+       unsigned int i;
+
+       for (i = 0; i < num_req_dev_types; i++) {
+               wpa_hexdump(MSG_DEBUG, "WPS: * Requested Device Type",
+                           req_dev_types + i * WPS_DEV_TYPE_LEN,
+                           WPS_DEV_TYPE_LEN);
+               wpabuf_put_be16(msg, ATTR_REQUESTED_DEV_TYPE);
+               wpabuf_put_be16(msg, WPS_DEV_TYPE_LEN);
+               wpabuf_put_data(msg, req_dev_types + i * WPS_DEV_TYPE_LEN,
+                               WPS_DEV_TYPE_LEN);
+       }
+
+       return 0;
+}
+
+
 int wps_build_dev_name(struct wps_device_data *dev, struct wpabuf *msg)
 {
        size_t len;
@@ -173,12 +202,42 @@ int wps_build_os_version(struct wps_device_data *dev, struct wpabuf *msg)
 }
 
 
-int wps_build_rf_bands(struct wps_device_data *dev, struct wpabuf *msg)
+int wps_build_vendor_ext_m1(struct wps_device_data *dev, struct wpabuf *msg)
 {
-       wpa_printf(MSG_DEBUG, "WPS:  * RF Bands (%x)", dev->rf_bands);
-       wpabuf_put_be16(msg, ATTR_RF_BANDS);
-       wpabuf_put_be16(msg, 1);
-       wpabuf_put_u8(msg, dev->rf_bands);
+       if (dev->vendor_ext_m1 != NULL) {
+               wpa_hexdump(MSG_DEBUG, "WPS:  * Vendor Extension M1",
+                           wpabuf_head_u8(dev->vendor_ext_m1),
+                           wpabuf_len(dev->vendor_ext_m1));
+               wpabuf_put_be16(msg, ATTR_VENDOR_EXT);
+               wpabuf_put_be16(msg, wpabuf_len(dev->vendor_ext_m1));
+               wpabuf_put_buf(msg, dev->vendor_ext_m1);
+       }
+       return 0;
+}
+
+
+int wps_build_rf_bands(struct wps_device_data *dev, struct wpabuf *msg,
+                      u8 rf_band)
+{
+       return wps_build_rf_bands_attr(msg, rf_band ? rf_band : dev->rf_bands);
+}
+
+
+int wps_build_vendor_ext(struct wps_device_data *dev, struct wpabuf *msg)
+{
+       int i;
+
+       for (i = 0; i < MAX_WPS_VENDOR_EXTENSIONS; i++) {
+               if (dev->vendor_ext[i] == NULL)
+                       continue;
+               wpa_hexdump(MSG_DEBUG, "WPS:  * Vendor Extension",
+                           wpabuf_head_u8(dev->vendor_ext[i]),
+                           wpabuf_len(dev->vendor_ext[i]));
+               wpabuf_put_be16(msg, ATTR_VENDOR_EXT);
+               wpabuf_put_be16(msg, wpabuf_len(dev->vendor_ext[i]));
+               wpabuf_put_buf(msg, dev->vendor_ext[i]);
+       }
+
        return 0;
 }
 
@@ -194,11 +253,9 @@ static int wps_process_manufacturer(struct wps_device_data *dev, const u8 *str,
        wpa_hexdump_ascii(MSG_DEBUG, "WPS: Manufacturer", str, str_len);
 
        os_free(dev->manufacturer);
-       dev->manufacturer = os_malloc(str_len + 1);
+       dev->manufacturer = dup_binstr(str, str_len);
        if (dev->manufacturer == NULL)
                return -1;
-       os_memcpy(dev->manufacturer, str, str_len);
-       dev->manufacturer[str_len] = '\0';
 
        return 0;
 }
@@ -215,11 +272,9 @@ static int wps_process_model_name(struct wps_device_data *dev, const u8 *str,
        wpa_hexdump_ascii(MSG_DEBUG, "WPS: Model Name", str, str_len);
 
        os_free(dev->model_name);
-       dev->model_name = os_malloc(str_len + 1);
+       dev->model_name = dup_binstr(str, str_len);
        if (dev->model_name == NULL)
                return -1;
-       os_memcpy(dev->model_name, str, str_len);
-       dev->model_name[str_len] = '\0';
 
        return 0;
 }
@@ -236,11 +291,9 @@ static int wps_process_model_number(struct wps_device_data *dev, const u8 *str,
        wpa_hexdump_ascii(MSG_DEBUG, "WPS: Model Number", str, str_len);
 
        os_free(dev->model_number);
-       dev->model_number = os_malloc(str_len + 1);
+       dev->model_number = dup_binstr(str, str_len);
        if (dev->model_number == NULL)
                return -1;
-       os_memcpy(dev->model_number, str, str_len);
-       dev->model_number[str_len] = '\0';
 
        return 0;
 }
@@ -257,11 +310,9 @@ static int wps_process_serial_number(struct wps_device_data *dev,
        wpa_hexdump_ascii(MSG_DEBUG, "WPS: Serial Number", str, str_len);
 
        os_free(dev->serial_number);
-       dev->serial_number = os_malloc(str_len + 1);
+       dev->serial_number = dup_binstr(str, str_len);
        if (dev->serial_number == NULL)
                return -1;
-       os_memcpy(dev->serial_number, str, str_len);
-       dev->serial_number[str_len] = '\0';
 
        return 0;
 }
@@ -278,11 +329,9 @@ static int wps_process_dev_name(struct wps_device_data *dev, const u8 *str,
        wpa_hexdump_ascii(MSG_DEBUG, "WPS: Device Name", str, str_len);
 
        os_free(dev->device_name);
-       dev->device_name = os_malloc(str_len + 1);
+       dev->device_name = dup_binstr(str, str_len);
        if (dev->device_name == NULL)
                return -1;
-       os_memcpy(dev->device_name, str, str_len);
-       dev->device_name[str_len] = '\0';
 
        return 0;
 }
@@ -355,25 +404,6 @@ int wps_process_rf_bands(struct wps_device_data *dev, const u8 *bands)
 }
 
 
-void wps_device_data_dup(struct wps_device_data *dst,
-                        const struct wps_device_data *src)
-{
-       if (src->device_name)
-               dst->device_name = os_strdup(src->device_name);
-       if (src->manufacturer)
-               dst->manufacturer = os_strdup(src->manufacturer);
-       if (src->model_name)
-               dst->model_name = os_strdup(src->model_name);
-       if (src->model_number)
-               dst->model_number = os_strdup(src->model_number);
-       if (src->serial_number)
-               dst->serial_number = os_strdup(src->serial_number);
-       os_memcpy(dst->pri_dev_type, src->pri_dev_type, WPS_DEV_TYPE_LEN);
-       dst->os_version = src->os_version;
-       dst->rf_bands = src->rf_bands;
-}
-
-
 void wps_device_data_free(struct wps_device_data *dev)
 {
        os_free(dev->device_name);