Fix disconnection event processing

Commit 0d30cc240f forced
wpa_s->current_ssid and wpa_s->key_mgmt to be cleared in
wpa_supplicant_mark_disassoc() which gets called from
wpa_supplicant_event_disassoc(). This broke IEEE 802.1X authentication
failure processing and P2P deauthentication notification (group
termination).

Fix this by splitting wpa_supplicant_event_disassoc() into two parts and
make wpas_p2p_deauth_notif() indicate whether the interface was removed.
If so, the last part of disassocition event processing is skipped. Since
the wpa_supplicant_mark_disassoc() call is in the second part, the above
mentioned issues are resolved. In addition, this cleans up the P2P group
interface removal case by not trying to use fast reconnection mechanism
just before the interface gets removed.

Signed-hostap: Jouni Malinen <jouni@qca.qualcomm.com>
This commit is contained in:
Jouni Malinen 2012-08-31 22:04:41 +03:00 committed by Jouni Malinen
parent 6f1ca696c7
commit 0aadd5682f
3 changed files with 62 additions and 24 deletions

View file

@ -1768,6 +1768,35 @@ static void wpa_supplicant_event_disassoc(struct wpa_supplicant *wpa_s,
int locally_generated) int locally_generated)
{ {
const u8 *bssid; const u8 *bssid;
if (wpa_s->key_mgmt == WPA_KEY_MGMT_WPA_NONE) {
/*
* At least Host AP driver and a Prism3 card seemed to be
* generating streams of disconnected events when configuring
* IBSS for WPA-None. Ignore them for now.
*/
return;
}
bssid = wpa_s->bssid;
if (is_zero_ether_addr(bssid))
bssid = wpa_s->pending_bssid;
if (!is_zero_ether_addr(bssid) ||
wpa_s->wpa_state >= WPA_AUTHENTICATING) {
wpa_msg(wpa_s, MSG_INFO, WPA_EVENT_DISCONNECTED "bssid=" MACSTR
" reason=%d%s",
MAC2STR(bssid), reason_code,
locally_generated ? " locally_generated=1" : "");
}
}
static void wpa_supplicant_event_disassoc_finish(struct wpa_supplicant *wpa_s,
u16 reason_code,
int locally_generated)
{
const u8 *bssid;
int authenticating; int authenticating;
u8 prev_pending_bssid[ETH_ALEN]; u8 prev_pending_bssid[ETH_ALEN];
struct wpa_bss *fast_reconnect = NULL; struct wpa_bss *fast_reconnect = NULL;
@ -1835,13 +1864,6 @@ static void wpa_supplicant_event_disassoc(struct wpa_supplicant *wpa_s,
else else
wpa_s->disconnect_reason = reason_code; wpa_s->disconnect_reason = reason_code;
wpas_notify_disconnect_reason(wpa_s); wpas_notify_disconnect_reason(wpa_s);
if (!is_zero_ether_addr(bssid) ||
wpa_s->wpa_state >= WPA_AUTHENTICATING) {
wpa_msg(wpa_s, MSG_INFO, WPA_EVENT_DISCONNECTED "bssid=" MACSTR
" reason=%d%s",
MAC2STR(bssid), reason_code,
locally_generated ? " locally_generated=1" : "");
}
if (wpa_supplicant_dynamic_keys(wpa_s)) { if (wpa_supplicant_dynamic_keys(wpa_s)) {
wpa_dbg(wpa_s, MSG_DEBUG, "Disconnect event - remove keys"); wpa_dbg(wpa_s, MSG_DEBUG, "Disconnect event - remove keys");
wpa_s->keys_cleared = 0; wpa_s->keys_cleared = 0;
@ -2370,13 +2392,23 @@ void wpa_supplicant_event(void *ctx, enum wpa_event_type event,
wpas_auth_failed(wpa_s); wpas_auth_failed(wpa_s);
#ifdef CONFIG_P2P #ifdef CONFIG_P2P
if (event == EVENT_DEAUTH && data) { if (event == EVENT_DEAUTH && data) {
wpas_p2p_deauth_notif(wpa_s, data->deauth_info.addr, if (wpas_p2p_deauth_notif(wpa_s,
reason_code, data->deauth_info.addr,
data->deauth_info.ie, reason_code,
data->deauth_info.ie_len, data->deauth_info.ie,
locally_generated); data->deauth_info.ie_len,
locally_generated) > 0) {
/*
* The interface was removed, so cannot
* continue processing any additional
* operations after this.
*/
break;
}
} }
#endif /* CONFIG_P2P */ #endif /* CONFIG_P2P */
wpa_supplicant_event_disassoc_finish(wpa_s, reason_code,
locally_generated);
break; break;
case EVENT_MICHAEL_MIC_FAILURE: case EVENT_MICHAEL_MIC_FAILURE:
wpa_supplicant_event_michael_mic_failure(wpa_s, data); wpa_supplicant_event_michael_mic_failure(wpa_s, data);

View file

@ -333,7 +333,7 @@ static int wpas_p2p_group_delete(struct wpa_supplicant *wpa_s,
if (wpa_s && ifname) if (wpa_s && ifname)
wpa_drv_if_remove(wpa_s, type, ifname); wpa_drv_if_remove(wpa_s, type, ifname);
os_free(ifname); os_free(ifname);
return 0; return 1;
} }
wpa_printf(MSG_DEBUG, "P2P: Remove temporary group network"); wpa_printf(MSG_DEBUG, "P2P: Remove temporary group network");
@ -4625,14 +4625,15 @@ static void wpas_p2p_set_group_idle_timeout(struct wpa_supplicant *wpa_s)
} }
void wpas_p2p_deauth_notif(struct wpa_supplicant *wpa_s, const u8 *bssid, /* Returns 1 if the interface was removed */
u16 reason_code, const u8 *ie, size_t ie_len, int wpas_p2p_deauth_notif(struct wpa_supplicant *wpa_s, const u8 *bssid,
int locally_generated) u16 reason_code, const u8 *ie, size_t ie_len,
int locally_generated)
{ {
if (wpa_s->global->p2p_disabled || wpa_s->global->p2p == NULL) if (wpa_s->global->p2p_disabled || wpa_s->global->p2p == NULL)
return; return 0;
if (wpa_s->drv_flags & WPA_DRIVER_FLAGS_P2P_MGMT) if (wpa_s->drv_flags & WPA_DRIVER_FLAGS_P2P_MGMT)
return; return 0;
if (!locally_generated) if (!locally_generated)
p2p_deauth_notif(wpa_s->global->p2p, bssid, reason_code, ie, p2p_deauth_notif(wpa_s->global->p2p, bssid, reason_code, ie,
@ -4644,9 +4645,13 @@ void wpas_p2p_deauth_notif(struct wpa_supplicant *wpa_s, const u8 *bssid,
wpa_s->current_ssid->mode == WPAS_MODE_INFRA) { wpa_s->current_ssid->mode == WPAS_MODE_INFRA) {
wpa_printf(MSG_DEBUG, "P2P: GO indicated that the P2P Group " wpa_printf(MSG_DEBUG, "P2P: GO indicated that the P2P Group "
"session is ending"); "session is ending");
wpas_p2p_group_delete(wpa_s, if (wpas_p2p_group_delete(wpa_s,
P2P_GROUP_REMOVAL_GO_ENDING_SESSION); P2P_GROUP_REMOVAL_GO_ENDING_SESSION)
> 0)
return 1;
} }
return 0;
} }
@ -5081,7 +5086,8 @@ int wpas_p2p_disconnect(struct wpa_supplicant *wpa_s)
if (wpa_s == NULL) if (wpa_s == NULL)
return -1; return -1;
return wpas_p2p_group_delete(wpa_s, P2P_GROUP_REMOVAL_REQUESTED); return wpas_p2p_group_delete(wpa_s, P2P_GROUP_REMOVAL_REQUESTED) < 0 ?
-1 : 0;
} }

View file

@ -113,9 +113,9 @@ int wpas_p2p_presence_req(struct wpa_supplicant *wpa_s, u32 duration1,
u32 interval1, u32 duration2, u32 interval2); u32 interval1, u32 duration2, u32 interval2);
int wpas_p2p_ext_listen(struct wpa_supplicant *wpa_s, unsigned int period, int wpas_p2p_ext_listen(struct wpa_supplicant *wpa_s, unsigned int period,
unsigned int interval); unsigned int interval);
void wpas_p2p_deauth_notif(struct wpa_supplicant *wpa_s, const u8 *bssid, int wpas_p2p_deauth_notif(struct wpa_supplicant *wpa_s, const u8 *bssid,
u16 reason_code, const u8 *ie, size_t ie_len, u16 reason_code, const u8 *ie, size_t ie_len,
int locally_generated); int locally_generated);
void wpas_p2p_disassoc_notif(struct wpa_supplicant *wpa_s, const u8 *bssid, void wpas_p2p_disassoc_notif(struct wpa_supplicant *wpa_s, const u8 *bssid,
u16 reason_code, const u8 *ie, size_t ie_len, u16 reason_code, const u8 *ie, size_t ie_len,
int locally_generated); int locally_generated);