aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorJouni Malinen <jouni.malinen@atheros.com>2010-07-26 18:54:29 -0700
committerJouni Malinen <j@w1.fi>2010-09-09 07:17:21 -0700
commit5ded7d3eb8f168b7bb9b48ae2206f4ceeb8c8b18 (patch)
tree0afc7d0f503634ead32bd811d7d8a717ca9f5477 /src
parent3dfda83d9c69f521e4d4c1c6979be557c549d173 (diff)
downloadexternal_wpa_supplicant_8_ti-5ded7d3eb8f168b7bb9b48ae2206f4ceeb8c8b18.zip
external_wpa_supplicant_8_ti-5ded7d3eb8f168b7bb9b48ae2206f4ceeb8c8b18.tar.gz
external_wpa_supplicant_8_ti-5ded7d3eb8f168b7bb9b48ae2206f4ceeb8c8b18.tar.bz2
P2P: Fix GAS fragmentation to match with IEEE 802.11u
P2P specification v1.15 fixed the description of the GAS fragmentation to not duplicate NQP Query Response Field header in all fragments. This change makes the fragmentation match with the description in IEEE 802.11u. The change is not backwards compatible with previous P2P specification versions as far as fragmented SD responses are concerned.
Diffstat (limited to 'src')
-rw-r--r--src/p2p/p2p_i.h1
-rw-r--r--src/p2p/p2p_sd.c48
2 files changed, 33 insertions, 16 deletions
diff --git a/src/p2p/p2p_i.h b/src/p2p/p2p_i.h
index f349bc7..8225d64 100644
--- a/src/p2p/p2p_i.h
+++ b/src/p2p/p2p_i.h
@@ -322,6 +322,7 @@ struct p2p_data {
u8 sd_frag_id;
struct wpabuf *sd_rx_resp; /* Reassembled SD response */
+ u16 sd_rx_update_indic;
/* P2P Invitation data */
enum p2p_invite_role inv_role;
diff --git a/src/p2p/p2p_sd.c b/src/p2p/p2p_sd.c
index 2be6eea..873356a 100644
--- a/src/p2p/p2p_sd.c
+++ b/src/p2p/p2p_sd.c
@@ -210,10 +210,11 @@ static struct wpabuf * p2p_build_gas_comeback_resp(u8 dialog_token,
u16 status_code,
u16 update_indic,
const u8 *data, size_t len,
- u8 frag_id, u8 more)
+ u8 frag_id, u8 more,
+ u16 total_len)
{
struct wpabuf *buf;
- u8 *len_pos, *len_pos2;
+ u8 *len_pos;
buf = wpabuf_alloc(1000 + len);
if (buf == NULL)
@@ -235,17 +236,18 @@ static struct wpabuf * p2p_build_gas_comeback_resp(u8 dialog_token,
/* Query Response */
len_pos = wpabuf_put(buf, 2); /* Length (to be filled) */
- /* NQP Query Response Frame */
- wpabuf_put_le16(buf, NQP_VENDOR_SPECIFIC); /* Info ID */
- len_pos2 = wpabuf_put(buf, 2); /* Length (to be filled) */
- wpabuf_put_be24(buf, OUI_WFA);
- wpabuf_put_u8(buf, P2P_OUI_TYPE);
- /* Service Update Indicator */
- wpabuf_put_le16(buf, update_indic);
+ if (frag_id == 0) {
+ /* NQP Query Response Frame */
+ wpabuf_put_le16(buf, NQP_VENDOR_SPECIFIC); /* Info ID */
+ wpabuf_put_le16(buf, 3 + 1 + 2 + total_len);
+ wpabuf_put_be24(buf, OUI_WFA);
+ wpabuf_put_u8(buf, P2P_OUI_TYPE);
+ /* Service Update Indicator */
+ wpabuf_put_le16(buf, update_indic);
+ }
wpabuf_put_data(buf, data, len);
- WPA_PUT_LE16(len_pos2, (u8 *) wpabuf_put(buf, 0) - len_pos2 - 2);
WPA_PUT_LE16(len_pos, (u8 *) wpabuf_put(buf, 0) - len_pos - 2);
return buf;
@@ -667,7 +669,8 @@ void p2p_rx_gas_comeback_req(struct p2p_data *p2p, const u8 *sa,
p2p->srv_update_indic,
wpabuf_head_u8(p2p->sd_resp) +
p2p->sd_resp_pos, frag_len,
- p2p->sd_frag_id, more);
+ p2p->sd_frag_id, more,
+ wpabuf_len(p2p->sd_resp));
if (resp == NULL)
return;
wpa_msg(p2p->cfg->msg_ctx, MSG_DEBUG, "P2P: Send GAS Comeback "
@@ -711,7 +714,6 @@ void p2p_rx_gas_comeback_resp(struct p2p_data *p2p, const u8 *sa,
u8 more_frags;
u16 comeback_delay;
u16 slen;
- u16 update_indic;
wpa_hexdump(MSG_DEBUG, "P2P: RX GAS Comeback Response", data, len);
@@ -804,6 +806,14 @@ void p2p_rx_gas_comeback_resp(struct p2p_data *p2p, const u8 *sa,
}
end = pos + slen;
+ if (p2p->sd_rx_resp) {
+ /*
+ * NQP header is only included in the first fragment; rest of
+ * the fragments start with continue TLVs.
+ */
+ goto skip_nqp_header;
+ }
+
/* NQP Query Response */
if (pos + 4 > end)
return;
@@ -816,11 +826,15 @@ void p2p_rx_gas_comeback_resp(struct p2p_data *p2p, const u8 *sa,
slen = WPA_GET_LE16(pos);
pos += 2;
- if (pos + slen > end || slen < 3 + 1) {
+ wpa_msg(p2p->cfg->msg_ctx, MSG_DEBUG, "P2P: NQP Query Response "
+ "length: %u", slen);
+ if (slen < 3 + 1) {
wpa_msg(p2p->cfg->msg_ctx, MSG_DEBUG,
"P2P: Invalid NQP Query Response length");
return;
}
+ if (pos + 4 > end)
+ return;
if (WPA_GET_BE24(pos) != OUI_WFA) {
wpa_msg(p2p->cfg->msg_ctx, MSG_DEBUG,
@@ -838,11 +852,12 @@ void p2p_rx_gas_comeback_resp(struct p2p_data *p2p, const u8 *sa,
if (pos + 2 > end)
return;
- update_indic = WPA_GET_LE16(pos);
+ p2p->sd_rx_update_indic = WPA_GET_LE16(pos);
wpa_msg(p2p->cfg->msg_ctx, MSG_DEBUG,
- "P2P: Service Update Indicator: %u", update_indic);
+ "P2P: Service Update Indicator: %u", p2p->sd_rx_update_indic);
pos += 2;
+skip_nqp_header:
if (wpabuf_resize(&p2p->sd_rx_resp, end - pos) < 0)
return;
wpabuf_put_data(p2p->sd_rx_resp, pos, end - pos);
@@ -883,7 +898,8 @@ void p2p_rx_gas_comeback_resp(struct p2p_data *p2p, const u8 *sa,
}
if (p2p->cfg->sd_response)
- p2p->cfg->sd_response(p2p->cfg->cb_ctx, sa, update_indic,
+ p2p->cfg->sd_response(p2p->cfg->cb_ctx, sa,
+ p2p->sd_rx_update_indic,
wpabuf_head(p2p->sd_rx_resp),
wpabuf_len(p2p->sd_rx_resp));
wpabuf_free(p2p->sd_rx_resp);