|
@@ -2510,8 +2510,68 @@ static void wpas_invitation_received(void *ctx, const u8 *sa, const u8 *bssid,
|
|
|
}
|
|
|
|
|
|
|
|
|
+static void wpas_remove_persistent_peer(struct wpa_supplicant *wpa_s,
|
|
|
+ struct wpa_ssid *ssid,
|
|
|
+ const u8 *peer)
|
|
|
+{
|
|
|
+ size_t i;
|
|
|
+
|
|
|
+ if (ssid == NULL)
|
|
|
+ return;
|
|
|
+
|
|
|
+ for (i = 0; ssid->p2p_client_list && i < ssid->num_p2p_clients; i++) {
|
|
|
+ if (os_memcmp(ssid->p2p_client_list + i * ETH_ALEN, peer,
|
|
|
+ ETH_ALEN) == 0)
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (i >= ssid->num_p2p_clients) {
|
|
|
+ if (ssid->mode != WPAS_MODE_P2P_GO &&
|
|
|
+ os_memcmp(ssid->bssid, peer, ETH_ALEN) == 0) {
|
|
|
+ wpa_printf(MSG_DEBUG, "P2P: Remove persistent group %d "
|
|
|
+ "due to invitation result", ssid->id);
|
|
|
+ wpas_notify_network_removed(wpa_s, ssid);
|
|
|
+ wpa_config_remove_network(wpa_s->conf, ssid->id);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ return; /* Peer not found in client list */
|
|
|
+ }
|
|
|
+
|
|
|
+ wpa_printf(MSG_DEBUG, "P2P: Remove peer " MACSTR " from persistent "
|
|
|
+ "group %d client list due to invitation result",
|
|
|
+ MAC2STR(peer), ssid->id);
|
|
|
+ os_memmove(ssid->p2p_client_list + i * ETH_ALEN,
|
|
|
+ ssid->p2p_client_list + (i + 1) * ETH_ALEN,
|
|
|
+ (ssid->num_p2p_clients - i - 1) * ETH_ALEN);
|
|
|
+ ssid->num_p2p_clients--;
|
|
|
+#ifndef CONFIG_NO_CONFIG_WRITE
|
|
|
+ if (wpa_s->parent->conf->update_config &&
|
|
|
+ wpa_config_write(wpa_s->parent->confname, wpa_s->parent->conf))
|
|
|
+ wpa_printf(MSG_DEBUG, "P2P: Failed to update configuration");
|
|
|
+#endif /* CONFIG_NO_CONFIG_WRITE */
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+static void wpas_remove_persistent_client(struct wpa_supplicant *wpa_s,
|
|
|
+ const u8 *peer)
|
|
|
+{
|
|
|
+ struct wpa_ssid *ssid;
|
|
|
+
|
|
|
+ wpa_s = wpa_s->global->p2p_invite_group;
|
|
|
+ if (wpa_s == NULL)
|
|
|
+ return; /* No known invitation group */
|
|
|
+ ssid = wpa_s->current_ssid;
|
|
|
+ if (ssid == NULL || ssid->mode != WPAS_MODE_P2P_GO ||
|
|
|
+ !ssid->p2p_persistent_group)
|
|
|
+ return; /* Not operating as a GO in persistent group */
|
|
|
+ ssid = wpas_p2p_get_persistent(wpa_s->parent, peer,
|
|
|
+ ssid->ssid, ssid->ssid_len);
|
|
|
+ wpas_remove_persistent_peer(wpa_s, ssid, peer);
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
static void wpas_invitation_result(void *ctx, int status, const u8 *bssid,
|
|
|
- const struct p2p_channels *channels)
|
|
|
+ const struct p2p_channels *channels,
|
|
|
+ const u8 *peer)
|
|
|
{
|
|
|
struct wpa_supplicant *wpa_s = ctx;
|
|
|
struct wpa_ssid *ssid;
|
|
@@ -2526,8 +2586,13 @@ static void wpas_invitation_result(void *ctx, int status, const u8 *bssid,
|
|
|
}
|
|
|
wpas_notify_p2p_invitation_result(wpa_s, status, bssid);
|
|
|
|
|
|
- if (wpa_s->pending_invite_ssid_id == -1)
|
|
|
+ wpa_printf(MSG_DEBUG, "P2P: Invitation result - status=%d peer=" MACSTR,
|
|
|
+ status, MAC2STR(peer));
|
|
|
+ if (wpa_s->pending_invite_ssid_id == -1) {
|
|
|
+ if (status == P2P_SC_FAIL_UNKNOWN_GROUP)
|
|
|
+ wpas_remove_persistent_client(wpa_s, peer);
|
|
|
return; /* Invitation to active group */
|
|
|
+ }
|
|
|
|
|
|
if (status == P2P_SC_FAIL_INFO_CURRENTLY_UNAVAILABLE) {
|
|
|
wpa_printf(MSG_DEBUG, "P2P: Waiting for peer to start another "
|
|
@@ -2536,6 +2601,11 @@ static void wpas_invitation_result(void *ctx, int status, const u8 *bssid,
|
|
|
}
|
|
|
|
|
|
if (status != P2P_SC_SUCCESS) {
|
|
|
+ if (status == P2P_SC_FAIL_UNKNOWN_GROUP) {
|
|
|
+ ssid = wpa_config_get_network(
|
|
|
+ wpa_s->conf, wpa_s->pending_invite_ssid_id);
|
|
|
+ wpas_remove_persistent_peer(wpa_s, ssid, peer);
|
|
|
+ }
|
|
|
wpas_p2p_remove_pending_group_interface(wpa_s);
|
|
|
return;
|
|
|
}
|
|
@@ -4687,6 +4757,7 @@ int wpas_p2p_invite(struct wpa_supplicant *wpa_s, const u8 *peer_addr,
|
|
|
int force_freq = 0, oper_freq = 0;
|
|
|
int res;
|
|
|
|
|
|
+ wpa_s->global->p2p_invite_group = NULL;
|
|
|
if (peer_addr)
|
|
|
os_memcpy(wpa_s->p2p_auth_invite, peer_addr, ETH_ALEN);
|
|
|
else
|
|
@@ -4779,6 +4850,7 @@ int wpas_p2p_invite_group(struct wpa_supplicant *wpa_s, const char *ifname,
|
|
|
return -1;
|
|
|
}
|
|
|
|
|
|
+ wpa_s->global->p2p_invite_group = wpa_s;
|
|
|
persistent = ssid->p2p_persistent_group &&
|
|
|
wpas_p2p_get_persistent(wpa_s->parent, peer_addr,
|
|
|
ssid->ssid, ssid->ssid_len);
|