2019-04-11 03:10:24 +02:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Wireless daemon for Linux
|
|
|
|
*
|
|
|
|
* Copyright (C) 2019 Intel Corporation. All rights reserved.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <linux/if_ether.h>
|
|
|
|
#include <fnmatch.h>
|
|
|
|
|
|
|
|
#include <ell/ell.h>
|
|
|
|
|
|
|
|
#include "linux/nl80211.h"
|
|
|
|
|
|
|
|
#include "src/iwd.h"
|
2019-04-11 03:10:26 +02:00
|
|
|
#include "src/netdev.h"
|
2019-04-11 03:10:24 +02:00
|
|
|
#include "src/wiphy.h"
|
|
|
|
#include "src/util.h"
|
|
|
|
#include "src/common.h"
|
|
|
|
|
|
|
|
static struct l_genl_family *nl80211 = NULL;
|
2019-04-11 03:10:29 +02:00
|
|
|
static char **whitelist_filter;
|
|
|
|
static char **blacklist_filter;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
struct wiphy_setup_state {
|
|
|
|
uint32_t id;
|
|
|
|
struct wiphy *wiphy;
|
|
|
|
struct l_timeout *setup_timeout;
|
|
|
|
unsigned int pending_cmd_count;
|
|
|
|
bool aborted;
|
2019-04-11 03:10:28 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Data we may need if the driver does not seem to support interface
|
|
|
|
* manipulation and we fall back to using the driver-created default
|
|
|
|
* interface.
|
|
|
|
*/
|
|
|
|
bool use_default;
|
|
|
|
struct l_genl_msg *default_if_msg;
|
2019-04-11 03:10:26 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct l_queue *pending_wiphys;
|
|
|
|
|
2019-05-08 03:15:27 +02:00
|
|
|
/* With these drivers don't even try creating our interfaces */
|
|
|
|
const char *default_if_driver_list[] = {
|
|
|
|
/*
|
|
|
|
* The out-of-tree rtl88x2bu crashes the kernel hard. Seemingly
|
|
|
|
* many other drivers are built from the same source code so
|
|
|
|
* blacklist all of them. Unfortunately there are in-tree drivers
|
|
|
|
* that also match these names and may be fine. Use
|
|
|
|
* use_default_interface to override.
|
|
|
|
*/
|
|
|
|
"rtl81*",
|
|
|
|
"rtl87*",
|
|
|
|
"rtl88*",
|
|
|
|
"rtw_*",
|
|
|
|
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
static void wiphy_setup_state_free(void *data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = data;
|
|
|
|
|
|
|
|
if (state->setup_timeout)
|
|
|
|
l_timeout_remove(state->setup_timeout);
|
|
|
|
|
2019-04-11 03:10:28 +02:00
|
|
|
if (state->default_if_msg)
|
|
|
|
l_genl_msg_unref(state->default_if_msg);
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
L_WARN_ON(state->pending_cmd_count);
|
|
|
|
l_free(state);
|
|
|
|
}
|
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
static void wiphy_setup_state_destroy(struct wiphy_setup_state *state)
|
|
|
|
{
|
|
|
|
l_queue_remove(pending_wiphys, state);
|
|
|
|
wiphy_setup_state_free(state);
|
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:28 +02:00
|
|
|
static bool manager_use_default(struct wiphy_setup_state *state)
|
|
|
|
{
|
|
|
|
l_debug("");
|
|
|
|
|
|
|
|
if (!state->default_if_msg) {
|
|
|
|
l_error("No default interface for wiphy %u",
|
|
|
|
(unsigned int) state->id);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
netdev_create_from_genl(state->default_if_msg);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
static void manager_new_interface_cb(struct l_genl_msg *msg, void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
|
|
|
|
if (state->aborted)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (l_genl_msg_get_error(msg) < 0) {
|
|
|
|
l_error("NEW_INTERFACE failed: %s",
|
|
|
|
strerror(-l_genl_msg_get_error(msg)));
|
|
|
|
/*
|
|
|
|
* Nothing we can do to use this wiphy since by now we
|
|
|
|
* will have successfully deleted any default interface
|
|
|
|
* there may have been.
|
|
|
|
*/
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
netdev_create_from_genl(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_new_interface_done(void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
|
|
|
|
state->pending_cmd_count--;
|
2019-05-08 03:15:25 +02:00
|
|
|
wiphy_setup_state_destroy(state);
|
2019-04-11 03:10:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_create_interfaces(struct wiphy_setup_state *state)
|
|
|
|
{
|
|
|
|
struct l_genl_msg *msg;
|
|
|
|
char ifname[10];
|
|
|
|
uint32_t iftype = NL80211_IFTYPE_STATION;
|
|
|
|
unsigned cmd_id;
|
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
if (state->aborted)
|
2019-04-11 03:10:26 +02:00
|
|
|
return;
|
|
|
|
|
2019-04-11 03:10:28 +02:00
|
|
|
if (state->use_default) {
|
|
|
|
manager_use_default(state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
/*
|
|
|
|
* Current policy: we maintain one netdev per wiphy for station,
|
|
|
|
* AP and Ad-Hoc modes, one optional p2p-device and zero or more
|
|
|
|
* p2p-GOs or p2p-clients. The P2P-related interfaces will be
|
|
|
|
* created on request.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* To be improved */
|
|
|
|
snprintf(ifname, sizeof(ifname), "wlan%i", (int) state->id);
|
|
|
|
l_debug("creating %s", ifname);
|
|
|
|
|
|
|
|
msg = l_genl_msg_new(NL80211_CMD_NEW_INTERFACE);
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_WIPHY, 4, &state->id);
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_IFTYPE, 4, &iftype);
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_IFNAME,
|
|
|
|
strlen(ifname) + 1, ifname);
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_4ADDR, 1, "\0");
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_SOCKET_OWNER, 0, "");
|
|
|
|
cmd_id = l_genl_family_send(nl80211, msg,
|
|
|
|
manager_new_interface_cb, state,
|
|
|
|
manager_new_interface_done);
|
|
|
|
|
|
|
|
if (!cmd_id) {
|
2019-05-13 15:44:31 +02:00
|
|
|
l_error("Error sending NEW_INTERFACE for %s", ifname);
|
2019-04-11 03:10:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
state->pending_cmd_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_setup_cmd_done(void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
|
|
|
|
state->pending_cmd_count--;
|
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
if (state->pending_cmd_count)
|
|
|
|
return;
|
|
|
|
|
|
|
|
manager_create_interfaces(state);
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
if (!state->pending_cmd_count)
|
2019-05-08 03:15:25 +02:00
|
|
|
wiphy_setup_state_destroy(state);
|
2019-04-11 03:10:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_del_interface_cb(struct l_genl_msg *msg, void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
|
|
|
|
if (state->aborted)
|
|
|
|
return;
|
|
|
|
|
2019-04-11 03:10:28 +02:00
|
|
|
if (l_genl_msg_get_error(msg) < 0) {
|
2019-04-11 03:10:26 +02:00
|
|
|
l_error("DEL_INTERFACE failed: %s",
|
|
|
|
strerror(-l_genl_msg_get_error(msg)));
|
2019-04-11 03:10:28 +02:00
|
|
|
state->use_default = true;
|
|
|
|
}
|
2019-04-11 03:10:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_get_interface_cb(struct l_genl_msg *msg, void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
struct l_genl_attr attr;
|
|
|
|
uint16_t type, len;
|
|
|
|
const void *data;
|
|
|
|
const uint32_t *ifindex = NULL, *iftype = NULL;
|
|
|
|
const uint64_t *wdev_idx = NULL;
|
|
|
|
const char *ifname = NULL;
|
|
|
|
struct l_genl_msg *del_msg;
|
|
|
|
unsigned cmd_id;
|
2019-04-11 03:10:29 +02:00
|
|
|
char *pattern;
|
|
|
|
unsigned int i;
|
|
|
|
bool whitelisted = false, blacklisted = false;
|
2019-04-11 03:10:26 +02:00
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
|
|
|
|
if (state->aborted)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (l_genl_attr_next(&attr, &type, &len, &data)) {
|
|
|
|
switch (type) {
|
|
|
|
case NL80211_ATTR_IFINDEX:
|
|
|
|
if (len != sizeof(uint32_t)) {
|
|
|
|
l_warn("Invalid interface index attribute");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ifindex = data;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_ATTR_WDEV:
|
|
|
|
if (len != sizeof(uint64_t)) {
|
|
|
|
l_warn("Invalid wdev index attribute");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wdev_idx = data;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_ATTR_WIPHY:
|
|
|
|
if (len != sizeof(uint32_t) ||
|
|
|
|
*((uint32_t *) data) != state->id) {
|
|
|
|
l_warn("Invalid wiphy attribute");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_ATTR_IFTYPE:
|
|
|
|
if (len != sizeof(uint32_t)) {
|
|
|
|
l_warn("Invalid interface type attribute");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
iftype = data;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_ATTR_IFNAME:
|
|
|
|
if (len < 1 || !memchr(data + 1, 0, len - 1)) {
|
|
|
|
l_warn("Invalid interface name attribute");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ifname = data;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-13 15:44:31 +02:00
|
|
|
if (!wdev_idx || !iftype)
|
2019-04-11 03:10:26 +02:00
|
|
|
return;
|
|
|
|
|
2019-05-13 15:44:31 +02:00
|
|
|
if (ifindex) {
|
|
|
|
if (!ifname)
|
|
|
|
return;
|
2019-04-11 03:10:29 +02:00
|
|
|
|
2019-05-13 15:44:31 +02:00
|
|
|
if (whitelist_filter) {
|
|
|
|
for (i = 0; (pattern = whitelist_filter[i]); i++) {
|
|
|
|
if (fnmatch(pattern, ifname, 0) != 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
whitelisted = true;
|
|
|
|
break;
|
|
|
|
}
|
2019-04-11 03:10:29 +02:00
|
|
|
}
|
|
|
|
|
2019-05-13 15:44:31 +02:00
|
|
|
if (blacklist_filter) {
|
|
|
|
for (i = 0; (pattern = blacklist_filter[i]); i++) {
|
|
|
|
if (fnmatch(pattern, ifname, 0) != 0)
|
|
|
|
continue;
|
2019-04-11 03:10:29 +02:00
|
|
|
|
2019-05-13 15:44:31 +02:00
|
|
|
blacklisted = true;
|
|
|
|
break;
|
|
|
|
}
|
2019-04-11 03:10:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:28 +02:00
|
|
|
/*
|
|
|
|
* If this interface is usable as our default netdev in case the
|
|
|
|
* driver does not support interface manipulation, save the message
|
|
|
|
* just in case.
|
|
|
|
*/
|
|
|
|
if ((*iftype == NL80211_IFTYPE_ADHOC ||
|
|
|
|
*iftype == NL80211_IFTYPE_STATION ||
|
|
|
|
*iftype == NL80211_IFTYPE_AP) &&
|
|
|
|
ifindex && *ifindex != 0 &&
|
2019-04-11 03:10:29 +02:00
|
|
|
!state->default_if_msg &&
|
|
|
|
(!whitelist_filter || whitelisted) &&
|
|
|
|
!blacklisted)
|
2019-04-11 03:10:28 +02:00
|
|
|
state->default_if_msg = l_genl_msg_ref(msg);
|
|
|
|
|
|
|
|
if (state->use_default)
|
|
|
|
return;
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
del_msg = l_genl_msg_new(NL80211_CMD_DEL_INTERFACE);
|
2019-05-13 15:44:31 +02:00
|
|
|
|
|
|
|
if (ifindex)
|
|
|
|
l_genl_msg_append_attr(del_msg, NL80211_ATTR_IFINDEX, 4, ifindex);
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
l_genl_msg_append_attr(del_msg, NL80211_ATTR_WDEV, 8, wdev_idx);
|
|
|
|
l_genl_msg_append_attr(del_msg, NL80211_ATTR_WIPHY, 4, &state->id);
|
|
|
|
cmd_id = l_genl_family_send(nl80211, del_msg,
|
|
|
|
manager_del_interface_cb, state,
|
|
|
|
manager_setup_cmd_done);
|
|
|
|
|
|
|
|
if (!cmd_id) {
|
2019-05-13 15:44:31 +02:00
|
|
|
l_error("Sending DEL_INTERFACE for %s failed",
|
|
|
|
ifname ?: "unnamed interface");
|
2019-04-11 03:10:28 +02:00
|
|
|
state->use_default = true;
|
2019-04-11 03:10:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
state->pending_cmd_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_wiphy_dump_interfaces(struct wiphy_setup_state *state)
|
|
|
|
{
|
|
|
|
struct l_genl_msg *msg;
|
|
|
|
unsigned cmd_id;
|
|
|
|
|
|
|
|
if (state->setup_timeout) {
|
|
|
|
l_timeout_remove(state->setup_timeout);
|
|
|
|
state->setup_timeout = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* As the first step after new wiphy is detected we will query
|
|
|
|
* the initial interface setup, delete the default interfaces
|
|
|
|
* and create interfaces for our own use with NL80211_ATTR_SOCKET_OWNER
|
|
|
|
* on them. After that if new interfaces are created outside of
|
|
|
|
* IWD, or removed outside of IWD, we don't touch them and will
|
|
|
|
* try to minimally adapt to handle the removals correctly. It's
|
|
|
|
* a very unlikely situation in any case but it wouldn't make
|
|
|
|
* sense to try to continually enforce our setup fighting against
|
|
|
|
* some other process, and it wouldn't make sense to try to
|
|
|
|
* manage and use additional interfaces beyond the one or two
|
|
|
|
* we need for our operations.
|
|
|
|
*/
|
|
|
|
|
|
|
|
msg = l_genl_msg_new(NL80211_CMD_GET_INTERFACE);
|
|
|
|
l_genl_msg_append_attr(msg, NL80211_ATTR_WIPHY, 4, &state->id);
|
|
|
|
cmd_id = l_genl_family_dump(nl80211, msg,
|
|
|
|
manager_get_interface_cb, state,
|
|
|
|
manager_setup_cmd_done);
|
|
|
|
|
|
|
|
if (!cmd_id) {
|
|
|
|
l_error("Querying interface information for wiphy %u failed",
|
|
|
|
(unsigned int) state->id);
|
2019-05-08 03:15:25 +02:00
|
|
|
wiphy_setup_state_destroy(state);
|
2019-04-11 03:10:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
state->pending_cmd_count++;
|
|
|
|
}
|
|
|
|
|
2019-04-16 23:52:15 +02:00
|
|
|
static struct wiphy_setup_state *manager_rx_cmd_new_wiphy(
|
|
|
|
struct l_genl_msg *msg)
|
2019-04-11 03:10:24 +02:00
|
|
|
{
|
2019-04-16 23:52:15 +02:00
|
|
|
struct wiphy_setup_state *state = NULL;
|
2019-04-11 03:10:24 +02:00
|
|
|
struct wiphy *wiphy;
|
|
|
|
struct l_genl_attr attr;
|
|
|
|
uint32_t id;
|
|
|
|
const char *name;
|
2019-05-08 03:15:27 +02:00
|
|
|
const char *driver, **driver_bad;
|
2019-05-02 12:55:21 +02:00
|
|
|
bool use_default;
|
|
|
|
const struct l_settings *settings = iwd_get_config();
|
2019-04-11 03:10:24 +02:00
|
|
|
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
2019-04-16 23:52:15 +02:00
|
|
|
return NULL;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
|
|
|
if (!wiphy_parse_id_and_name(&attr, &id, &name))
|
2019-04-16 23:52:15 +02:00
|
|
|
return NULL;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
2019-04-16 21:42:06 +02:00
|
|
|
/*
|
|
|
|
* A Wiphy split dump can generate many (6+) NEW_WIPHY messages
|
|
|
|
* We need to parse attributes from all of them, but only perform
|
|
|
|
* initialization steps once for each new wiphy detected
|
|
|
|
*/
|
2019-04-11 03:10:24 +02:00
|
|
|
wiphy = wiphy_find(id);
|
2019-04-16 21:42:06 +02:00
|
|
|
if (wiphy)
|
|
|
|
goto done;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
2019-04-16 21:42:06 +02:00
|
|
|
wiphy = wiphy_create(id, name);
|
|
|
|
if (!wiphy)
|
2019-04-16 23:52:15 +02:00
|
|
|
return NULL;
|
2019-04-11 03:10:26 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* We've got a new wiphy, flag it as new and wait for a
|
|
|
|
* NEW_INTERFACE event for this wiphy's default driver-created
|
|
|
|
* interface. That event's handler will check the flag and
|
|
|
|
* finish setting up the interfaces for this new wiphy and then
|
|
|
|
* clear the flag. In some corner cases there may be no
|
|
|
|
* default interface on this wiphy and no user-space created
|
|
|
|
* interfaces from before IWD started, so set a 1-second timeout
|
|
|
|
* for the event. The timeout pointer is also used as the flag.
|
|
|
|
*/
|
|
|
|
|
|
|
|
state = l_new(struct wiphy_setup_state, 1);
|
|
|
|
state->id = id;
|
|
|
|
state->wiphy = wiphy;
|
|
|
|
l_queue_push_tail(pending_wiphys, state);
|
2019-04-16 21:42:06 +02:00
|
|
|
|
2019-05-08 03:15:27 +02:00
|
|
|
driver = wiphy_get_driver(wiphy);
|
|
|
|
|
|
|
|
for (driver_bad = default_if_driver_list; *driver_bad; driver_bad++)
|
|
|
|
if (fnmatch(*driver_bad, driver, 0) == 0)
|
|
|
|
state->use_default = true;
|
|
|
|
|
2019-05-02 12:55:20 +02:00
|
|
|
/*
|
|
|
|
* If whitelist/blacklist were given only try to use existing
|
|
|
|
* interfaces same as when the driver does not support NEW_INTERFACE
|
|
|
|
* or DEL_INTERFACE, otherwise the interface names will become
|
|
|
|
* meaningless after we've created our own interface(s). Optimally
|
|
|
|
* phy name white/blacklists should be used.
|
|
|
|
*/
|
|
|
|
if (whitelist_filter || blacklist_filter)
|
|
|
|
state->use_default = true;
|
|
|
|
|
2019-05-02 12:55:21 +02:00
|
|
|
/* The setting overrides our attempts to ensure things work */
|
|
|
|
if (l_settings_get_bool(settings, "General",
|
|
|
|
"use_default_interface", &use_default))
|
|
|
|
state->use_default = use_default;
|
|
|
|
|
|
|
|
if (state->use_default)
|
|
|
|
l_info("Wiphy %s will only use the default interface", name);
|
|
|
|
|
2019-04-16 21:42:06 +02:00
|
|
|
done:
|
|
|
|
wiphy_update_from_genl(wiphy, msg);
|
2019-04-16 23:52:15 +02:00
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_wiphy_setup_timeout(struct l_timeout *timeout,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state = user_data;
|
|
|
|
|
|
|
|
manager_wiphy_dump_interfaces(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_new_wiphy_event(struct l_genl_msg *msg)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state;
|
|
|
|
|
|
|
|
if (!pending_wiphys)
|
|
|
|
return;
|
|
|
|
|
|
|
|
state = manager_rx_cmd_new_wiphy(msg);
|
|
|
|
if (!state)
|
|
|
|
return;
|
|
|
|
|
2019-04-20 22:29:06 +02:00
|
|
|
wiphy_create_complete(state->wiphy);
|
|
|
|
|
2019-04-16 23:52:15 +02:00
|
|
|
/* Setup a timer just in case a default interface is not created */
|
|
|
|
state->setup_timeout = l_timeout_create(1, manager_wiphy_setup_timeout,
|
|
|
|
state, NULL);
|
2019-04-11 03:10:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool manager_wiphy_state_match(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const struct wiphy_setup_state *state = a;
|
|
|
|
uint32_t id = L_PTR_TO_UINT(b);
|
|
|
|
|
|
|
|
return (state->id == id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct wiphy_setup_state *manager_find_pending(uint32_t id)
|
|
|
|
{
|
|
|
|
return l_queue_find(pending_wiphys, manager_wiphy_state_match,
|
|
|
|
L_UINT_TO_PTR(id));
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t manager_parse_ifindex(struct l_genl_attr *attr)
|
|
|
|
{
|
|
|
|
uint16_t type, len;
|
|
|
|
const void *data;
|
|
|
|
|
|
|
|
while (l_genl_attr_next(attr, &type, &len, &data)) {
|
|
|
|
if (type != NL80211_ATTR_IFINDEX)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (len != sizeof(uint32_t))
|
|
|
|
break;
|
|
|
|
|
|
|
|
return *((uint32_t *) data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
2019-04-11 03:10:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t manager_parse_wiphy_id(struct l_genl_attr *attr)
|
|
|
|
{
|
|
|
|
uint16_t type, len;
|
|
|
|
const void *data;
|
|
|
|
|
|
|
|
while (l_genl_attr_next(attr, &type, &len, &data)) {
|
|
|
|
if (type != NL80211_ATTR_WIPHY)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (len != sizeof(uint32_t))
|
|
|
|
break;
|
|
|
|
|
|
|
|
return *((uint32_t *) data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_del_wiphy_event(struct l_genl_msg *msg)
|
|
|
|
{
|
2019-04-11 03:10:26 +02:00
|
|
|
struct wiphy_setup_state *state;
|
2019-04-11 03:10:24 +02:00
|
|
|
struct wiphy *wiphy;
|
|
|
|
struct l_genl_attr attr;
|
|
|
|
uint32_t id;
|
|
|
|
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
|
|
|
return;
|
|
|
|
|
|
|
|
id = manager_parse_wiphy_id(&attr);
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
state = manager_find_pending(id);
|
|
|
|
if (state) {
|
|
|
|
if (state->pending_cmd_count)
|
|
|
|
state->aborted = true;
|
|
|
|
else
|
2019-05-08 03:15:25 +02:00
|
|
|
wiphy_setup_state_destroy(state);
|
2019-04-11 03:10:26 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:24 +02:00
|
|
|
wiphy = wiphy_find(id);
|
|
|
|
if (wiphy)
|
|
|
|
wiphy_destroy(wiphy);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_config_notify(struct l_genl_msg *msg, void *user_data)
|
|
|
|
{
|
|
|
|
uint8_t cmd;
|
2019-04-11 03:10:26 +02:00
|
|
|
struct wiphy_setup_state *state;
|
|
|
|
struct l_genl_attr attr;
|
|
|
|
struct netdev *netdev;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
|
|
|
cmd = l_genl_msg_get_command(msg);
|
|
|
|
|
|
|
|
l_debug("Notification of command %u", cmd);
|
|
|
|
|
|
|
|
switch (cmd) {
|
|
|
|
case NL80211_CMD_NEW_WIPHY:
|
|
|
|
manager_new_wiphy_event(msg);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_CMD_DEL_WIPHY:
|
|
|
|
manager_del_wiphy_event(msg);
|
|
|
|
break;
|
2019-04-11 03:10:26 +02:00
|
|
|
|
|
|
|
case NL80211_CMD_NEW_INTERFACE:
|
|
|
|
/*
|
|
|
|
* If we have a NEW_INTERFACE for a freshly detected wiphy
|
|
|
|
* assume we can now query for the default or pre-created
|
|
|
|
* interfaces, remove any we don't need and create our own.
|
|
|
|
*/
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
|
|
|
break;
|
|
|
|
|
|
|
|
state = manager_find_pending(manager_parse_wiphy_id(&attr));
|
|
|
|
if (!state || !state->setup_timeout)
|
|
|
|
break;
|
|
|
|
|
|
|
|
manager_wiphy_dump_interfaces(state);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case NL80211_CMD_DEL_INTERFACE:
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
|
|
|
break;
|
|
|
|
|
|
|
|
netdev = netdev_find(manager_parse_ifindex(&attr));
|
|
|
|
if (!netdev)
|
|
|
|
break;
|
|
|
|
|
|
|
|
netdev_destroy(netdev);
|
|
|
|
break;
|
2019-04-11 03:10:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-17 00:37:41 +02:00
|
|
|
static void manager_interface_dump_callback(struct l_genl_msg *msg,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct wiphy_setup_state *state;
|
|
|
|
struct l_genl_attr attr;
|
|
|
|
|
|
|
|
l_debug("");
|
|
|
|
|
|
|
|
if (!l_genl_attr_init(&attr, msg))
|
|
|
|
return;
|
|
|
|
|
|
|
|
state = manager_find_pending(manager_parse_wiphy_id(&attr));
|
|
|
|
if (!state || state->setup_timeout)
|
|
|
|
return;
|
|
|
|
|
|
|
|
manager_get_interface_cb(msg, state);
|
|
|
|
}
|
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
static bool manager_check_create_interfaces(const void *a, const void *b)
|
2019-04-17 00:37:41 +02:00
|
|
|
{
|
2019-05-08 03:15:25 +02:00
|
|
|
struct wiphy_setup_state *state = (void *) a;
|
2019-04-17 00:37:41 +02:00
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
/* phy might have been detected after the initial dump */
|
|
|
|
if (state->setup_timeout)
|
|
|
|
return false;
|
2019-04-17 00:37:41 +02:00
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
wiphy_create_complete(state->wiphy);
|
2019-04-20 22:29:06 +02:00
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
if (state->pending_cmd_count)
|
|
|
|
return false;
|
2019-04-20 22:29:06 +02:00
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
/* If we are here, then there are no interfaces for this phy */
|
|
|
|
manager_create_interfaces(state);
|
2019-04-17 00:37:41 +02:00
|
|
|
|
2019-05-08 03:15:25 +02:00
|
|
|
if (state->pending_cmd_count)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
wiphy_setup_state_free(state);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void manager_interface_dump_done(void *user_data)
|
|
|
|
{
|
|
|
|
l_queue_remove_if(pending_wiphys,
|
|
|
|
manager_check_create_interfaces, NULL);
|
2019-04-17 00:37:41 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:24 +02:00
|
|
|
static void manager_wiphy_dump_callback(struct l_genl_msg *msg, void *user_data)
|
|
|
|
{
|
|
|
|
l_debug("");
|
|
|
|
|
2019-04-17 00:37:41 +02:00
|
|
|
manager_rx_cmd_new_wiphy(msg);
|
2019-04-11 03:10:24 +02:00
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:29 +02:00
|
|
|
bool manager_init(struct l_genl_family *in,
|
|
|
|
const char *if_whitelist, const char *if_blacklist)
|
2019-04-11 03:10:24 +02:00
|
|
|
{
|
2019-04-16 23:18:59 +02:00
|
|
|
struct l_genl_msg *msg;
|
2019-04-17 00:37:41 +02:00
|
|
|
unsigned int wiphy_dump;
|
|
|
|
unsigned int interface_dump;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
2019-04-16 23:18:59 +02:00
|
|
|
nl80211 = in;
|
2019-04-11 03:10:24 +02:00
|
|
|
|
2019-04-11 03:10:29 +02:00
|
|
|
if (if_whitelist)
|
|
|
|
whitelist_filter = l_strsplit(if_whitelist, ',');
|
|
|
|
|
|
|
|
if (if_blacklist)
|
|
|
|
blacklist_filter = l_strsplit(if_blacklist, ',');
|
|
|
|
|
2019-04-11 03:10:26 +02:00
|
|
|
pending_wiphys = l_queue_new();
|
|
|
|
|
2019-04-16 23:18:59 +02:00
|
|
|
if (!l_genl_family_register(nl80211, "config", manager_config_notify,
|
|
|
|
NULL, NULL)) {
|
|
|
|
l_error("Registering for config notifications failed");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg = l_genl_msg_new(NL80211_CMD_GET_WIPHY);
|
2019-04-17 00:37:41 +02:00
|
|
|
wiphy_dump = l_genl_family_dump(nl80211, msg,
|
|
|
|
manager_wiphy_dump_callback,
|
|
|
|
NULL, NULL);
|
|
|
|
if (!wiphy_dump) {
|
2019-04-16 23:18:59 +02:00
|
|
|
l_error("Initial wiphy information dump failed");
|
|
|
|
l_genl_msg_unref(msg);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-04-17 00:37:41 +02:00
|
|
|
msg = l_genl_msg_new(NL80211_CMD_GET_INTERFACE);
|
|
|
|
interface_dump = l_genl_family_dump(nl80211, msg,
|
|
|
|
manager_interface_dump_callback,
|
|
|
|
NULL,
|
|
|
|
manager_interface_dump_done);
|
|
|
|
if (!interface_dump) {
|
|
|
|
l_error("Initial interface information dump failed");
|
|
|
|
l_genl_msg_unref(msg);
|
|
|
|
l_genl_family_cancel(nl80211, wiphy_dump);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-04-11 03:10:24 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void manager_exit(void)
|
|
|
|
{
|
2019-04-11 03:10:29 +02:00
|
|
|
l_strfreev(whitelist_filter);
|
|
|
|
l_strfreev(blacklist_filter);
|
|
|
|
|
2019-06-21 19:22:48 +02:00
|
|
|
l_queue_destroy(pending_wiphys, wiphy_setup_state_free);
|
2019-04-11 03:10:26 +02:00
|
|
|
pending_wiphys = NULL;
|
|
|
|
|
2019-04-11 03:10:24 +02:00
|
|
|
nl80211 = NULL;
|
|
|
|
}
|