Commit 5260a5b2 authored by Johannes Berg's avatar Johannes Berg

mac80211: track scheduled scan virtual interface

Instead of tracking whether or not we're in a
scheduled scan, track the virtual interface
(sdata) in an RCU-protected pointer to make it
usable from RX to check the MAC address.
Signed-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent e2fd5dbc
...@@ -965,9 +965,9 @@ struct ieee80211_local { ...@@ -965,9 +965,9 @@ struct ieee80211_local {
int scan_channel_idx; int scan_channel_idx;
int scan_ies_len; int scan_ies_len;
bool sched_scanning;
struct ieee80211_sched_scan_ies sched_scan_ies; struct ieee80211_sched_scan_ies sched_scan_ies;
struct work_struct sched_scan_stopped_work; struct work_struct sched_scan_stopped_work;
struct ieee80211_sub_if_data __rcu *sched_scan_sdata;
unsigned long leave_oper_channel_time; unsigned long leave_oper_channel_time;
enum mac80211_scan_state next_scan_state; enum mac80211_scan_state next_scan_state;
......
...@@ -322,7 +322,8 @@ static void ieee80211_restart_work(struct work_struct *work) ...@@ -322,7 +322,8 @@ static void ieee80211_restart_work(struct work_struct *work)
mutex_lock(&local->mtx); mutex_lock(&local->mtx);
WARN(test_bit(SCAN_HW_SCANNING, &local->scanning) || WARN(test_bit(SCAN_HW_SCANNING, &local->scanning) ||
local->sched_scanning, rcu_dereference_protected(local->sched_scan_sdata,
lockdep_is_held(&local->mtx)),
"%s called with hardware scan in progress\n", __func__); "%s called with hardware scan in progress\n", __func__);
mutex_unlock(&local->mtx); mutex_unlock(&local->mtx);
......
...@@ -421,13 +421,13 @@ ieee80211_rx_h_passive_scan(struct ieee80211_rx_data *rx) ...@@ -421,13 +421,13 @@ ieee80211_rx_h_passive_scan(struct ieee80211_rx_data *rx)
struct sk_buff *skb = rx->skb; struct sk_buff *skb = rx->skb;
if (likely(!(status->rx_flags & IEEE80211_RX_IN_SCAN) && if (likely(!(status->rx_flags & IEEE80211_RX_IN_SCAN) &&
!local->sched_scanning)) !rcu_access_pointer(local->sched_scan_sdata)))
return RX_CONTINUE; return RX_CONTINUE;
if (test_bit(SCAN_HW_SCANNING, &local->scanning) || if (test_bit(SCAN_HW_SCANNING, &local->scanning) ||
test_bit(SCAN_SW_SCANNING, &local->scanning) || test_bit(SCAN_SW_SCANNING, &local->scanning) ||
test_bit(SCAN_ONCHANNEL_SCANNING, &local->scanning) || test_bit(SCAN_ONCHANNEL_SCANNING, &local->scanning) ||
local->sched_scanning) rcu_access_pointer(local->sched_scan_sdata))
return ieee80211_scan_rx(rx->sdata, skb); return ieee80211_scan_rx(rx->sdata, skb);
/* scanning finished during invoking of handlers */ /* scanning finished during invoking of handlers */
......
...@@ -930,9 +930,9 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata, ...@@ -930,9 +930,9 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata,
struct ieee80211_local *local = sdata->local; struct ieee80211_local *local = sdata->local;
int ret, i; int ret, i;
mutex_lock(&sdata->local->mtx); mutex_lock(&local->mtx);
if (local->sched_scanning) { if (rcu_access_pointer(local->sched_scan_sdata)) {
ret = -EBUSY; ret = -EBUSY;
goto out; goto out;
} }
...@@ -966,7 +966,7 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata, ...@@ -966,7 +966,7 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata,
ret = drv_sched_scan_start(local, sdata, req, ret = drv_sched_scan_start(local, sdata, req,
&local->sched_scan_ies); &local->sched_scan_ies);
if (ret == 0) { if (ret == 0) {
local->sched_scanning = true; rcu_assign_pointer(local->sched_scan_sdata, sdata);
goto out; goto out;
} }
...@@ -974,7 +974,7 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata, ...@@ -974,7 +974,7 @@ int ieee80211_request_sched_scan_start(struct ieee80211_sub_if_data *sdata,
while (i > 0) while (i > 0)
kfree(local->sched_scan_ies.ie[--i]); kfree(local->sched_scan_ies.ie[--i]);
out: out:
mutex_unlock(&sdata->local->mtx); mutex_unlock(&local->mtx);
return ret; return ret;
} }
...@@ -983,22 +983,22 @@ int ieee80211_request_sched_scan_stop(struct ieee80211_sub_if_data *sdata) ...@@ -983,22 +983,22 @@ int ieee80211_request_sched_scan_stop(struct ieee80211_sub_if_data *sdata)
struct ieee80211_local *local = sdata->local; struct ieee80211_local *local = sdata->local;
int ret = 0, i; int ret = 0, i;
mutex_lock(&sdata->local->mtx); mutex_lock(&local->mtx);
if (!local->ops->sched_scan_stop) { if (!local->ops->sched_scan_stop) {
ret = -ENOTSUPP; ret = -ENOTSUPP;
goto out; goto out;
} }
if (local->sched_scanning) { if (rcu_access_pointer(local->sched_scan_sdata)) {
for (i = 0; i < IEEE80211_NUM_BANDS; i++) for (i = 0; i < IEEE80211_NUM_BANDS; i++)
kfree(local->sched_scan_ies.ie[i]); kfree(local->sched_scan_ies.ie[i]);
drv_sched_scan_stop(local, sdata); drv_sched_scan_stop(local, sdata);
local->sched_scanning = false; rcu_assign_pointer(local->sched_scan_sdata, NULL);
} }
out: out:
mutex_unlock(&sdata->local->mtx); mutex_unlock(&local->mtx);
return ret; return ret;
} }
...@@ -1022,7 +1022,7 @@ void ieee80211_sched_scan_stopped_work(struct work_struct *work) ...@@ -1022,7 +1022,7 @@ void ieee80211_sched_scan_stopped_work(struct work_struct *work)
mutex_lock(&local->mtx); mutex_lock(&local->mtx);
if (!local->sched_scanning) { if (!rcu_access_pointer(local->sched_scan_sdata)) {
mutex_unlock(&local->mtx); mutex_unlock(&local->mtx);
return; return;
} }
...@@ -1030,7 +1030,7 @@ void ieee80211_sched_scan_stopped_work(struct work_struct *work) ...@@ -1030,7 +1030,7 @@ void ieee80211_sched_scan_stopped_work(struct work_struct *work)
for (i = 0; i < IEEE80211_NUM_BANDS; i++) for (i = 0; i < IEEE80211_NUM_BANDS; i++)
kfree(local->sched_scan_ies.ie[i]); kfree(local->sched_scan_ies.ie[i]);
local->sched_scanning = false; rcu_assign_pointer(local->sched_scan_sdata, NULL);
mutex_unlock(&local->mtx); mutex_unlock(&local->mtx);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment