mirror of
https://git.kernel.org/pub/scm/network/wireless/iwd.git
synced 2024-11-22 14:49:24 +01:00
scan: refactor to use wiphy radio work queue
To use the wiphy radio work queue, scanning mostly remained the same. start_next_scan_request was modified to be used as the work callback, as well as not start the next scan if the current one was done (since this is taken care of by wiphy work queue now). All calls to start_next_scan_request were removed, and more or less replaced with wiphy_radio_work_done. scan_{suspend,resume} were both removed since radio management priorities solve this for us. ANQP requests can be inserted ahead of scan requests, which accomplishes the same thing.
This commit is contained in:
parent
a3d0eebe74
commit
5f7b28d501
139
src/scan.c
139
src/scan.c
@ -56,7 +56,8 @@
|
|||||||
static struct l_queue *scan_contexts;
|
static struct l_queue *scan_contexts;
|
||||||
|
|
||||||
static struct l_genl_family *nl80211;
|
static struct l_genl_family *nl80211;
|
||||||
static uint32_t next_scan_request_id;
|
|
||||||
|
struct scan_context;
|
||||||
|
|
||||||
struct scan_periodic {
|
struct scan_periodic {
|
||||||
struct l_timeout *timeout;
|
struct l_timeout *timeout;
|
||||||
@ -70,7 +71,7 @@ struct scan_periodic {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct scan_request {
|
struct scan_request {
|
||||||
uint32_t id;
|
struct scan_context *sc;
|
||||||
scan_trigger_func_t trigger;
|
scan_trigger_func_t trigger;
|
||||||
scan_notify_func_t callback;
|
scan_notify_func_t callback;
|
||||||
void *userdata;
|
void *userdata;
|
||||||
@ -79,6 +80,7 @@ struct scan_request {
|
|||||||
struct l_queue *cmds;
|
struct l_queue *cmds;
|
||||||
/* The time the current scan was started. Reported in TRIGGER_SCAN */
|
/* The time the current scan was started. Reported in TRIGGER_SCAN */
|
||||||
uint64_t start_time_tsf;
|
uint64_t start_time_tsf;
|
||||||
|
struct wiphy_radio_work_item work;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct scan_context {
|
struct scan_context {
|
||||||
@ -106,7 +108,7 @@ struct scan_context {
|
|||||||
bool triggered:1;
|
bool triggered:1;
|
||||||
/* Whether any commands from current request's queue have started */
|
/* Whether any commands from current request's queue have started */
|
||||||
bool started:1;
|
bool started:1;
|
||||||
bool suspended:1;
|
bool work_started:1;
|
||||||
struct wiphy *wiphy;
|
struct wiphy *wiphy;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -118,7 +120,7 @@ struct scan_results {
|
|||||||
struct scan_request *sr;
|
struct scan_request *sr;
|
||||||
};
|
};
|
||||||
|
|
||||||
static bool start_next_scan_request(struct scan_context *sc);
|
static bool start_next_scan_request(struct wiphy_radio_work_item *item);
|
||||||
static void scan_periodic_rearm(struct scan_context *sc);
|
static void scan_periodic_rearm(struct scan_context *sc);
|
||||||
|
|
||||||
static bool scan_context_match(const void *a, const void *b)
|
static bool scan_context_match(const void *a, const void *b)
|
||||||
@ -134,12 +136,13 @@ static bool scan_request_match(const void *a, const void *b)
|
|||||||
const struct scan_request *sr = a;
|
const struct scan_request *sr = a;
|
||||||
uint32_t id = L_PTR_TO_UINT(b);
|
uint32_t id = L_PTR_TO_UINT(b);
|
||||||
|
|
||||||
return sr->id == id;
|
return sr->work.id == id;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void scan_request_free(void *data)
|
static void scan_request_free(struct wiphy_radio_work_item *item)
|
||||||
{
|
{
|
||||||
struct scan_request *sr = data;
|
struct scan_request *sr = l_container_of(item, struct scan_request,
|
||||||
|
work);
|
||||||
|
|
||||||
if (sr->destroy)
|
if (sr->destroy)
|
||||||
sr->destroy(sr->userdata);
|
sr->destroy(sr->userdata);
|
||||||
@ -159,12 +162,12 @@ static void scan_request_failed(struct scan_context *sc,
|
|||||||
else if (sr->callback)
|
else if (sr->callback)
|
||||||
sr->callback(err, NULL, sr->userdata);
|
sr->callback(err, NULL, sr->userdata);
|
||||||
|
|
||||||
scan_request_free(sr);
|
wiphy_radio_work_done(sc->wiphy, sr->work.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct scan_context *scan_context_new(uint64_t wdev_id)
|
static struct scan_context *scan_context_new(uint64_t wdev_id)
|
||||||
{
|
{
|
||||||
struct wiphy *wiphy = wiphy_find(wdev_id >> 32);
|
struct wiphy *wiphy = wiphy_find_by_wdev(wdev_id);
|
||||||
struct scan_context *sc;
|
struct scan_context *sc;
|
||||||
|
|
||||||
if (!wiphy)
|
if (!wiphy)
|
||||||
@ -180,11 +183,18 @@ static struct scan_context *scan_context_new(uint64_t wdev_id)
|
|||||||
return sc;
|
return sc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void scan_request_cancel(void *data)
|
||||||
|
{
|
||||||
|
struct scan_request *sr = data;
|
||||||
|
|
||||||
|
wiphy_radio_work_done(sr->sc->wiphy, sr->work.id);
|
||||||
|
}
|
||||||
|
|
||||||
static void scan_context_free(struct scan_context *sc)
|
static void scan_context_free(struct scan_context *sc)
|
||||||
{
|
{
|
||||||
l_debug("sc: %p", sc);
|
l_debug("sc: %p", sc);
|
||||||
|
|
||||||
l_queue_destroy(sc->requests, scan_request_free);
|
l_queue_destroy(sc->requests, scan_request_cancel);
|
||||||
|
|
||||||
if (sc->sp.timeout)
|
if (sc->sp.timeout)
|
||||||
l_timeout_remove(sc->sp.timeout);
|
l_timeout_remove(sc->sp.timeout);
|
||||||
@ -215,7 +225,6 @@ static void scan_request_triggered(struct l_genl_msg *msg, void *userdata)
|
|||||||
}
|
}
|
||||||
|
|
||||||
l_queue_remove(sc->requests, sr);
|
l_queue_remove(sc->requests, sr);
|
||||||
start_next_scan_request(sc);
|
|
||||||
|
|
||||||
scan_request_failed(sc, sr, err);
|
scan_request_failed(sc, sr, err);
|
||||||
|
|
||||||
@ -517,6 +526,11 @@ static int scan_request_send_trigger(struct scan_context *sc,
|
|||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const struct wiphy_radio_work_item_ops work_ops = {
|
||||||
|
.do_work = start_next_scan_request,
|
||||||
|
.destroy = scan_request_free,
|
||||||
|
};
|
||||||
|
|
||||||
static uint32_t scan_common(uint64_t wdev_id, bool passive,
|
static uint32_t scan_common(uint64_t wdev_id, bool passive,
|
||||||
const struct scan_parameters *params,
|
const struct scan_parameters *params,
|
||||||
scan_trigger_func_t trigger,
|
scan_trigger_func_t trigger,
|
||||||
@ -532,36 +546,19 @@ static uint32_t scan_common(uint64_t wdev_id, bool passive,
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
sr = l_new(struct scan_request, 1);
|
sr = l_new(struct scan_request, 1);
|
||||||
|
sr->sc = sc;
|
||||||
sr->trigger = trigger;
|
sr->trigger = trigger;
|
||||||
sr->callback = notify;
|
sr->callback = notify;
|
||||||
sr->userdata = userdata;
|
sr->userdata = userdata;
|
||||||
sr->destroy = destroy;
|
sr->destroy = destroy;
|
||||||
sr->passive = passive;
|
sr->passive = passive;
|
||||||
sr->id = ++next_scan_request_id;
|
|
||||||
sr->cmds = l_queue_new();
|
sr->cmds = l_queue_new();
|
||||||
|
|
||||||
scan_cmds_add(sr->cmds, sc, passive, params);
|
scan_cmds_add(sr->cmds, sc, passive, params);
|
||||||
|
|
||||||
/* Queue empty implies !sc->triggered && !sc->start_cmd_id */
|
|
||||||
if (!l_queue_isempty(sc->requests))
|
|
||||||
goto done;
|
|
||||||
|
|
||||||
if (sc->suspended)
|
|
||||||
goto done;
|
|
||||||
|
|
||||||
if (sc->state != SCAN_STATE_NOT_RUNNING)
|
|
||||||
goto done;
|
|
||||||
|
|
||||||
if (!scan_request_send_trigger(sc, sr))
|
|
||||||
goto done;
|
|
||||||
|
|
||||||
sr->destroy = NULL; /* Don't call destroy when returning error */
|
|
||||||
scan_request_free(sr);
|
|
||||||
return 0;
|
|
||||||
done:
|
|
||||||
l_queue_push_tail(sc->requests, sr);
|
l_queue_push_tail(sc->requests, sr);
|
||||||
|
|
||||||
return sr->id;
|
return wiphy_radio_work_insert(sc->wiphy, &sr->work, 2, &work_ops);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t scan_passive(uint64_t wdev_id, struct scan_freq_set *freqs,
|
uint32_t scan_passive(uint64_t wdev_id, struct scan_freq_set *freqs,
|
||||||
@ -649,11 +646,11 @@ bool scan_cancel(uint64_t wdev_id, uint32_t id)
|
|||||||
sc->start_cmd_id = 0;
|
sc->start_cmd_id = 0;
|
||||||
l_queue_remove(sc->requests, sr);
|
l_queue_remove(sc->requests, sr);
|
||||||
sc->started = false;
|
sc->started = false;
|
||||||
start_next_scan_request(sc);
|
|
||||||
} else
|
} else
|
||||||
l_queue_remove(sc->requests, sr);
|
l_queue_remove(sc->requests, sr);
|
||||||
|
|
||||||
scan_request_free(sr);
|
wiphy_radio_work_done(sc->wiphy, sr->work.id);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -832,34 +829,30 @@ static void scan_periodic_rearm(struct scan_context *sc)
|
|||||||
scan_periodic_timeout_destroy);
|
scan_periodic_timeout_destroy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool start_next_scan_request(struct scan_context *sc)
|
static bool start_next_scan_request(struct wiphy_radio_work_item *item)
|
||||||
{
|
{
|
||||||
struct scan_request *sr = l_queue_peek_head(sc->requests);
|
struct scan_request *sr = l_container_of(item,
|
||||||
|
struct scan_request, work);
|
||||||
|
struct scan_context *sc = sr->sc;
|
||||||
|
|
||||||
if (sc->suspended)
|
sc->work_started = true;
|
||||||
return true;
|
|
||||||
|
|
||||||
if (sc->state != SCAN_STATE_NOT_RUNNING)
|
if (sc->state != SCAN_STATE_NOT_RUNNING)
|
||||||
return true;
|
return false;
|
||||||
|
|
||||||
if (sc->start_cmd_id || sc->get_scan_cmd_id)
|
if (!scan_request_send_trigger(sc, sr))
|
||||||
return true;
|
return false;
|
||||||
|
|
||||||
while (sr) {
|
sc->work_started = false;
|
||||||
if (!scan_request_send_trigger(sc, sr))
|
|
||||||
return true;
|
|
||||||
|
|
||||||
scan_request_failed(sc, sr, -EIO);
|
scan_request_failed(sc, sr, -EIO);
|
||||||
|
|
||||||
sr = l_queue_peek_head(sc->requests);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sc->sp.retry) {
|
if (sc->sp.retry) {
|
||||||
sc->sp.retry = false;
|
sc->sp.retry = false;
|
||||||
scan_periodic_queue(sc);
|
scan_periodic_queue(sc);
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool scan_parse_vendor_specific(struct scan_bss *bss, const void *data,
|
static bool scan_parse_vendor_specific(struct scan_bss *bss, const void *data,
|
||||||
@ -1500,6 +1493,7 @@ static void scan_finished(struct scan_context *sc,
|
|||||||
if (sr) {
|
if (sr) {
|
||||||
l_queue_remove(sc->requests, sr);
|
l_queue_remove(sc->requests, sr);
|
||||||
sc->started = false;
|
sc->started = false;
|
||||||
|
sc->work_started = false;
|
||||||
|
|
||||||
if (sr->callback)
|
if (sr->callback)
|
||||||
new_owner = sr->callback(err, bss_list, sr->userdata);
|
new_owner = sr->callback(err, bss_list, sr->userdata);
|
||||||
@ -1511,9 +1505,7 @@ static void scan_finished(struct scan_context *sc,
|
|||||||
* taken care of sending the next command for a new or ongoing
|
* taken care of sending the next command for a new or ongoing
|
||||||
* scan, or scheduling the next periodic scan.
|
* scan, or scheduling the next periodic scan.
|
||||||
*/
|
*/
|
||||||
start_next_scan_request(sc);
|
wiphy_radio_work_done(sc->wiphy, sr->work.id);
|
||||||
|
|
||||||
scan_request_free(sr);
|
|
||||||
} else if (sc->sp.callback)
|
} else if (sc->sp.callback)
|
||||||
new_owner = sc->sp.callback(err, bss_list, sc->sp.userdata);
|
new_owner = sc->sp.callback(err, bss_list, sc->sp.userdata);
|
||||||
|
|
||||||
@ -1677,9 +1669,21 @@ static void scan_notify(struct l_genl_msg *msg, void *user_data)
|
|||||||
sr = NULL;
|
sr = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Send the next command of a new or an ongoing request */
|
/*
|
||||||
if (send_next)
|
* Send the next command of an ongoing request, or continue with
|
||||||
start_next_scan_request(sc);
|
* a previously busy scan attempt due to an external scan. A
|
||||||
|
* temporary scan_request object is used (rather than 'sr')
|
||||||
|
* because the state of 'sr' tells us if the results should
|
||||||
|
* be used either as normal scan results, or to take advantage
|
||||||
|
* of the external scan as a 'free' periodic scan of sorts.
|
||||||
|
*/
|
||||||
|
if (sc->work_started && send_next) {
|
||||||
|
struct scan_request *next = l_queue_peek_head(
|
||||||
|
sc->requests);
|
||||||
|
|
||||||
|
if (next)
|
||||||
|
start_next_scan_request(&next->work);
|
||||||
|
}
|
||||||
|
|
||||||
if (!get_results)
|
if (!get_results)
|
||||||
break;
|
break;
|
||||||
@ -1730,7 +1734,8 @@ static void scan_notify(struct l_genl_msg *msg, void *user_data)
|
|||||||
* hardware or the driver because of another activity
|
* hardware or the driver because of another activity
|
||||||
* starting in which case we should just get an EBUSY.
|
* starting in which case we should just get an EBUSY.
|
||||||
*/
|
*/
|
||||||
start_next_scan_request(sc);
|
if (sc->work_started)
|
||||||
|
start_next_scan_request(&sr->work);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
@ -2133,32 +2138,6 @@ bool scan_wdev_remove(uint64_t wdev_id)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool scan_suspend(uint64_t wdev_id)
|
|
||||||
{
|
|
||||||
struct scan_context *sc;
|
|
||||||
|
|
||||||
sc = l_queue_find(scan_contexts, scan_context_match, &wdev_id);
|
|
||||||
if (!sc)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
sc->suspended = true;
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void scan_resume(uint64_t wdev_id)
|
|
||||||
{
|
|
||||||
struct scan_context *sc;
|
|
||||||
|
|
||||||
sc = l_queue_find(scan_contexts, scan_context_match, &wdev_id);
|
|
||||||
if (!sc)
|
|
||||||
return;
|
|
||||||
|
|
||||||
sc->suspended = false;
|
|
||||||
|
|
||||||
start_next_scan_request(sc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int scan_init(void)
|
static int scan_init(void)
|
||||||
{
|
{
|
||||||
const struct l_settings *config = iwd_get_config();
|
const struct l_settings *config = iwd_get_config();
|
||||||
|
@ -170,6 +170,3 @@ bool scan_freq_set_isempty(const struct scan_freq_set *set);
|
|||||||
|
|
||||||
bool scan_wdev_add(uint64_t wdev_id);
|
bool scan_wdev_add(uint64_t wdev_id);
|
||||||
bool scan_wdev_remove(uint64_t wdev_id);
|
bool scan_wdev_remove(uint64_t wdev_id);
|
||||||
|
|
||||||
bool scan_suspend(uint64_t wdev_id);
|
|
||||||
void scan_resume(uint64_t wdev_id);
|
|
||||||
|
@ -531,8 +531,6 @@ request_done:
|
|||||||
station_network_foreach(station, network_add_foreach, station);
|
station_network_foreach(station, network_add_foreach, station);
|
||||||
station_autoconnect_next(station);
|
station_autoconnect_next(station);
|
||||||
}
|
}
|
||||||
|
|
||||||
scan_resume(netdev_get_wdev_id(station->netdev));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool station_start_anqp(struct station *station, struct network *network,
|
static bool station_start_anqp(struct station *station, struct network *network,
|
||||||
@ -662,19 +660,7 @@ void station_set_scan_results(struct station *station,
|
|||||||
|
|
||||||
l_hashmap_foreach_remove(station->networks, process_network, station);
|
l_hashmap_foreach_remove(station->networks, process_network, station);
|
||||||
|
|
||||||
/*
|
if (!wait_for_anqp && add_to_autoconnect) {
|
||||||
* ANQP requests are scheduled in the same manor as scans, and cannot
|
|
||||||
* be done simultaneously. To avoid long queue times (waiting for a
|
|
||||||
* scan to finish) its best to stop scanning, do ANQP, then resume
|
|
||||||
* scanning.
|
|
||||||
*
|
|
||||||
* TODO: It may be possible for some hardware to actually scan and do
|
|
||||||
* ANQP at the same time. Detecting this could allow us to continue
|
|
||||||
* scanning.
|
|
||||||
*/
|
|
||||||
if (wait_for_anqp)
|
|
||||||
scan_suspend(netdev_get_wdev_id(station->netdev));
|
|
||||||
else if (add_to_autoconnect) {
|
|
||||||
station_network_foreach(station, network_add_foreach, station);
|
station_network_foreach(station, network_add_foreach, station);
|
||||||
station_autoconnect_next(station);
|
station_autoconnect_next(station);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user