mirror of
https://git.kernel.org/pub/scm/network/wireless/iwd.git
synced 2025-01-10 01:02:39 +01:00
station: Move scanning code from device
This commit is contained in:
parent
1b22af2052
commit
1378a3c945
245
src/device.c
245
src/device.c
@ -51,11 +51,7 @@
|
|||||||
struct device {
|
struct device {
|
||||||
uint32_t index;
|
uint32_t index;
|
||||||
enum device_state state;
|
enum device_state state;
|
||||||
struct l_queue *bss_list;
|
|
||||||
struct l_queue *old_bss_list;
|
|
||||||
struct l_dbus_message *scan_pending;
|
struct l_dbus_message *scan_pending;
|
||||||
struct l_hashmap *networks;
|
|
||||||
struct l_queue *networks_sorted;
|
|
||||||
struct scan_bss *connected_bss;
|
struct scan_bss *connected_bss;
|
||||||
struct network *connected_network;
|
struct network *connected_network;
|
||||||
struct l_dbus_message *connect_pending;
|
struct l_dbus_message *connect_pending;
|
||||||
@ -77,7 +73,6 @@ struct device {
|
|||||||
bool signal_low : 1;
|
bool signal_low : 1;
|
||||||
bool roam_no_orig_ap : 1;
|
bool roam_no_orig_ap : 1;
|
||||||
bool ap_directed_roaming : 1;
|
bool ap_directed_roaming : 1;
|
||||||
bool seen_hidden_networks : 1;
|
|
||||||
|
|
||||||
uint32_t ap_roam_watch;
|
uint32_t ap_roam_watch;
|
||||||
};
|
};
|
||||||
@ -108,25 +103,6 @@ void __iwd_device_foreach(iwd_device_foreach_func func, void *user_data)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *iwd_network_get_path(struct device *device,
|
|
||||||
const char *ssid,
|
|
||||||
enum security security)
|
|
||||||
{
|
|
||||||
static char path[256];
|
|
||||||
unsigned int pos, i;
|
|
||||||
|
|
||||||
pos = snprintf(path, sizeof(path), "%s/", device_get_path(device));
|
|
||||||
|
|
||||||
for (i = 0; ssid[i] && pos < sizeof(path); i++)
|
|
||||||
pos += snprintf(path + pos, sizeof(path) - pos, "%02x",
|
|
||||||
ssid[i]);
|
|
||||||
|
|
||||||
snprintf(path + pos, sizeof(path) - pos, "_%s",
|
|
||||||
security_to_str(security));
|
|
||||||
|
|
||||||
return path;
|
|
||||||
}
|
|
||||||
|
|
||||||
static const char *device_state_to_string(enum device_state state)
|
static const char *device_state_to_string(enum device_state state)
|
||||||
{
|
{
|
||||||
switch (state) {
|
switch (state) {
|
||||||
@ -149,173 +125,10 @@ static const char *device_state_to_string(enum device_state state)
|
|||||||
return "invalid";
|
return "invalid";
|
||||||
}
|
}
|
||||||
|
|
||||||
static void bss_free(void *data)
|
|
||||||
{
|
|
||||||
struct scan_bss *bss = data;
|
|
||||||
const char *addr;
|
|
||||||
|
|
||||||
addr = util_address_to_string(bss->addr);
|
|
||||||
l_debug("Freeing BSS %s", addr);
|
|
||||||
|
|
||||||
scan_bss_free(bss);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void network_free(void *data)
|
|
||||||
{
|
|
||||||
struct network *network = data;
|
|
||||||
|
|
||||||
network_remove(network, -ESHUTDOWN);
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool process_network(const void *key, void *data, void *user_data)
|
|
||||||
{
|
|
||||||
struct network *network = data;
|
|
||||||
struct device *device = user_data;
|
|
||||||
|
|
||||||
if (!network_bss_list_isempty(network)) {
|
|
||||||
/* Build the network list ordered by rank */
|
|
||||||
network_rank_update(network);
|
|
||||||
|
|
||||||
l_queue_insert(device->networks_sorted, network,
|
|
||||||
network_rank_compare, NULL);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Drop networks that have no more BSSs in range */
|
|
||||||
l_debug("No remaining BSSs for SSID: %s -- Removing network",
|
|
||||||
network_get_ssid(network));
|
|
||||||
network_remove(network, -ERANGE);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Returns the network object the BSS was added to or NULL if ignored.
|
|
||||||
*/
|
|
||||||
static struct network *add_seen_bss(struct device *device, struct scan_bss *bss)
|
|
||||||
{
|
|
||||||
struct network *network;
|
|
||||||
struct ie_rsn_info info;
|
|
||||||
int r;
|
|
||||||
enum security security;
|
|
||||||
const char *path;
|
|
||||||
char ssid[33];
|
|
||||||
|
|
||||||
l_debug("Found BSS '%s' with SSID: %s, freq: %u, rank: %u, "
|
|
||||||
"strength: %i",
|
|
||||||
util_address_to_string(bss->addr),
|
|
||||||
util_ssid_to_utf8(bss->ssid_len, bss->ssid),
|
|
||||||
bss->frequency, bss->rank, bss->signal_strength);
|
|
||||||
|
|
||||||
if (util_ssid_is_hidden(bss->ssid_len, bss->ssid)) {
|
|
||||||
l_debug("Ignoring BSS with hidden SSID");
|
|
||||||
device->seen_hidden_networks = true;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!util_ssid_is_utf8(bss->ssid_len, bss->ssid)) {
|
|
||||||
l_debug("Ignoring BSS with non-UTF8 SSID");
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
memcpy(ssid, bss->ssid, bss->ssid_len);
|
|
||||||
ssid[bss->ssid_len] = '\0';
|
|
||||||
|
|
||||||
memset(&info, 0, sizeof(info));
|
|
||||||
r = scan_bss_get_rsn_info(bss, &info);
|
|
||||||
if (r < 0) {
|
|
||||||
if (r != -ENOENT)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
security = security_determine(bss->capability, NULL);
|
|
||||||
} else
|
|
||||||
security = security_determine(bss->capability, &info);
|
|
||||||
|
|
||||||
path = iwd_network_get_path(device, ssid, security);
|
|
||||||
|
|
||||||
network = l_hashmap_lookup(device->networks, path);
|
|
||||||
if (!network) {
|
|
||||||
network = network_create(device, ssid, security);
|
|
||||||
|
|
||||||
if (!network_register(network, path)) {
|
|
||||||
network_remove(network, -EINVAL);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
l_hashmap_insert(device->networks,
|
|
||||||
network_get_path(network), network);
|
|
||||||
l_debug("Added new Network \"%s\" security %s",
|
|
||||||
network_get_ssid(network), security_to_str(security));
|
|
||||||
}
|
|
||||||
|
|
||||||
network_bss_add(network, bss);
|
|
||||||
|
|
||||||
return network;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool bss_match(const void *a, const void *b)
|
|
||||||
{
|
|
||||||
const struct scan_bss *bss_a = a;
|
|
||||||
const struct scan_bss *bss_b = b;
|
|
||||||
|
|
||||||
return !memcmp(bss_a->addr, bss_b->addr, sizeof(bss_a->addr));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Used when scan results were obtained; either from passive scan running
|
|
||||||
* inside device.c or active scans running in other state machines, e.g. wsc.c
|
|
||||||
*/
|
|
||||||
void device_set_scan_results(struct device *device, struct l_queue *bss_list,
|
void device_set_scan_results(struct device *device, struct l_queue *bss_list,
|
||||||
bool add_to_autoconnect)
|
bool add_to_autoconnect)
|
||||||
{
|
{
|
||||||
struct network *network;
|
station_set_scan_results(device->station, bss_list, add_to_autoconnect);
|
||||||
const struct l_queue_entry *bss_entry;
|
|
||||||
|
|
||||||
device->old_bss_list = device->bss_list;
|
|
||||||
device->bss_list = bss_list;
|
|
||||||
|
|
||||||
device->seen_hidden_networks = false;
|
|
||||||
|
|
||||||
while ((network = l_queue_pop_head(device->networks_sorted)))
|
|
||||||
network_bss_list_clear(network);
|
|
||||||
|
|
||||||
l_queue_destroy(device->station->autoconnect_list, l_free);
|
|
||||||
device->station->autoconnect_list = l_queue_new();
|
|
||||||
|
|
||||||
for (bss_entry = l_queue_get_entries(bss_list); bss_entry;
|
|
||||||
bss_entry = bss_entry->next) {
|
|
||||||
struct scan_bss *bss = bss_entry->data;
|
|
||||||
struct network *network = add_seen_bss(device, bss);
|
|
||||||
|
|
||||||
if (network && add_to_autoconnect)
|
|
||||||
station_add_autoconnect_bss(device->station,
|
|
||||||
network, bss);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (device->connected_bss) {
|
|
||||||
struct scan_bss *bss;
|
|
||||||
|
|
||||||
bss = l_queue_find(device->bss_list, bss_match,
|
|
||||||
device->connected_bss);
|
|
||||||
|
|
||||||
if (!bss) {
|
|
||||||
l_warn("Connected BSS not in scan results!");
|
|
||||||
device->connected_bss->rank = 0;
|
|
||||||
l_queue_push_tail(device->bss_list,
|
|
||||||
device->connected_bss);
|
|
||||||
network_bss_add(device->connected_network,
|
|
||||||
device->connected_bss);
|
|
||||||
l_queue_remove(device->old_bss_list,
|
|
||||||
device->connected_bss);
|
|
||||||
} else
|
|
||||||
device->connected_bss = bss;
|
|
||||||
}
|
|
||||||
|
|
||||||
l_hashmap_foreach_remove(device->networks, process_network, device);
|
|
||||||
|
|
||||||
l_queue_destroy(device->old_bss_list, bss_free);
|
|
||||||
device->old_bss_list = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool new_scan_results(uint32_t wiphy_id, uint32_t ifindex, int err,
|
static bool new_scan_results(uint32_t wiphy_id, uint32_t ifindex, int err,
|
||||||
@ -419,9 +232,7 @@ bool device_remove_state_watch(struct device *device, uint32_t id)
|
|||||||
struct network *device_network_find(struct device *device, const char *ssid,
|
struct network *device_network_find(struct device *device, const char *ssid,
|
||||||
enum security security)
|
enum security security)
|
||||||
{
|
{
|
||||||
const char *path = iwd_network_get_path(device, ssid, security);
|
return station_network_find(device->station, ssid, security);
|
||||||
|
|
||||||
return l_hashmap_lookup(device->networks, path);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void device_enter_state(struct device *device, enum device_state state)
|
static void device_enter_state(struct device *device, enum device_state state)
|
||||||
@ -492,6 +303,7 @@ static void device_reset_connection_state(struct device *device)
|
|||||||
scan_cancel(device->index, device->roam_scan_id);
|
scan_cancel(device->index, device->roam_scan_id);
|
||||||
|
|
||||||
device->connected_bss = NULL;
|
device->connected_bss = NULL;
|
||||||
|
station->connected_bss = NULL;
|
||||||
device->connected_network = NULL;
|
device->connected_network = NULL;
|
||||||
station->connected_network = NULL;
|
station->connected_network = NULL;
|
||||||
|
|
||||||
@ -605,6 +417,8 @@ static void device_transition_reassociate(struct device *device,
|
|||||||
struct scan_bss *bss,
|
struct scan_bss *bss,
|
||||||
struct handshake_state *new_hs)
|
struct handshake_state *new_hs)
|
||||||
{
|
{
|
||||||
|
struct station *station = device->station;
|
||||||
|
|
||||||
if (netdev_reassociate(device->netdev, bss, device->connected_bss,
|
if (netdev_reassociate(device->netdev, bss, device->connected_bss,
|
||||||
new_hs, device_netdev_event,
|
new_hs, device_netdev_event,
|
||||||
device_reassociate_cb, device) < 0) {
|
device_reassociate_cb, device) < 0) {
|
||||||
@ -615,6 +429,7 @@ static void device_transition_reassociate(struct device *device,
|
|||||||
}
|
}
|
||||||
|
|
||||||
device->connected_bss = bss;
|
device->connected_bss = bss;
|
||||||
|
station->connected_bss = bss;
|
||||||
device->preparing_roam = false;
|
device->preparing_roam = false;
|
||||||
device_enter_state(device, DEVICE_STATE_ROAMING);
|
device_enter_state(device, DEVICE_STATE_ROAMING);
|
||||||
}
|
}
|
||||||
@ -642,7 +457,7 @@ static void device_preauthenticate_cb(struct netdev *netdev,
|
|||||||
if (!device->preparing_roam || result == NETDEV_RESULT_ABORTED)
|
if (!device->preparing_roam || result == NETDEV_RESULT_ABORTED)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
bss = l_queue_find(device->bss_list, bss_match_bssid,
|
bss = l_queue_find(device->station->bss_list, bss_match_bssid,
|
||||||
device->preauth_bssid);
|
device->preauth_bssid);
|
||||||
if (!bss) {
|
if (!bss) {
|
||||||
l_error("Roam target BSS not found");
|
l_error("Roam target BSS not found");
|
||||||
@ -732,6 +547,7 @@ static void device_transition_start(struct device *device, struct scan_bss *bss)
|
|||||||
}
|
}
|
||||||
|
|
||||||
device->connected_bss = bss;
|
device->connected_bss = bss;
|
||||||
|
station->connected_bss = bss;
|
||||||
device->preparing_roam = false;
|
device->preparing_roam = false;
|
||||||
device_enter_state(device, DEVICE_STATE_ROAMING);
|
device_enter_state(device, DEVICE_STATE_ROAMING);
|
||||||
|
|
||||||
@ -895,7 +711,7 @@ next:
|
|||||||
goto fail_free_bss;
|
goto fail_free_bss;
|
||||||
|
|
||||||
/* See if we have anywhere to roam to */
|
/* See if we have anywhere to roam to */
|
||||||
if (!best_bss || bss_match(best_bss, device->connected_bss))
|
if (!best_bss || scan_bss_addr_eq(best_bss, device->connected_bss))
|
||||||
goto fail_free_bss;
|
goto fail_free_bss;
|
||||||
|
|
||||||
bss = network_bss_find_by_addr(network, best_bss->addr);
|
bss = network_bss_find_by_addr(network, best_bss->addr);
|
||||||
@ -904,7 +720,7 @@ next:
|
|||||||
best_bss = bss;
|
best_bss = bss;
|
||||||
} else {
|
} else {
|
||||||
network_bss_add(network, best_bss);
|
network_bss_add(network, best_bss);
|
||||||
l_queue_push_tail(device->bss_list, best_bss);
|
l_queue_push_tail(device->station->bss_list, best_bss);
|
||||||
}
|
}
|
||||||
|
|
||||||
device_transition_start(device, best_bss);
|
device_transition_start(device, best_bss);
|
||||||
@ -1433,6 +1249,7 @@ int __device_connect_network(struct device *device, struct network *network,
|
|||||||
}
|
}
|
||||||
|
|
||||||
device->connected_bss = bss;
|
device->connected_bss = bss;
|
||||||
|
station->connected_bss = bss;
|
||||||
device->connected_network = network;
|
device->connected_network = network;
|
||||||
station->connected_network = network;
|
station->connected_network = network;
|
||||||
|
|
||||||
@ -1516,7 +1333,7 @@ static struct l_dbus_message *device_scan(struct l_dbus *dbus,
|
|||||||
* use passive scanning to hide our MAC address
|
* use passive scanning to hide our MAC address
|
||||||
*/
|
*/
|
||||||
if (!device->connected_bss &&
|
if (!device->connected_bss &&
|
||||||
!(device->seen_hidden_networks &&
|
!(device->station->seen_hidden_networks &&
|
||||||
known_networks_has_hidden())) {
|
known_networks_has_hidden())) {
|
||||||
if (!scan_passive(device->index, device_scan_triggered,
|
if (!scan_passive(device->index, device_scan_triggered,
|
||||||
new_scan_results, device, NULL))
|
new_scan_results, device, NULL))
|
||||||
@ -1625,6 +1442,7 @@ static struct l_dbus_message *device_get_networks(struct l_dbus *dbus,
|
|||||||
struct device *device = user_data;
|
struct device *device = user_data;
|
||||||
struct l_dbus_message *reply;
|
struct l_dbus_message *reply;
|
||||||
struct l_dbus_message_builder *builder;
|
struct l_dbus_message_builder *builder;
|
||||||
|
struct l_queue *sorted = device->station->networks_sorted;
|
||||||
const struct l_queue_entry *entry;
|
const struct l_queue_entry *entry;
|
||||||
|
|
||||||
reply = l_dbus_message_new_method_return(message);
|
reply = l_dbus_message_new_method_return(message);
|
||||||
@ -1632,8 +1450,7 @@ static struct l_dbus_message *device_get_networks(struct l_dbus *dbus,
|
|||||||
|
|
||||||
l_dbus_message_builder_enter_array(builder, "(osns)");
|
l_dbus_message_builder_enter_array(builder, "(osns)");
|
||||||
|
|
||||||
for (entry = l_queue_get_entries(device->networks_sorted); entry;
|
for (entry = l_queue_get_entries(sorted); entry; entry = entry->next) {
|
||||||
entry = entry->next) {
|
|
||||||
const struct network *network = entry->data;
|
const struct network *network = entry->data;
|
||||||
enum security security = network_get_security(network);
|
enum security security = network_get_security(network);
|
||||||
int16_t signal_strength = network_get_signal_strength(network);
|
int16_t signal_strength = network_get_signal_strength(network);
|
||||||
@ -1736,15 +1553,6 @@ static struct l_dbus_message *device_signal_agent_register(struct l_dbus *dbus,
|
|||||||
return l_dbus_message_new_method_return(message);
|
return l_dbus_message_new_method_return(message);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool device_remove_network(const void *key, void *data, void *user_data)
|
|
||||||
{
|
|
||||||
struct network *network = data;
|
|
||||||
|
|
||||||
network_remove(network, -ESHUTDOWN);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct l_dbus_message *device_signal_agent_unregister(
|
static struct l_dbus_message *device_signal_agent_unregister(
|
||||||
struct l_dbus *dbus,
|
struct l_dbus *dbus,
|
||||||
struct l_dbus_message *message,
|
struct l_dbus_message *message,
|
||||||
@ -1826,8 +1634,8 @@ static bool device_hidden_network_scan_results(uint32_t wiphy_id,
|
|||||||
memcmp(bss->ssid, ssid, ssid_len))
|
memcmp(bss->ssid, ssid, ssid_len))
|
||||||
goto next;
|
goto next;
|
||||||
|
|
||||||
if (add_seen_bss(device, bss)) {
|
if (station_add_seen_bss(device->station, bss)) {
|
||||||
l_queue_push_tail(device->bss_list, bss);
|
l_queue_push_tail(device->station->bss_list, bss);
|
||||||
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -2324,15 +2132,6 @@ static void device_netdev_notify(struct netdev *netdev,
|
|||||||
|
|
||||||
device_reset_connection_state(device);
|
device_reset_connection_state(device);
|
||||||
|
|
||||||
l_hashmap_foreach_remove(device->networks,
|
|
||||||
device_remove_network, device);
|
|
||||||
|
|
||||||
l_queue_destroy(device->bss_list, bss_free);
|
|
||||||
device->bss_list = l_queue_new();
|
|
||||||
|
|
||||||
l_queue_destroy(device->networks_sorted, NULL);
|
|
||||||
device->networks_sorted = l_queue_new();
|
|
||||||
|
|
||||||
l_dbus_property_changed(dbus, device_get_path(device),
|
l_dbus_property_changed(dbus, device_get_path(device),
|
||||||
IWD_DEVICE_INTERFACE, "Powered");
|
IWD_DEVICE_INTERFACE, "Powered");
|
||||||
break;
|
break;
|
||||||
@ -2357,13 +2156,7 @@ struct device *device_create(struct wiphy *wiphy, struct netdev *netdev)
|
|||||||
const uint8_t action_ap_roam_prefix[2] = { 0x0a, 0x07 };
|
const uint8_t action_ap_roam_prefix[2] = { 0x0a, 0x07 };
|
||||||
|
|
||||||
device = l_new(struct device, 1);
|
device = l_new(struct device, 1);
|
||||||
device->bss_list = l_queue_new();
|
|
||||||
device->networks = l_hashmap_new();
|
|
||||||
watchlist_init(&device->state_watches, NULL);
|
watchlist_init(&device->state_watches, NULL);
|
||||||
l_hashmap_set_hash_function(device->networks, l_str_hash);
|
|
||||||
l_hashmap_set_compare_function(device->networks,
|
|
||||||
(l_hashmap_compare_func_t) strcmp);
|
|
||||||
device->networks_sorted = l_queue_new();
|
|
||||||
device->index = ifindex;
|
device->index = ifindex;
|
||||||
device->wiphy = wiphy;
|
device->wiphy = wiphy;
|
||||||
device->netdev = netdev;
|
device->netdev = netdev;
|
||||||
@ -2425,12 +2218,6 @@ static void device_free(void *user)
|
|||||||
dbus = dbus_get_bus();
|
dbus = dbus_get_bus();
|
||||||
l_dbus_unregister_object(dbus, device_get_path(device));
|
l_dbus_unregister_object(dbus, device_get_path(device));
|
||||||
|
|
||||||
l_queue_destroy(device->networks_sorted, NULL);
|
|
||||||
l_hashmap_destroy(device->networks, network_free);
|
|
||||||
|
|
||||||
l_queue_destroy(device->bss_list, bss_free);
|
|
||||||
l_queue_destroy(device->old_bss_list, bss_free);
|
|
||||||
|
|
||||||
l_timeout_remove(device->roam_trigger_timeout);
|
l_timeout_remove(device->roam_trigger_timeout);
|
||||||
|
|
||||||
scan_ifindex_remove(device->index);
|
scan_ifindex_remove(device->index);
|
||||||
|
208
src/station.c
208
src/station.c
@ -24,6 +24,7 @@
|
|||||||
#include <config.h>
|
#include <config.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
|
||||||
#include <ell/ell.h>
|
#include <ell/ell.h>
|
||||||
@ -96,6 +97,203 @@ void station_add_autoconnect_bss(struct station *station,
|
|||||||
autoconnect_rank_compare, NULL);
|
autoconnect_rank_compare, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void bss_free(void *data)
|
||||||
|
{
|
||||||
|
struct scan_bss *bss = data;
|
||||||
|
const char *addr;
|
||||||
|
|
||||||
|
addr = util_address_to_string(bss->addr);
|
||||||
|
l_debug("Freeing BSS %s", addr);
|
||||||
|
|
||||||
|
scan_bss_free(bss);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void network_free(void *data)
|
||||||
|
{
|
||||||
|
struct network *network = data;
|
||||||
|
|
||||||
|
network_remove(network, -ESHUTDOWN);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool process_network(const void *key, void *data, void *user_data)
|
||||||
|
{
|
||||||
|
struct network *network = data;
|
||||||
|
struct station *station = user_data;
|
||||||
|
|
||||||
|
if (!network_bss_list_isempty(network)) {
|
||||||
|
/* Build the network list ordered by rank */
|
||||||
|
network_rank_update(network);
|
||||||
|
|
||||||
|
l_queue_insert(station->networks_sorted, network,
|
||||||
|
network_rank_compare, NULL);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Drop networks that have no more BSSs in range */
|
||||||
|
l_debug("No remaining BSSs for SSID: %s -- Removing network",
|
||||||
|
network_get_ssid(network));
|
||||||
|
network_remove(network, -ERANGE);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *iwd_network_get_path(struct station *station,
|
||||||
|
const char *ssid,
|
||||||
|
enum security security)
|
||||||
|
{
|
||||||
|
static char path[256];
|
||||||
|
unsigned int pos, i;
|
||||||
|
|
||||||
|
pos = snprintf(path, sizeof(path), "%s/",
|
||||||
|
netdev_get_path(station->netdev));
|
||||||
|
|
||||||
|
for (i = 0; ssid[i] && pos < sizeof(path); i++)
|
||||||
|
pos += snprintf(path + pos, sizeof(path) - pos, "%02x",
|
||||||
|
ssid[i]);
|
||||||
|
|
||||||
|
snprintf(path + pos, sizeof(path) - pos, "_%s",
|
||||||
|
security_to_str(security));
|
||||||
|
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct network *station_network_find(struct station *station, const char *ssid,
|
||||||
|
enum security security)
|
||||||
|
{
|
||||||
|
const char *path = iwd_network_get_path(station, ssid, security);
|
||||||
|
|
||||||
|
return l_hashmap_lookup(station->networks, path);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns the network object the BSS was added to or NULL if ignored.
|
||||||
|
*/
|
||||||
|
struct network *station_add_seen_bss(struct station *station,
|
||||||
|
struct scan_bss *bss)
|
||||||
|
{
|
||||||
|
struct device *device = netdev_get_device(station->netdev);
|
||||||
|
struct network *network;
|
||||||
|
struct ie_rsn_info info;
|
||||||
|
int r;
|
||||||
|
enum security security;
|
||||||
|
const char *path;
|
||||||
|
char ssid[33];
|
||||||
|
|
||||||
|
l_debug("Found BSS '%s' with SSID: %s, freq: %u, rank: %u, "
|
||||||
|
"strength: %i",
|
||||||
|
util_address_to_string(bss->addr),
|
||||||
|
util_ssid_to_utf8(bss->ssid_len, bss->ssid),
|
||||||
|
bss->frequency, bss->rank, bss->signal_strength);
|
||||||
|
|
||||||
|
if (util_ssid_is_hidden(bss->ssid_len, bss->ssid)) {
|
||||||
|
l_debug("Ignoring BSS with hidden SSID");
|
||||||
|
station->seen_hidden_networks = true;
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!util_ssid_is_utf8(bss->ssid_len, bss->ssid)) {
|
||||||
|
l_debug("Ignoring BSS with non-UTF8 SSID");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(ssid, bss->ssid, bss->ssid_len);
|
||||||
|
ssid[bss->ssid_len] = '\0';
|
||||||
|
|
||||||
|
memset(&info, 0, sizeof(info));
|
||||||
|
r = scan_bss_get_rsn_info(bss, &info);
|
||||||
|
if (r < 0) {
|
||||||
|
if (r != -ENOENT)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
security = security_determine(bss->capability, NULL);
|
||||||
|
} else
|
||||||
|
security = security_determine(bss->capability, &info);
|
||||||
|
|
||||||
|
path = iwd_network_get_path(station, ssid, security);
|
||||||
|
|
||||||
|
network = l_hashmap_lookup(station->networks, path);
|
||||||
|
if (!network) {
|
||||||
|
network = network_create(device, ssid, security);
|
||||||
|
|
||||||
|
if (!network_register(network, path)) {
|
||||||
|
network_remove(network, -EINVAL);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
l_hashmap_insert(station->networks,
|
||||||
|
network_get_path(network), network);
|
||||||
|
l_debug("Added new Network \"%s\" security %s",
|
||||||
|
network_get_ssid(network), security_to_str(security));
|
||||||
|
}
|
||||||
|
|
||||||
|
network_bss_add(network, bss);
|
||||||
|
|
||||||
|
return network;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool bss_match(const void *a, const void *b)
|
||||||
|
{
|
||||||
|
const struct scan_bss *bss_a = a;
|
||||||
|
const struct scan_bss *bss_b = b;
|
||||||
|
|
||||||
|
return !memcmp(bss_a->addr, bss_b->addr, sizeof(bss_a->addr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Used when scan results were obtained; either from passive scan running
|
||||||
|
* inside station module or active scans running in other state machines, e.g.
|
||||||
|
* wsc
|
||||||
|
*/
|
||||||
|
void station_set_scan_results(struct station *station, struct l_queue *bss_list,
|
||||||
|
bool add_to_autoconnect)
|
||||||
|
{
|
||||||
|
struct l_queue *old_bss_list = station->bss_list;
|
||||||
|
struct network *network;
|
||||||
|
const struct l_queue_entry *bss_entry;
|
||||||
|
|
||||||
|
station->bss_list = bss_list;
|
||||||
|
|
||||||
|
station->seen_hidden_networks = false;
|
||||||
|
|
||||||
|
while ((network = l_queue_pop_head(station->networks_sorted)))
|
||||||
|
network_bss_list_clear(network);
|
||||||
|
|
||||||
|
l_queue_destroy(station->autoconnect_list, l_free);
|
||||||
|
station->autoconnect_list = l_queue_new();
|
||||||
|
|
||||||
|
for (bss_entry = l_queue_get_entries(bss_list); bss_entry;
|
||||||
|
bss_entry = bss_entry->next) {
|
||||||
|
struct scan_bss *bss = bss_entry->data;
|
||||||
|
struct network *network = station_add_seen_bss(station, bss);
|
||||||
|
|
||||||
|
if (network && add_to_autoconnect)
|
||||||
|
station_add_autoconnect_bss(station, network, bss);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (station->connected_bss) {
|
||||||
|
struct scan_bss *bss;
|
||||||
|
|
||||||
|
bss = l_queue_find(station->bss_list, bss_match,
|
||||||
|
station->connected_bss);
|
||||||
|
|
||||||
|
if (!bss) {
|
||||||
|
l_warn("Connected BSS not in scan results!");
|
||||||
|
station->connected_bss->rank = 0;
|
||||||
|
l_queue_push_tail(station->bss_list,
|
||||||
|
station->connected_bss);
|
||||||
|
network_bss_add(station->connected_network,
|
||||||
|
station->connected_bss);
|
||||||
|
l_queue_remove(old_bss_list, station->connected_bss);
|
||||||
|
} else
|
||||||
|
station->connected_bss = bss;
|
||||||
|
}
|
||||||
|
|
||||||
|
l_hashmap_foreach_remove(station->networks, process_network, station);
|
||||||
|
|
||||||
|
l_queue_destroy(old_bss_list, bss_free);
|
||||||
|
}
|
||||||
|
|
||||||
static enum ie_rsn_akm_suite select_akm_suite(struct network *network,
|
static enum ie_rsn_akm_suite select_akm_suite(struct network *network,
|
||||||
struct scan_bss *bss,
|
struct scan_bss *bss,
|
||||||
struct ie_rsn_info *info)
|
struct ie_rsn_info *info)
|
||||||
@ -304,6 +502,13 @@ struct station *station_create(struct wiphy *wiphy, struct netdev *netdev)
|
|||||||
|
|
||||||
station = l_new(struct station, 1);
|
station = l_new(struct station, 1);
|
||||||
|
|
||||||
|
station->bss_list = l_queue_new();
|
||||||
|
station->networks = l_hashmap_new();
|
||||||
|
l_hashmap_set_hash_function(station->networks, l_str_hash);
|
||||||
|
l_hashmap_set_compare_function(station->networks,
|
||||||
|
(l_hashmap_compare_func_t) strcmp);
|
||||||
|
station->networks_sorted = l_queue_new();
|
||||||
|
|
||||||
station->wiphy = wiphy;
|
station->wiphy = wiphy;
|
||||||
station->netdev = netdev;
|
station->netdev = netdev;
|
||||||
|
|
||||||
@ -319,6 +524,9 @@ void station_free(struct station *station)
|
|||||||
if (!l_queue_remove(station_list, station))
|
if (!l_queue_remove(station_list, station))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
l_queue_destroy(station->networks_sorted, NULL);
|
||||||
|
l_hashmap_destroy(station->networks, network_free);
|
||||||
|
l_queue_destroy(station->bss_list, bss_free);
|
||||||
l_queue_destroy(station->autoconnect_list, l_free);
|
l_queue_destroy(station->autoconnect_list, l_free);
|
||||||
|
|
||||||
l_free(station);
|
l_free(station);
|
||||||
|
@ -29,11 +29,17 @@ struct scan_bss;
|
|||||||
struct network;
|
struct network;
|
||||||
|
|
||||||
struct station {
|
struct station {
|
||||||
|
struct scan_bss *connected_bss;
|
||||||
struct network *connected_network;
|
struct network *connected_network;
|
||||||
struct l_queue *autoconnect_list;
|
struct l_queue *autoconnect_list;
|
||||||
|
struct l_queue *bss_list;
|
||||||
|
struct l_hashmap *networks;
|
||||||
|
struct l_queue *networks_sorted;
|
||||||
|
|
||||||
struct wiphy *wiphy;
|
struct wiphy *wiphy;
|
||||||
struct netdev *netdev;
|
struct netdev *netdev;
|
||||||
|
|
||||||
|
bool seen_hidden_networks : 1;
|
||||||
};
|
};
|
||||||
|
|
||||||
void station_autoconnect_next(struct station *station);
|
void station_autoconnect_next(struct station *station);
|
||||||
@ -41,6 +47,15 @@ void station_add_autoconnect_bss(struct station *station,
|
|||||||
struct network *network,
|
struct network *network,
|
||||||
struct scan_bss *bss);
|
struct scan_bss *bss);
|
||||||
|
|
||||||
|
struct network *station_network_find(struct station *station, const char *ssid,
|
||||||
|
enum security security);
|
||||||
|
|
||||||
|
struct network *station_add_seen_bss(struct station *station,
|
||||||
|
struct scan_bss *bss);
|
||||||
|
|
||||||
|
void station_set_scan_results(struct station *station, struct l_queue *bss_list,
|
||||||
|
bool add_to_autoconnect);
|
||||||
|
|
||||||
struct handshake_state *station_handshake_setup(struct station *station,
|
struct handshake_state *station_handshake_setup(struct station *station,
|
||||||
struct network *network,
|
struct network *network,
|
||||||
struct scan_bss *bss);
|
struct scan_bss *bss);
|
||||||
|
Loading…
Reference in New Issue
Block a user