MBO: Add support to send ANQP request to get cellular preference

This extends ANQP_GET command to support querying MBO cellular
preference also. The cellular preference can be requested along with
neigbor report by appending mbo:1 to the command arguments.

For example:
ANQP_GET <bssid> 272,mbo:1

Signed-off-by: Jouni Malinen <jouni@qca.qualcomm.com>
This commit is contained in:
vamsi krishna 2016-09-09 17:35:50 +05:30 committed by Jouni Malinen
parent 5b71cb552b
commit 8f47917493
6 changed files with 73 additions and 9 deletions

View file

@ -1180,6 +1180,10 @@ enum wfa_wnm_notif_subelem_id {
WFA_WNM_NOTIF_SUBELEM_CELL_DATA_CAPA = 3, WFA_WNM_NOTIF_SUBELEM_CELL_DATA_CAPA = 3,
}; };
/* MBO v0.0_r25, 4.3: MBO ANQP-elements */
#define MBO_ANQP_OUI_TYPE 0x12
#define MBO_ANQP_SUBTYPE_CELL_CONN_PREF 1
/* Wi-Fi Direct (P2P) */ /* Wi-Fi Direct (P2P) */
#define P2P_OUI_TYPE 9 #define P2P_OUI_TYPE 9

View file

@ -6383,6 +6383,7 @@ static int get_anqp(struct wpa_supplicant *wpa_s, char *dst)
u16 id[MAX_ANQP_INFO_ID]; u16 id[MAX_ANQP_INFO_ID];
size_t num_id = 0; size_t num_id = 0;
u32 subtypes = 0; u32 subtypes = 0;
int get_cell_pref = 0;
used = hwaddr_aton2(dst, dst_addr); used = hwaddr_aton2(dst, dst_addr);
if (used < 0) if (used < 0)
@ -6400,6 +6401,15 @@ static int get_anqp(struct wpa_supplicant *wpa_s, char *dst)
#else /* CONFIG_HS20 */ #else /* CONFIG_HS20 */
return -1; return -1;
#endif /* CONFIG_HS20 */ #endif /* CONFIG_HS20 */
} else if (os_strncmp(pos, "mbo:", 4) == 0) {
#ifdef CONFIG_MBO
int num = atoi(pos + 4);
if (num != MBO_ANQP_SUBTYPE_CELL_CONN_PREF)
return -1;
get_cell_pref = 1;
#else /* CONFIG_MBO */
return -1;
#endif /* CONFIG_MBO */
} else { } else {
id[num_id] = atoi(pos); id[num_id] = atoi(pos);
if (id[num_id]) if (id[num_id])
@ -6414,7 +6424,8 @@ static int get_anqp(struct wpa_supplicant *wpa_s, char *dst)
if (num_id == 0) if (num_id == 0)
return -1; return -1;
return anqp_send_req(wpa_s, dst_addr, id, num_id, subtypes); return anqp_send_req(wpa_s, dst_addr, id, num_id, subtypes,
get_cell_pref);
} }

View file

