On 17/02/2016 16:58, Roy Marples wrote: > Add interface matching support with -M, guarded by CONFIG_MATCH_IFACE. > > This is very useful for systems without udev (Linux) or devd (FreeBSD). > > This patch requires a prior patch I submitted with the subject line: > [PATCH] Find correct driver for interface additions/removals > which can be found here: > http://lists.infradead.org/pipermail/hostap/2016-February/034778.html The above patch for finding the correct driver has been re-worked and was submitted here: http://lists.infradead.org/pipermail/hostap/2016-March/035385.html New patch for interface matching attached as a result. Roy
commit db7b1aca86220f6caae0e1ba4bf1f549cbc7e86f Author: Roy Marples <roy@xxxxxxxxxxxx> Date: Tue Mar 15 13:34:36 2016 +0000 Add interface matching support with -M, guarded by CONFIG_MATCH_IFACE. Signed-off-by: Roy Marples <roy@xxxxxxxxxxxx> diff --git a/wpa_supplicant/Makefile b/wpa_supplicant/Makefile index 8fa2b5a..0031899 100644 --- a/wpa_supplicant/Makefile +++ b/wpa_supplicant/Makefile @@ -315,6 +315,10 @@ CFLAGS += -DCONFIG_IBSS_RSN OBJS += ibss_rsn.o endif +ifdef CONFIG_MATCH_IFACE +CFLAGS += -DCONFIG_MATCH_IFACE +endif + ifdef CONFIG_P2P OBJS += p2p_supplicant.o OBJS += p2p_supplicant_sd.o diff --git a/wpa_supplicant/README b/wpa_supplicant/README index fefc0d3..cb031b5 100644 --- a/wpa_supplicant/README +++ b/wpa_supplicant/README @@ -410,7 +410,7 @@ usage: wpa_supplicant [-BddfhKLqqtuvW] [-P<pid file>] [-g<global ctrl>] \ [-G<group>] \ -i<ifname> -c<config file> [-C<ctrl>] [-D<driver>] [-p<driver_param>] \ - [-b<br_ifname> [-N -i<ifname> -c<conf> [-C<ctrl>] [-D<driver>] \ + [-b<br_ifname> [-MN -i<ifname> -c<conf> [-C<ctrl>] [-D<driver>] \ [-p<driver_param>] [-b<br_ifname>] [-m<P2P Device config file>] ... options: @@ -434,6 +434,7 @@ options: -u = enable DBus control interface -v = show version -W = wait for a control interface monitor before starting + -M = start describing matching interface -N = start describing new interface -m = Configuration file for the P2P Device @@ -476,6 +477,21 @@ wpa_supplicant \ -c wpa2.conf -i wlan1 -D wext +If the interfaces on which wpa_supplicant is to run are not known or +do not exist, wpa_supplicant can match an interface when it arrives. +Each matched interface is separated with -M argument and the +-i argument now allows for pattern matching. +As an example, the following command would start wpa_supplicant for a +specific wired interface called lan, any interface starting with wlan +and lastly any other interface. Each match has it's own configuration +file, and for the wired interface a specific driver has also been given. + +wpa_supplicant \ + -M -c wpa_wired.conf -ilan0 -D wired \ + -M -c wpa1.conf -iwlan* \ + -M -c wpa2.conf + + If the interface is added in a Linux bridge (e.g., br0), the bridge interface needs to be configured to wpa_supplicant in addition to the main interface: diff --git a/wpa_supplicant/defconfig b/wpa_supplicant/defconfig index 79632e6..1d05198 100644 --- a/wpa_supplicant/defconfig +++ b/wpa_supplicant/defconfig @@ -467,6 +467,9 @@ CONFIG_PEERKEY=y # Hotspot 2.0 #CONFIG_HS20=y +# Enable interface matching in wpa_supplicant +#CONFIG_MATCH_IFACE=y + # Disable roaming in wpa_supplicant #CONFIG_NO_ROAMING=y diff --git a/wpa_supplicant/events.c b/wpa_supplicant/events.c index f4cdfba..833c8ca 100644 --- a/wpa_supplicant/events.c +++ b/wpa_supplicant/events.c @@ -2756,6 +2756,13 @@ wpa_supplicant_event_interface_status(struct wpa_supplicant *wpa_s, } #endif /* CONFIG_P2P */ +#ifdef CONFIG_MATCH_IFACE + if (wpa_s->matched) { + wpa_supplicant_remove_iface(wpa_s->global, wpa_s, 0); + break; + } +#endif /* CONFIG_MATCH_IFACE */ + #ifdef CONFIG_TERMINATE_ONLASTIF /* check if last interface */ if (!any_interfaces(wpa_s->global->ifaces)) @@ -4029,6 +4036,20 @@ wpa_supplicant_event_global(void *ctx, enum wpa_event_type event, return; } } +#ifdef CONFIG_MATCH_IFACE + else if (data->interface_status.ievent == EVENT_INTERFACE_ADDED) { + struct wpa_interface *wpa_i; + + wpa_i = wpa_supplicant_match_iface(ctx, + data->interface_status.ifname); + if (wpa_i == NULL) + return; + wpa_s = wpa_supplicant_add_iface(ctx, wpa_i, NULL); + os_free(wpa_i); + if (wpa_s != NULL) + wpa_s->matched = 1; + } +#endif /* CONFIG_MATCH_IFACE */ if (wpa_s != NULL) wpa_supplicant_event(wpa_s, event, data); diff --git a/wpa_supplicant/main.c b/wpa_supplicant/main.c index 3ab80a5..289ed4f 100644 --- a/wpa_supplicant/main.c +++ b/wpa_supplicant/main.c @@ -81,6 +81,9 @@ static void usage(void) #ifdef CONFIG_P2P " -m = Configuration file for the P2P Device interface\n" #endif /* CONFIG_P2P */ +#ifdef CONFIG_MATCH_IFACE + " -M = start describing new matching interface\n" +#endif /* CONFIG_MATCH_IFACE */ " -N = start describing new interface\n" " -o = override driver parameter for new interfaces\n" " -O = override ctrl_interface parameter for new interfaces\n" @@ -153,6 +156,25 @@ static void wpa_supplicant_fd_workaround(int start) } +#ifdef CONFIG_MATCH_IFACE +static int wpa_supplicant_init_match(struct wpa_global *global) +{ + + /* The assumption is that the first driver is the primary driver + * and will handle the arrival / deperature of interfaces. */ + if (wpa_drivers[0]->global_init && global->drv_priv[0] == NULL) { + global->drv_priv[0] = wpa_drivers[0]->global_init(global); + if (global->drv_priv[0] == NULL) { + wpa_printf(MSG_ERROR, "Failed to initialize driver " + "'%s'", wpa_drivers[0]->name); + return -1; + } + } + return 0; +} +#endif /* CONFIG_MATCH_IFACE */ + + int main(int argc, char *argv[]) { int c, i; @@ -176,7 +198,7 @@ int main(int argc, char *argv[]) for (;;) { c = getopt(argc, argv, - "b:Bc:C:D:de:f:g:G:hi:I:KLm:No:O:p:P:qsTtuvW"); + "b:Bc:C:D:de:f:g:G:hi:I:KLMm:No:O:p:P:qsTtuvW"); if (c < 0) break; switch (c) { @@ -282,6 +304,19 @@ int main(int argc, char *argv[]) case 'W': params.wait_for_monitor++; break; +#ifdef CONFIG_MATCH_IFACE + case 'M': + params.Miface_count++; + iface = os_realloc_array(params.Mifaces, + params.Miface_count, + sizeof(struct wpa_interface)); + if (iface == NULL) + goto out; + params.Mifaces = iface; + iface = ¶ms.Mifaces[params.Miface_count - 1]; + os_memset(iface, 0, sizeof(*iface)); + break; +#endif /* CONFIG_MATCH_IFACE */ case 'N': iface_count++; iface = os_realloc_array(ifaces, iface_count, @@ -328,6 +363,9 @@ int main(int argc, char *argv[]) ifaces[i].ctrl_interface == NULL) || ifaces[i].ifname == NULL) { if (iface_count == 1 && (params.ctrl_interface || +#ifdef CONFIG_MATCH_IFACE + params.Miface_count || +#endif /* CONFIG_MATCH_IFACE */ params.dbus_ctrl_interface)) break; usage(); @@ -341,6 +379,11 @@ int main(int argc, char *argv[]) } } +#ifdef CONFIG_MATCH_IFACE + if (exitcode == 0) + exitcode = wpa_supplicant_init_match(global); +#endif /* CONFIG_MATCH_IFACE */ + if (exitcode == 0) exitcode = wpa_supplicant_run(global); @@ -351,6 +394,9 @@ int main(int argc, char *argv[]) out: wpa_supplicant_fd_workaround(0); os_free(ifaces); +#ifdef CONFIG_MATCH_IFACE + os_free(params.Mifaces); +#endif /* CONFIG_MATCH_IFACE */ os_free(params.pid_file); os_program_deinit(); diff --git a/wpa_supplicant/wpa_supplicant.c b/wpa_supplicant/wpa_supplicant.c index 7b43600..2853f0b 100644 --- a/wpa_supplicant/wpa_supplicant.c +++ b/wpa_supplicant/wpa_supplicant.c @@ -56,6 +56,9 @@ #include "wpas_kay.h" #include "mesh.h" +#include <net/if.h> +#include <fnmatch.h> + const char *const wpa_supplicant_version = "wpa_supplicant v" VERSION_STR "\n" "Copyright (c) 2003-2016, Jouni Malinen <j@xxxxx> and contributors"; @@ -4911,6 +4914,68 @@ static void wpa_supplicant_deinit_iface(struct wpa_supplicant *wpa_s, } +#ifdef CONFIG_MATCH_IFACE +/** + * wpa_supplicant_match_iface - Match an interface description to a name + * @global: Pointer to global data from wpa_supplicant_init() + * @ifname: Name of the interface to match + * Returns: Pointer to the created interface description or %NULL on failure + */ +struct wpa_interface * wpa_supplicant_match_iface(struct wpa_global *global, + const char *ifname) +{ + int i; + struct wpa_interface *iface, *miface; + + for (i = 0; i < global->params.Miface_count; i++) { + miface = &global->params.Mifaces[i]; + if (miface->ifname == NULL || + fnmatch(miface->ifname, ifname, 0) == 0) + { + iface = os_zalloc(sizeof(*iface)); + *iface = *miface; + iface->ifname = ifname; + return iface; + } + } + return NULL; +} + + +/** + * wpa_supplicant_match_existing - Match existing interfaces + * @global: Pointer to global data from wpa_supplicant_init() + * Returns: 0 on success, -1 on failure + */ +static int wpa_supplicant_match_existing(struct wpa_global *global) +{ + struct if_nameindex *ifi, *ifp; + struct wpa_supplicant *wpa_s; + struct wpa_interface *iface; + + ifp = if_nameindex(); + if (ifp == NULL) { + wpa_printf(MSG_ERROR, "if_nameindex: %s", strerror(errno)); + return -1; + } + + for (ifi = ifp; ifi->if_name; ifi++) { + wpa_s = wpa_supplicant_get_iface(global, ifi->if_name); + if (wpa_s != NULL) + continue; + iface = wpa_supplicant_match_iface(global, ifi->if_name); + if (iface != NULL) { + wpa_s = wpa_supplicant_add_iface(global, iface, NULL); + os_free(iface); + if (wpa_s != NULL) + wpa_s->matched = 1; + } + } + if_freenameindex(ifp); + return 0; +} +#endif /* CONFIG_MATCH_IFACE */ + /** * wpa_supplicant_add_iface - Add a new network interface * @global: Pointer to global data from wpa_supplicant_init() @@ -5212,6 +5277,16 @@ struct wpa_global * wpa_supplicant_init(struct wpa_params *params) if (params->override_ctrl_interface) global->params.override_ctrl_interface = os_strdup(params->override_ctrl_interface); +#ifdef CONFIG_MATCH_IFACE + global->params.Miface_count = params->Miface_count; + if (params->Miface_count) { + global->params.Mifaces = os_calloc(params->Miface_count, + sizeof(struct wpa_interface)); + os_memcpy(global->params.Mifaces, + params->Mifaces, + params->Miface_count * sizeof(struct wpa_interface)); + } +#endif /* CONFIG_MATCH_IFACE */ #ifdef CONFIG_P2P if (params->conf_p2p_dev) global->params.conf_p2p_dev = @@ -5291,6 +5366,11 @@ int wpa_supplicant_run(struct wpa_global *global) eloop_sock_requeue())) return -1; +#ifdef CONFIG_MATCH_IFACE + if (wpa_supplicant_match_existing(global)) + return -1; +#endif + if (global->params.wait_for_monitor) { for (wpa_s = global->ifaces; wpa_s; wpa_s = wpa_s->next) if (wpa_s->ctrl_iface && !wpa_s->p2p_mgmt) @@ -5359,6 +5439,9 @@ void wpa_supplicant_deinit(struct wpa_global *global) os_free(global->params.ctrl_interface_group); os_free(global->params.override_driver); os_free(global->params.override_ctrl_interface); +#ifdef CONFIG_MATCH_IFACE + os_free(global->params.Mifaces); +#endif /* CONFIG_MATCH_IFACE */ #ifdef CONFIG_P2P os_free(global->params.conf_p2p_dev); #endif /* CONFIG_P2P */ diff --git a/wpa_supplicant/wpa_supplicant_i.h b/wpa_supplicant/wpa_supplicant_i.h index 269bac0..5d372a9 100644 --- a/wpa_supplicant/wpa_supplicant_i.h +++ b/wpa_supplicant/wpa_supplicant_i.h @@ -217,6 +217,18 @@ struct wpa_params { */ char *entropy_file; +#ifdef CONFIG_MATCH_IFACE + /** + * Mifaces - Interface descriptions to match. + */ + struct wpa_interface *Mifaces; + + /** + * Miface_count - Number of defined matching interfaces. + */ + int Miface_count; +#endif /* CONFIG_MATCH_IFACE */ + #ifdef CONFIG_P2P /** * conf_p2p_dev - Configuration file used to hold the @@ -458,6 +470,9 @@ struct wpa_supplicant { unsigned char own_addr[ETH_ALEN]; unsigned char perm_addr[ETH_ALEN]; char ifname[100]; +#ifdef CONFIG_MATCH_IFACE + int matched; +#endif #ifdef CONFIG_CTRL_IFACE_DBUS char *dbus_path; #endif /* CONFIG_CTRL_IFACE_DBUS */ @@ -1109,6 +1124,8 @@ void free_hw_features(struct wpa_supplicant *wpa_s); void wpa_show_license(void); +struct wpa_interface * wpa_supplicant_match_iface(struct wpa_global *global, + const char *ifname); struct wpa_supplicant * wpa_supplicant_add_iface(struct wpa_global *global, struct wpa_interface *iface, struct wpa_supplicant *parent);
_______________________________________________ Hostap mailing list Hostap@xxxxxxxxxxxxxxxxxxx http://lists.infradead.org/mailman/listinfo/hostap