3
0
mirror of https://git.kernel.org/pub/scm/network/wireless/iwd.git synced 2025-01-03 10:32:33 +01:00

station: autoconnect based on network, not BSS

Prior to the BSS blacklist a BSS based autoconnect list made
the most sense, but now station actually retries all BSS's upon
failure. This means that for each BSS in the autoconnect list
every other BSS under that SSID will be attempted to connect to
if there is a failure. Essentially this is a network based
autoconnect list, just an indirect way of doing it.

Intead the autoconnect list can be purely network based, using
the network rank for sorting. This avoids the need for a special
autoconnect_entry struct as well as ensures the last connected
network is chosen first (simply based on existing network ranking
logic).
This commit is contained in:
James Prestwood 2021-05-07 13:26:17 -07:00 committed by Denis Kenzior
parent 0be5beffc8
commit 877d910a44

View File

@ -152,12 +152,6 @@ static bool station_is_autoconnecting(struct station *station)
station->state == STATION_STATE_AUTOCONNECT_QUICK; station->state == STATION_STATE_AUTOCONNECT_QUICK;
} }
struct autoconnect_entry {
uint16_t rank;
struct network *network;
struct scan_bss *bss;
};
static void station_property_set_scanning(struct station *station, static void station_property_set_scanning(struct station *station,
bool scanning) bool scanning)
{ {
@ -176,25 +170,23 @@ static void station_enter_state(struct station *station,
static void station_autoconnect_next(struct station *station) static void station_autoconnect_next(struct station *station)
{ {
struct autoconnect_entry *entry; struct network *network;
int r; int r;
while ((entry = l_queue_pop_head(station->autoconnect_list))) { while ((network = l_queue_pop_head(station->autoconnect_list))) {
const char *ssid = network_get_ssid(entry->network); const char *ssid = network_get_ssid(network);
struct scan_bss *bss = network_bss_select(network, false);
l_debug("Considering autoconnecting to BSS '%s' with SSID: %s," l_debug("Considering autoconnecting to BSS '%s' with SSID: %s,"
" freq: %u, rank: %u, strength: %i", " freq: %u, rank: %u, strength: %i",
util_address_to_string(entry->bss->addr), ssid, util_address_to_string(bss->addr), ssid,
entry->bss->frequency, entry->rank, bss->frequency, bss->rank,
entry->bss->signal_strength); bss->signal_strength);
if (blacklist_contains_bss(entry->bss->addr)) { if (blacklist_contains_bss(bss->addr))
l_free(entry);
continue; continue;
}
r = network_autoconnect(entry->network, entry->bss); r = network_autoconnect(network, bss);
l_free(entry);
if (!r) { if (!r) {
station_enter_state(station, STATION_STATE_CONNECTING); station_enter_state(station, STATION_STATE_CONNECTING);
@ -212,33 +204,6 @@ static void station_autoconnect_next(struct station *station)
} }
} }
static int autoconnect_rank_compare(const void *a, const void *b, void *user)
{
const struct autoconnect_entry *new_ae = a;
const struct autoconnect_entry *ae = b;
return (ae->rank > new_ae->rank) ? 1 : -1;
}
static void station_add_autoconnect_bss(struct station *station,
struct network *network,
struct scan_bss *bss)
{
double rankmod;
struct autoconnect_entry *entry;
/* See if network is autoconnectable (is a known network) */
if (!network_rankmod(network, &rankmod))
return;
entry = l_new(struct autoconnect_entry, 1);
entry->network = network;
entry->bss = bss;
entry->rank = bss->rank * rankmod;
l_queue_insert(station->autoconnect_list, entry,
autoconnect_rank_compare, NULL);
}
static void bss_free(void *data) static void bss_free(void *data)
{ {
struct scan_bss *bss = data; struct scan_bss *bss = data;
@ -457,12 +422,9 @@ static bool match_nai_realms(const struct network_info *info, void *user_data)
static void network_add_foreach(struct network *network, void *user_data) static void network_add_foreach(struct network *network, void *user_data)
{ {
struct station *station = user_data; struct station *station = user_data;
struct scan_bss *bss = network_bss_select(network, false);
if (!bss) l_queue_insert(station->autoconnect_list, network,
return; network_rank_compare, NULL);
station_add_autoconnect_bss(station, network, bss);
} }
static bool match_pending(const void *a, const void *b) static bool match_pending(const void *a, const void *b)
@ -553,7 +515,7 @@ request_done:
/* Notify all watchers now that every ANQP request has finished */ /* Notify all watchers now that every ANQP request has finished */
l_queue_foreach_remove(station->anqp_pending, anqp_entry_foreach, NULL); l_queue_foreach_remove(station->anqp_pending, anqp_entry_foreach, NULL);
l_queue_destroy(station->autoconnect_list, l_free); l_queue_destroy(station->autoconnect_list, NULL);
station->autoconnect_list = l_queue_new(); station->autoconnect_list = l_queue_new();
if (station_is_autoconnecting(station)) { if (station_is_autoconnecting(station)) {
@ -662,7 +624,7 @@ void station_set_scan_results(struct station *station,
l_queue_clear(station->hidden_bss_list_sorted, NULL); l_queue_clear(station->hidden_bss_list_sorted, NULL);
l_queue_destroy(station->autoconnect_list, l_free); l_queue_destroy(station->autoconnect_list, NULL);
station->autoconnect_list = l_queue_new(); station->autoconnect_list = l_queue_new();
station_bss_list_remove_expired_bsses(station, freqs); station_bss_list_remove_expired_bsses(station, freqs);
@ -3631,7 +3593,7 @@ static void station_free(struct station *station)
l_hashmap_destroy(station->networks, network_free); l_hashmap_destroy(station->networks, network_free);
l_queue_destroy(station->bss_list, bss_free); l_queue_destroy(station->bss_list, bss_free);
l_queue_destroy(station->hidden_bss_list_sorted, NULL); l_queue_destroy(station->hidden_bss_list_sorted, NULL);
l_queue_destroy(station->autoconnect_list, l_free); l_queue_destroy(station->autoconnect_list, NULL);
watchlist_destroy(&station->state_watches); watchlist_destroy(&station->state_watches);