@ -2692,10 +2692,11 @@ void interworking_stop_fetch_anqp(struct wpa_supplicant *wpa_s)
int anqp_send_req(struct wpa_supplicant *wpa_s, const u8 *dst, int anqp_send_req(struct wpa_supplicant *wpa_s, const u8 *dst,
u16 info_ids[], size_t num_ids, u32 subtypes) u16 info_ids[], size_t num_ids, u32 subtypes,
int get_cell_pref)
{ {
struct wpabuf *buf; struct wpabuf *buf;
struct wpabuf *hs20_buf = NULL; struct wpabuf *extra_buf = NULL;
int ret = 0; int ret = 0;
int freq; int freq;
struct wpa_bss *bss; struct wpa_bss *bss;
@ -2718,15 +2719,31 @@ int anqp_send_req(struct wpa_supplicant *wpa_s, const u8 *dst,
#ifdef CONFIG_HS20 #ifdef CONFIG_HS20
if (subtypes != 0) { if (subtypes != 0) {
hs20_buf = wpabuf_alloc(100); extra_buf = wpabuf_alloc(100);
if (hs20_buf == NULL) if (extra_buf == NULL)
return -1; return -1;
hs20_put_anqp_req(subtypes, NULL, 0, hs20_buf); hs20_put_anqp_req(subtypes, NULL, 0, extra_buf);
} }
#endif /* CONFIG_HS20 */ #endif /* CONFIG_HS20 */
buf = anqp_build_req(info_ids, num_ids, hs20_buf); #ifdef CONFIG_MBO
wpabuf_free(hs20_buf); if (get_cell_pref) {
struct wpabuf *mbo;
mbo = mbo_build_anqp_buf(wpa_s, bss);
if (mbo) {
if (wpabuf_resize(&extra_buf, wpabuf_len(mbo))) {
wpabuf_free(extra_buf);
return -1;
}
wpabuf_put_buf(extra_buf, mbo);
wpabuf_free(mbo);
}
}
#endif /* CONFIG_MBO */
buf = anqp_build_req(info_ids, num_ids, extra_buf);
wpabuf_free(extra_buf);
if (buf == NULL) if (buf == NULL)
return -1; return -1;

View file

@ -12,7 +12,8 @@
enum gas_query_result; enum gas_query_result;
int anqp_send_req(struct wpa_supplicant *wpa_s, const u8 *dst, int anqp_send_req(struct wpa_supplicant *wpa_s, const u8 *dst,
u16 info_ids[], size_t num_ids, u32 subtypes); u16 info_ids[], size_t num_ids, u32 subtypes,
int get_cell_pref);
void anqp_resp_cb(void *ctx, const u8 *dst, u8 dialog_token, void anqp_resp_cb(void *ctx, const u8 *dst, u8 dialog_token,
enum gas_query_result result, enum gas_query_result result,
const struct wpabuf *adv_proto, const struct wpabuf *adv_proto,

View file

@ -14,6 +14,7 @@
#include "utils/common.h" #include "utils/common.h"
#include "common/ieee802_11_defs.h" #include "common/ieee802_11_defs.h"
#include "common/gas.h"
#include "config.h" #include "config.h"
#include "wpa_supplicant_i.h" #include "wpa_supplicant_i.h"
#include "driver_i.h" #include "driver_i.h"
@ -805,3 +806,31 @@ void wpas_mbo_update_cell_capa(struct wpa_supplicant *wpa_s, u8 mbo_cell_capa)
wpas_mbo_send_wnm_notification(wpa_s, cell_capa, 7); wpas_mbo_send_wnm_notification(wpa_s, cell_capa, 7);
wpa_supplicant_set_default_scan_ies(wpa_s); wpa_supplicant_set_default_scan_ies(wpa_s);
} }
struct wpabuf * mbo_build_anqp_buf(struct wpa_supplicant *wpa_s,
struct wpa_bss *bss)
{
struct wpabuf *anqp_buf;
u8 *len_pos;
if (!wpa_bss_get_vendor_ie(bss, MBO_IE_VENDOR_TYPE)) {
wpa_printf(MSG_INFO, "MBO: " MACSTR
" does not support MBO - cannot request MBO ANQP elements from it",
MAC2STR(bss->bssid));
return NULL;
}
anqp_buf = wpabuf_alloc(10);
if (!anqp_buf)
return NULL;
len_pos = gas_anqp_add_element(anqp_buf, ANQP_VENDOR_SPECIFIC);
wpabuf_put_be24(anqp_buf, OUI_WFA);
wpabuf_put_u8(anqp_buf, MBO_ANQP_OUI_TYPE);
wpabuf_put_u8(anqp_buf, MBO_ANQP_SUBTYPE_CELL_CONN_PREF);
gas_anqp_set_element_len(anqp_buf, len_pos);
return anqp_buf;
}

View file

@ -1204,6 +1204,8 @@ size_t wpas_mbo_ie_bss_trans_reject(struct wpa_supplicant *wpa_s, u8 *pos,
size_t len, size_t len,
enum mbo_transition_reject_reason reason); enum mbo_transition_reject_reason reason);
void wpas_mbo_update_cell_capa(struct wpa_supplicant *wpa_s, u8 mbo_cell_capa); void wpas_mbo_update_cell_capa(struct wpa_supplicant *wpa_s, u8 mbo_cell_capa);
struct wpabuf * mbo_build_anqp_buf(struct wpa_supplicant *wpa_s,
struct wpa_bss *bss);
/** /**
* wpa_supplicant_ctrl_iface_ctrl_rsp_handle - Handle a control response * wpa_supplicant_ctrl_iface_ctrl_rsp_handle - Handle a control response