|
@@ -1,6 +1,6 @@
|
|
/*
|
|
/*
|
|
* BSS table
|
|
* BSS table
|
|
- * Copyright (c) 2009-2010, Jouni Malinen <j@w1.fi>
|
|
|
|
|
|
+ * Copyright (c) 2009-2012, Jouni Malinen <j@w1.fi>
|
|
*
|
|
*
|
|
* This software may be distributed under the terms of the BSD license.
|
|
* This software may be distributed under the terms of the BSD license.
|
|
* See README for more details.
|
|
* See README for more details.
|
|
@@ -51,6 +51,14 @@ static void wpa_bss_set_hessid(struct wpa_bss *bss)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_anqp_alloc - Allocate ANQP data structure for a BSS entry
|
|
|
|
+ * Returns: Allocated ANQP data structure or %NULL on failure
|
|
|
|
+ *
|
|
|
|
+ * The allocated ANQP data structure has its users count set to 1. It may be
|
|
|
|
+ * shared by multiple BSS entries and each shared entry is freed with
|
|
|
|
+ * wpa_bss_anqp_free().
|
|
|
|
+ */
|
|
struct wpa_bss_anqp * wpa_bss_anqp_alloc(void)
|
|
struct wpa_bss_anqp * wpa_bss_anqp_alloc(void)
|
|
{
|
|
{
|
|
struct wpa_bss_anqp *anqp;
|
|
struct wpa_bss_anqp *anqp;
|
|
@@ -62,6 +70,11 @@ struct wpa_bss_anqp * wpa_bss_anqp_alloc(void)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_anqp_clone - Clone an ANQP data structure
|
|
|
|
+ * @anqp: ANQP data structure from wpa_bss_anqp_alloc()
|
|
|
|
+ * Returns: Cloned ANQP data structure or %NULL on failure
|
|
|
|
+ */
|
|
static struct wpa_bss_anqp * wpa_bss_anqp_clone(struct wpa_bss_anqp *anqp)
|
|
static struct wpa_bss_anqp * wpa_bss_anqp_clone(struct wpa_bss_anqp *anqp)
|
|
{
|
|
{
|
|
struct wpa_bss_anqp *n;
|
|
struct wpa_bss_anqp *n;
|
|
@@ -92,6 +105,14 @@ static struct wpa_bss_anqp * wpa_bss_anqp_clone(struct wpa_bss_anqp *anqp)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_anqp_unshare_alloc - Unshare ANQP data (if shared) in a BSS entry
|
|
|
|
+ * @bss: BSS entry
|
|
|
|
+ * Returns: 0 on success, -1 on failure
|
|
|
|
+ *
|
|
|
|
+ * This function ensures the specific BSS entry has an ANQP data structure that
|
|
|
|
+ * is not shared with any other BSS entry.
|
|
|
|
+ */
|
|
int wpa_bss_anqp_unshare_alloc(struct wpa_bss *bss)
|
|
int wpa_bss_anqp_unshare_alloc(struct wpa_bss *bss)
|
|
{
|
|
{
|
|
struct wpa_bss_anqp *anqp;
|
|
struct wpa_bss_anqp *anqp;
|
|
@@ -116,6 +137,10 @@ int wpa_bss_anqp_unshare_alloc(struct wpa_bss *bss)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_anqp_free - Free an ANQP data structure
|
|
|
|
+ * @anqp: ANQP data structure from wpa_bss_anqp_alloc() or wpa_bss_anqp_clone()
|
|
|
|
+ */
|
|
static void wpa_bss_anqp_free(struct wpa_bss_anqp *anqp)
|
|
static void wpa_bss_anqp_free(struct wpa_bss_anqp *anqp)
|
|
{
|
|
{
|
|
if (anqp == NULL)
|
|
if (anqp == NULL)
|
|
@@ -175,6 +200,14 @@ static void wpa_bss_remove(struct wpa_supplicant *wpa_s, struct wpa_bss *bss,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get - Fetch a BSS table entry based on BSSID and SSID
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @bssid: BSSID
|
|
|
|
+ * @ssid: SSID
|
|
|
|
+ * @ssid_len: Length of @ssid
|
|
|
|
+ * Returns: Pointer to the BSS entry or %NULL if not found
|
|
|
|
+ */
|
|
struct wpa_bss * wpa_bss_get(struct wpa_supplicant *wpa_s, const u8 *bssid,
|
|
struct wpa_bss * wpa_bss_get(struct wpa_supplicant *wpa_s, const u8 *bssid,
|
|
const u8 *ssid, size_t ssid_len)
|
|
const u8 *ssid, size_t ssid_len)
|
|
{
|
|
{
|
|
@@ -496,6 +529,15 @@ wpa_bss_update(struct wpa_supplicant *wpa_s, struct wpa_bss *bss,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_update_start - Start a BSS table update from scan results
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ *
|
|
|
|
+ * This function is called at the start of each BSS table update round for new
|
|
|
|
+ * scan results. The actual scan result entries are indicated with calls to
|
|
|
|
+ * wpa_bss_update_scan_res() and the update round is finished with a call to
|
|
|
|
+ * wpa_bss_update_end().
|
|
|
|
+ */
|
|
void wpa_bss_update_start(struct wpa_supplicant *wpa_s)
|
|
void wpa_bss_update_start(struct wpa_supplicant *wpa_s)
|
|
{
|
|
{
|
|
wpa_s->bss_update_idx++;
|
|
wpa_s->bss_update_idx++;
|
|
@@ -505,6 +547,15 @@ void wpa_bss_update_start(struct wpa_supplicant *wpa_s)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_update_scan_res - Update a BSS table entry based on a scan result
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @res: Scan result
|
|
|
|
+ *
|
|
|
|
+ * This function updates a BSS table entry (or adds one) based on a scan result.
|
|
|
|
+ * This is called separately for each scan result between the calls to
|
|
|
|
+ * wpa_bss_update_start() and wpa_bss_update_end().
|
|
|
|
+ */
|
|
void wpa_bss_update_scan_res(struct wpa_supplicant *wpa_s,
|
|
void wpa_bss_update_scan_res(struct wpa_supplicant *wpa_s,
|
|
struct wpa_scan_res *res)
|
|
struct wpa_scan_res *res)
|
|
{
|
|
{
|
|
@@ -610,6 +661,16 @@ static int wpa_bss_included_in_scan(const struct wpa_bss *bss,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_update_end - End a BSS table update from scan results
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @info: Information about scan parameters
|
|
|
|
+ * @new_scan: Whether this update round was based on a new scan
|
|
|
|
+ *
|
|
|
|
+ * This function is called at the end of each BSS table update round for new
|
|
|
|
+ * scan results. The start of the update was indicated with a call to
|
|
|
|
+ * wpa_bss_update_start().
|
|
|
|
+ */
|
|
void wpa_bss_update_end(struct wpa_supplicant *wpa_s, struct scan_info *info,
|
|
void wpa_bss_update_end(struct wpa_supplicant *wpa_s, struct scan_info *info,
|
|
int new_scan)
|
|
int new_scan)
|
|
{
|
|
{
|
|
@@ -655,6 +716,13 @@ void wpa_bss_update_end(struct wpa_supplicant *wpa_s, struct scan_info *info,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_flush_by_age - Flush old BSS entries
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @age: Maximum entry age in seconds
|
|
|
|
+ *
|
|
|
|
+ * Remove BSS entries that have not been updated during the last @age seconds.
|
|
|
|
+ */
|
|
void wpa_bss_flush_by_age(struct wpa_supplicant *wpa_s, int age)
|
|
void wpa_bss_flush_by_age(struct wpa_supplicant *wpa_s, int age)
|
|
{
|
|
{
|
|
struct wpa_bss *bss, *n;
|
|
struct wpa_bss *bss, *n;
|
|
@@ -688,6 +756,14 @@ static void wpa_bss_timeout(void *eloop_ctx, void *timeout_ctx)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_init - Initialize BSS table
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * Returns: 0 on success, -1 on failure
|
|
|
|
+ *
|
|
|
|
+ * This prepares BSS table lists and timer for periodic updates. The BSS table
|
|
|
|
+ * is deinitialized with wpa_bss_deinit() once not needed anymore.
|
|
|
|
+ */
|
|
int wpa_bss_init(struct wpa_supplicant *wpa_s)
|
|
int wpa_bss_init(struct wpa_supplicant *wpa_s)
|
|
{
|
|
{
|
|
dl_list_init(&wpa_s->bss);
|
|
dl_list_init(&wpa_s->bss);
|
|
@@ -698,6 +774,10 @@ int wpa_bss_init(struct wpa_supplicant *wpa_s)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_flush - Flush all unused BSS entries
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ */
|
|
void wpa_bss_flush(struct wpa_supplicant *wpa_s)
|
|
void wpa_bss_flush(struct wpa_supplicant *wpa_s)
|
|
{
|
|
{
|
|
struct wpa_bss *bss, *n;
|
|
struct wpa_bss *bss, *n;
|
|
@@ -713,6 +793,10 @@ void wpa_bss_flush(struct wpa_supplicant *wpa_s)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_deinit - Deinitialize BSS table
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ */
|
|
void wpa_bss_deinit(struct wpa_supplicant *wpa_s)
|
|
void wpa_bss_deinit(struct wpa_supplicant *wpa_s)
|
|
{
|
|
{
|
|
eloop_cancel_timeout(wpa_bss_timeout, wpa_s, NULL);
|
|
eloop_cancel_timeout(wpa_bss_timeout, wpa_s, NULL);
|
|
@@ -720,6 +804,12 @@ void wpa_bss_deinit(struct wpa_supplicant *wpa_s)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_bssid - Fetch a BSS table entry based on BSSID
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @bssid: BSSID
|
|
|
|
+ * Returns: Pointer to the BSS entry or %NULL if not found
|
|
|
|
+ */
|
|
struct wpa_bss * wpa_bss_get_bssid(struct wpa_supplicant *wpa_s,
|
|
struct wpa_bss * wpa_bss_get_bssid(struct wpa_supplicant *wpa_s,
|
|
const u8 *bssid)
|
|
const u8 *bssid)
|
|
{
|
|
{
|
|
@@ -735,6 +825,12 @@ struct wpa_bss * wpa_bss_get_bssid(struct wpa_supplicant *wpa_s,
|
|
|
|
|
|
|
|
|
|
#ifdef CONFIG_P2P
|
|
#ifdef CONFIG_P2P
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_p2p_dev_addr - Fetch a BSS table entry based on P2P Device Addr
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @dev_addr: P2P Device Address of the GO
|
|
|
|
+ * Returns: Pointer to the BSS entry or %NULL if not found
|
|
|
|
+ */
|
|
struct wpa_bss * wpa_bss_get_p2p_dev_addr(struct wpa_supplicant *wpa_s,
|
|
struct wpa_bss * wpa_bss_get_p2p_dev_addr(struct wpa_supplicant *wpa_s,
|
|
const u8 *dev_addr)
|
|
const u8 *dev_addr)
|
|
{
|
|
{
|
|
@@ -751,6 +847,12 @@ struct wpa_bss * wpa_bss_get_p2p_dev_addr(struct wpa_supplicant *wpa_s,
|
|
#endif /* CONFIG_P2P */
|
|
#endif /* CONFIG_P2P */
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_id - Fetch a BSS table entry based on identifier
|
|
|
|
+ * @wpa_s: Pointer to wpa_supplicant data
|
|
|
|
+ * @id: Unique identifier (struct wpa_bss::id) assigned for the entry
|
|
|
|
+ * Returns: Pointer to the BSS entry or %NULL if not found
|
|
|
|
+ */
|
|
struct wpa_bss * wpa_bss_get_id(struct wpa_supplicant *wpa_s, unsigned int id)
|
|
struct wpa_bss * wpa_bss_get_id(struct wpa_supplicant *wpa_s, unsigned int id)
|
|
{
|
|
{
|
|
struct wpa_bss *bss;
|
|
struct wpa_bss *bss;
|
|
@@ -762,6 +864,15 @@ struct wpa_bss * wpa_bss_get_id(struct wpa_supplicant *wpa_s, unsigned int id)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_ie - Fetch a specified information element from a BSS entry
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * @ie: Information element identitifier (WLAN_EID_*)
|
|
|
|
+ * Returns: Pointer to the information element (id field) or %NULL if not found
|
|
|
|
+ *
|
|
|
|
+ * This function returns the first matching information element in the BSS
|
|
|
|
+ * entry.
|
|
|
|
+ */
|
|
const u8 * wpa_bss_get_ie(const struct wpa_bss *bss, u8 ie)
|
|
const u8 * wpa_bss_get_ie(const struct wpa_bss *bss, u8 ie)
|
|
{
|
|
{
|
|
const u8 *end, *pos;
|
|
const u8 *end, *pos;
|
|
@@ -781,6 +892,15 @@ const u8 * wpa_bss_get_ie(const struct wpa_bss *bss, u8 ie)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_vendor_ie - Fetch a vendor information element from a BSS entry
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * @vendor_type: Vendor type (four octets starting the IE payload)
|
|
|
|
+ * Returns: Pointer to the information element (id field) or %NULL if not found
|
|
|
|
+ *
|
|
|
|
+ * This function returns the first matching information element in the BSS
|
|
|
|
+ * entry.
|
|
|
|
+ */
|
|
const u8 * wpa_bss_get_vendor_ie(const struct wpa_bss *bss, u32 vendor_type)
|
|
const u8 * wpa_bss_get_vendor_ie(const struct wpa_bss *bss, u32 vendor_type)
|
|
{
|
|
{
|
|
const u8 *end, *pos;
|
|
const u8 *end, *pos;
|
|
@@ -801,6 +921,16 @@ const u8 * wpa_bss_get_vendor_ie(const struct wpa_bss *bss, u32 vendor_type)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_vendor_ie_multi - Fetch vendor IE data from a BSS entry
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * @vendor_type: Vendor type (four octets starting the IE payload)
|
|
|
|
+ * Returns: Pointer to the information element payload or %NULL if not found
|
|
|
|
+ *
|
|
|
|
+ * This function returns concatenated payload of possibly fragmented vendor
|
|
|
|
+ * specific information elements in the BSS entry. The caller is responsible for
|
|
|
|
+ * freeing the returned buffer.
|
|
|
|
+ */
|
|
struct wpabuf * wpa_bss_get_vendor_ie_multi(const struct wpa_bss *bss,
|
|
struct wpabuf * wpa_bss_get_vendor_ie_multi(const struct wpa_bss *bss,
|
|
u32 vendor_type)
|
|
u32 vendor_type)
|
|
{
|
|
{
|
|
@@ -832,6 +962,19 @@ struct wpabuf * wpa_bss_get_vendor_ie_multi(const struct wpa_bss *bss,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_vendor_ie_multi_beacon - Fetch vendor IE data from a BSS entry
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * @vendor_type: Vendor type (four octets starting the IE payload)
|
|
|
|
+ * Returns: Pointer to the information element payload or %NULL if not found
|
|
|
|
+ *
|
|
|
|
+ * This function returns concatenated payload of possibly fragmented vendor
|
|
|
|
+ * specific information elements in the BSS entry. The caller is responsible for
|
|
|
|
+ * freeing the returned buffer.
|
|
|
|
+ *
|
|
|
|
+ * This function is like wpa_bss_get_vendor_ie_multi(), but uses IE buffer only
|
|
|
|
+ * from Beacon frames instead of either Beacon or Probe Response frames.
|
|
|
|
+ */
|
|
struct wpabuf * wpa_bss_get_vendor_ie_multi_beacon(const struct wpa_bss *bss,
|
|
struct wpabuf * wpa_bss_get_vendor_ie_multi_beacon(const struct wpa_bss *bss,
|
|
u32 vendor_type)
|
|
u32 vendor_type)
|
|
{
|
|
{
|
|
@@ -864,6 +1007,11 @@ struct wpabuf * wpa_bss_get_vendor_ie_multi_beacon(const struct wpa_bss *bss,
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_max_rate - Get maximum legacy TX rate supported in a BSS
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * Returns: Maximum legacy rate in units of 500 kbps
|
|
|
|
+ */
|
|
int wpa_bss_get_max_rate(const struct wpa_bss *bss)
|
|
int wpa_bss_get_max_rate(const struct wpa_bss *bss)
|
|
{
|
|
{
|
|
int rate = 0;
|
|
int rate = 0;
|
|
@@ -886,6 +1034,15 @@ int wpa_bss_get_max_rate(const struct wpa_bss *bss)
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
+/**
|
|
|
|
+ * wpa_bss_get_bit_rates - Get legacy TX rates supported in a BSS
|
|
|
|
+ * @bss: BSS table entry
|
|
|
|
+ * @rates: Buffer for returning a pointer to the rates list (units of 500 kbps)
|
|
|
|
+ * Returns: number of legacy TX rates or -1 on failure
|
|
|
|
+ *
|
|
|
|
+ * The caller is responsible for freeing the returned buffer with os_free() in
|
|
|
|
+ * case of success.
|
|
|
|
+ */
|
|
int wpa_bss_get_bit_rates(const struct wpa_bss *bss, u8 **rates)
|
|
int wpa_bss_get_bit_rates(const struct wpa_bss *bss, u8 **rates)
|
|
{
|
|
{
|
|
const u8 *ie, *ie2;
|
|
const u8 *ie, *ie2;
|