P2P: Fix interface deinit for failed group interface initialization
wpa_supplicant_deinit_iface() ends up removing all P2P groups if the removed interface is the parent interface. This is correct behavior in general, but this resulted in issues in the new group interface initialization error path since wpa_s->parent was not assigned before hitting this check. Fix this by assigning wpa_s->parent as part of wpa_supplicant_add_iface(). Signed-off-by: Jouni Malinen <j@w1.fi>
This commit is contained in:
		
							parent
							
								
									3f9ebc439c
								
							
						
					
					
						commit
						1772d348ea
					
				
					 11 changed files with 19 additions and 18 deletions
				
			
		|  | @ -8548,7 +8548,7 @@ static int wpa_supplicant_global_iface_add(struct wpa_global *global, | ||||||
| 	if (wpa_supplicant_get_iface(global, iface.ifname)) | 	if (wpa_supplicant_get_iface(global, iface.ifname)) | ||||||
| 		return -1; | 		return -1; | ||||||
| 
 | 
 | ||||||
| 	return wpa_supplicant_add_iface(global, &iface) ? 0 : -1; | 	return wpa_supplicant_add_iface(global, &iface, NULL) ? 0 : -1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -599,7 +599,7 @@ DBusMessage * wpas_dbus_handler_create_interface(DBusMessage *message, | ||||||
| 		iface.confname = confname; | 		iface.confname = confname; | ||||||
| 		iface.bridge_ifname = bridge_ifname; | 		iface.bridge_ifname = bridge_ifname; | ||||||
| 		/* Otherwise, have wpa_supplicant attach to it. */ | 		/* Otherwise, have wpa_supplicant attach to it. */ | ||||||
| 		wpa_s = wpa_supplicant_add_iface(global, &iface); | 		wpa_s = wpa_supplicant_add_iface(global, &iface, NULL); | ||||||
| 		if (wpa_s) { | 		if (wpa_s) { | ||||||
| 			const char *path = wpa_s->dbus_new_path; | 			const char *path = wpa_s->dbus_new_path; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -165,7 +165,7 @@ DBusMessage * wpas_dbus_global_add_interface(DBusMessage *message, | ||||||
| 		iface.confname = confname; | 		iface.confname = confname; | ||||||
| 		iface.bridge_ifname = bridge_ifname; | 		iface.bridge_ifname = bridge_ifname; | ||||||
| 		/* Otherwise, have wpa_supplicant attach to it. */ | 		/* Otherwise, have wpa_supplicant attach to it. */ | ||||||
| 		wpa_s = wpa_supplicant_add_iface(global, &iface); | 		wpa_s = wpa_supplicant_add_iface(global, &iface, NULL); | ||||||
| 		if (wpa_s) { | 		if (wpa_s) { | ||||||
| 			const char *path = wpa_s->dbus_path; | 			const char *path = wpa_s->dbus_path; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -322,7 +322,7 @@ int main(int argc, char *argv[]) | ||||||
| 			exitcode = -1; | 			exitcode = -1; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
| 		wpa_s = wpa_supplicant_add_iface(global, &ifaces[i]); | 		wpa_s = wpa_supplicant_add_iface(global, &ifaces[i], NULL); | ||||||
| 		if (wpa_s == NULL) { | 		if (wpa_s == NULL) { | ||||||
| 			exitcode = -1; | 			exitcode = -1; | ||||||
| 			break; | 			break; | ||||||
|  |  | ||||||
|  | @ -28,7 +28,7 @@ int main(int argc, char *argv[]) | ||||||
| 	memset(&iface, 0, sizeof(iface)); | 	memset(&iface, 0, sizeof(iface)); | ||||||
| 	/* TODO: set interface parameters */ | 	/* TODO: set interface parameters */ | ||||||
| 
 | 
 | ||||||
| 	if (wpa_supplicant_add_iface(global, &iface) == NULL) | 	if (wpa_supplicant_add_iface(global, &iface, NULL) == NULL) | ||||||
| 		exitcode = -1; | 		exitcode = -1; | ||||||
| 
 | 
 | ||||||
| 	if (exitcode == 0) | 	if (exitcode == 0) | ||||||
|  |  | ||||||
|  | @ -61,7 +61,7 @@ int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, | ||||||
| 			exitcode = -1; | 			exitcode = -1; | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
| 		if (wpa_supplicant_add_iface(global, &ifaces[i]) == NULL) | 		if (wpa_supplicant_add_iface(global, &ifaces[i], NULL) == NULL) | ||||||
| 			exitcode = -1; | 			exitcode = -1; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -119,7 +119,7 @@ static int read_interface(struct wpa_global *global, HKEY _hk, | ||||||
| 
 | 
 | ||||||
| 	RegCloseKey(hk); | 	RegCloseKey(hk); | ||||||
| 
 | 
 | ||||||
| 	if (wpa_supplicant_add_iface(global, &iface) == NULL) { | 	if (wpa_supplicant_add_iface(global, &iface, NULL) == NULL) { | ||||||
| 		if (skip_on_error) | 		if (skip_on_error) | ||||||
| 			wpa_printf(MSG_DEBUG, "Skipped interface '%s' due to " | 			wpa_printf(MSG_DEBUG, "Skipped interface '%s' due to " | ||||||
| 				   "initialization failure", iface.ifname); | 				   "initialization failure", iface.ifname); | ||||||
|  |  | ||||||
|  | @ -527,7 +527,7 @@ int wpas_mesh_add_interface(struct wpa_supplicant *wpa_s, char *ifname, | ||||||
| 	iface.driver_param = wpa_s->conf->driver_param; | 	iface.driver_param = wpa_s->conf->driver_param; | ||||||
| 	iface.ctrl_interface = wpa_s->conf->ctrl_interface; | 	iface.ctrl_interface = wpa_s->conf->ctrl_interface; | ||||||
| 
 | 
 | ||||||
| 	mesh_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface); | 	mesh_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface, wpa_s); | ||||||
| 	if (!mesh_wpa_s) { | 	if (!mesh_wpa_s) { | ||||||
| 		wpa_printf(MSG_ERROR, | 		wpa_printf(MSG_ERROR, | ||||||
| 			   "mesh: Failed to create new wpa_supplicant interface"); | 			   "mesh: Failed to create new wpa_supplicant interface"); | ||||||
|  | @ -535,6 +535,5 @@ int wpas_mesh_add_interface(struct wpa_supplicant *wpa_s, char *ifname, | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
| 	mesh_wpa_s->mesh_if_created = 1; | 	mesh_wpa_s->mesh_if_created = 1; | ||||||
| 	mesh_wpa_s->parent = wpa_s; |  | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1990,14 +1990,13 @@ wpas_p2p_init_group_interface(struct wpa_supplicant *wpa_s, int go) | ||||||
| 	else | 	else | ||||||
| 		iface.ctrl_interface = wpa_s->conf->ctrl_interface; | 		iface.ctrl_interface = wpa_s->conf->ctrl_interface; | ||||||
| 	iface.driver_param = wpa_s->conf->driver_param; | 	iface.driver_param = wpa_s->conf->driver_param; | ||||||
| 	group_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface); | 	group_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface, wpa_s); | ||||||
| 	if (group_wpa_s == NULL) { | 	if (group_wpa_s == NULL) { | ||||||
| 		wpa_printf(MSG_ERROR, "P2P: Failed to create new " | 		wpa_printf(MSG_ERROR, "P2P: Failed to create new " | ||||||
| 			   "wpa_supplicant interface"); | 			   "wpa_supplicant interface"); | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} | 	} | ||||||
| 	wpa_s->pending_interface_name[0] = '\0'; | 	wpa_s->pending_interface_name[0] = '\0'; | ||||||
| 	group_wpa_s->parent = wpa_s; |  | ||||||
| 	group_wpa_s->p2p_group_interface = go ? P2P_GROUP_INTERFACE_GO : | 	group_wpa_s->p2p_group_interface = go ? P2P_GROUP_INTERFACE_GO : | ||||||
| 		P2P_GROUP_INTERFACE_CLIENT; | 		P2P_GROUP_INTERFACE_CLIENT; | ||||||
| 	wpa_s->global->p2p_group_formation = group_wpa_s; | 	wpa_s->global->p2p_group_formation = group_wpa_s; | ||||||
|  | @ -4669,12 +4668,11 @@ int wpas_p2p_add_p2pdev_interface(struct wpa_supplicant *wpa_s, | ||||||
| 	} | 	} | ||||||
| 	iface.conf_p2p_dev = NULL; | 	iface.conf_p2p_dev = NULL; | ||||||
| 
 | 
 | ||||||
| 	p2pdev_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface); | 	p2pdev_wpa_s = wpa_supplicant_add_iface(wpa_s->global, &iface, wpa_s); | ||||||
| 	if (!p2pdev_wpa_s) { | 	if (!p2pdev_wpa_s) { | ||||||
| 		wpa_printf(MSG_DEBUG, "P2P: Failed to add P2P Device interface"); | 		wpa_printf(MSG_DEBUG, "P2P: Failed to add P2P Device interface"); | ||||||
| 		return -1; | 		return -1; | ||||||
| 	} | 	} | ||||||
| 	p2pdev_wpa_s->parent = wpa_s; |  | ||||||
| 	wpa_s->p2p_dev = p2pdev_wpa_s; | 	wpa_s->p2p_dev = p2pdev_wpa_s; | ||||||
| 
 | 
 | ||||||
| 	wpa_s->pending_interface_name[0] = '\0'; | 	wpa_s->pending_interface_name[0] = '\0'; | ||||||
|  |  | ||||||
|  | @ -3230,7 +3230,8 @@ static int wpa_supplicant_daemon(const char *pid_file) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| static struct wpa_supplicant * wpa_supplicant_alloc(void) | static struct wpa_supplicant * | ||||||
|  | wpa_supplicant_alloc(struct wpa_supplicant *parent) | ||||||
| { | { | ||||||
| 	struct wpa_supplicant *wpa_s; | 	struct wpa_supplicant *wpa_s; | ||||||
| 
 | 
 | ||||||
|  | @ -3240,7 +3241,7 @@ static struct wpa_supplicant * wpa_supplicant_alloc(void) | ||||||
| 	wpa_s->scan_req = INITIAL_SCAN_REQ; | 	wpa_s->scan_req = INITIAL_SCAN_REQ; | ||||||
| 	wpa_s->scan_interval = 5; | 	wpa_s->scan_interval = 5; | ||||||
| 	wpa_s->new_connection = 1; | 	wpa_s->new_connection = 1; | ||||||
| 	wpa_s->parent = wpa_s; | 	wpa_s->parent = parent ? parent : wpa_s; | ||||||
| 	wpa_s->sched_scanning = 0; | 	wpa_s->sched_scanning = 0; | ||||||
| 
 | 
 | ||||||
| 	return wpa_s; | 	return wpa_s; | ||||||
|  | @ -4294,6 +4295,7 @@ static void wpa_supplicant_deinit_iface(struct wpa_supplicant *wpa_s, | ||||||
|  * wpa_supplicant_add_iface - Add a new network interface |  * wpa_supplicant_add_iface - Add a new network interface | ||||||
|  * @global: Pointer to global data from wpa_supplicant_init() |  * @global: Pointer to global data from wpa_supplicant_init() | ||||||
|  * @iface: Interface configuration options |  * @iface: Interface configuration options | ||||||
|  |  * @parent: Parent interface or %NULL to assign new interface as parent | ||||||
|  * Returns: Pointer to the created interface or %NULL on failure |  * Returns: Pointer to the created interface or %NULL on failure | ||||||
|  * |  * | ||||||
|  * This function is used to add new network interfaces for %wpa_supplicant. |  * This function is used to add new network interfaces for %wpa_supplicant. | ||||||
|  | @ -4303,7 +4305,8 @@ static void wpa_supplicant_deinit_iface(struct wpa_supplicant *wpa_s, | ||||||
|  * e.g., when a hotplug network adapter is inserted. |  * e.g., when a hotplug network adapter is inserted. | ||||||
|  */ |  */ | ||||||
| struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, | struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, | ||||||
| 						 struct wpa_interface *iface) | 						 struct wpa_interface *iface, | ||||||
|  | 						 struct wpa_supplicant *parent) | ||||||
| { | { | ||||||
| 	struct wpa_supplicant *wpa_s; | 	struct wpa_supplicant *wpa_s; | ||||||
| 	struct wpa_interface t_iface; | 	struct wpa_interface t_iface; | ||||||
|  | @ -4312,7 +4315,7 @@ struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, | ||||||
| 	if (global == NULL || iface == NULL) | 	if (global == NULL || iface == NULL) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
| 	wpa_s = wpa_supplicant_alloc(); | 	wpa_s = wpa_supplicant_alloc(parent); | ||||||
| 	if (wpa_s == NULL) | 	if (wpa_s == NULL) | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1035,7 +1035,8 @@ void free_hw_features(struct wpa_supplicant *wpa_s); | ||||||
| void wpa_show_license(void); | void wpa_show_license(void); | ||||||
| 
 | 
 | ||||||
| struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, | struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, | ||||||
| 						 struct wpa_interface *iface); | 						 struct wpa_interface *iface, | ||||||
|  | 						 struct wpa_supplicant *parent); | ||||||
| int wpa_supplicant_remove_iface(struct wpa_global *global, | int wpa_supplicant_remove_iface(struct wpa_global *global, | ||||||
| 				struct wpa_supplicant *wpa_s, | 				struct wpa_supplicant *wpa_s, | ||||||
| 				int terminate); | 				int terminate); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Jouni Malinen
						Jouni Malinen