Commit 3ab8227d authored by Sergey Matyukevich's avatar Sergey Matyukevich Committed by Johannes Berg

cfg80211: refactor cfg80211_bss_update

This patch implements minor refactoring for cfg80211_bss_update function.
Code path for updating known BSS is extracted into dedicated
cfg80211_update_known_bss function.
Signed-off-by: default avatarSergey Matyukevich <sergey.matyukevich.os@quantenna.com>
Link: https://lore.kernel.org/r/20190726163922.27509-2-sergey.matyukevich.os@quantenna.comSigned-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent ae6fa4d5
...@@ -1091,111 +1091,118 @@ struct cfg80211_non_tx_bss { ...@@ -1091,111 +1091,118 @@ struct cfg80211_non_tx_bss {
u8 bssid_index; u8 bssid_index;
}; };
/* Returned bss is reference counted and must be cleaned up appropriately. */ static bool
struct cfg80211_internal_bss * cfg80211_update_known_bss(struct cfg80211_registered_device *rdev,
cfg80211_bss_update(struct cfg80211_registered_device *rdev, struct cfg80211_internal_bss *known,
struct cfg80211_internal_bss *tmp, struct cfg80211_internal_bss *new,
bool signal_valid, unsigned long ts) bool signal_valid)
{ {
struct cfg80211_internal_bss *found = NULL; lockdep_assert_held(&rdev->bss_lock);
if (WARN_ON(!tmp->pub.channel))
return NULL;
tmp->ts = ts;
spin_lock_bh(&rdev->bss_lock);
if (WARN_ON(!rcu_access_pointer(tmp->pub.ies))) {
spin_unlock_bh(&rdev->bss_lock);
return NULL;
}
found = rb_find_bss(rdev, tmp, BSS_CMP_REGULAR);
if (found) {
/* Update IEs */ /* Update IEs */
if (rcu_access_pointer(tmp->pub.proberesp_ies)) { if (rcu_access_pointer(new->pub.proberesp_ies)) {
const struct cfg80211_bss_ies *old; const struct cfg80211_bss_ies *old;
old = rcu_access_pointer(found->pub.proberesp_ies); old = rcu_access_pointer(known->pub.proberesp_ies);
rcu_assign_pointer(found->pub.proberesp_ies, rcu_assign_pointer(known->pub.proberesp_ies,
tmp->pub.proberesp_ies); new->pub.proberesp_ies);
/* Override possible earlier Beacon frame IEs */ /* Override possible earlier Beacon frame IEs */
rcu_assign_pointer(found->pub.ies, rcu_assign_pointer(known->pub.ies,
tmp->pub.proberesp_ies); new->pub.proberesp_ies);
if (old) if (old)
kfree_rcu((struct cfg80211_bss_ies *)old, kfree_rcu((struct cfg80211_bss_ies *)old, rcu_head);
rcu_head); } else if (rcu_access_pointer(new->pub.beacon_ies)) {
} else if (rcu_access_pointer(tmp->pub.beacon_ies)) {
const struct cfg80211_bss_ies *old; const struct cfg80211_bss_ies *old;
struct cfg80211_internal_bss *bss; struct cfg80211_internal_bss *bss;
if (found->pub.hidden_beacon_bss && if (known->pub.hidden_beacon_bss &&
!list_empty(&found->hidden_list)) { !list_empty(&known->hidden_list)) {
const struct cfg80211_bss_ies *f; const struct cfg80211_bss_ies *f;
/* /* The known BSS struct is one of the probe
* The found BSS struct is one of the probe
* response members of a group, but we're * response members of a group, but we're
* receiving a beacon (beacon_ies in the tmp * receiving a beacon (beacon_ies in the new
* bss is used). This can only mean that the * bss is used). This can only mean that the
* AP changed its beacon from not having an * AP changed its beacon from not having an
* SSID to showing it, which is confusing so * SSID to showing it, which is confusing so
* drop this information. * drop this information.
*/ */
f = rcu_access_pointer(tmp->pub.beacon_ies); f = rcu_access_pointer(new->pub.beacon_ies);
kfree_rcu((struct cfg80211_bss_ies *)f, kfree_rcu((struct cfg80211_bss_ies *)f, rcu_head);
rcu_head); return false;
goto drop;
} }
old = rcu_access_pointer(found->pub.beacon_ies); old = rcu_access_pointer(known->pub.beacon_ies);
rcu_assign_pointer(found->pub.beacon_ies, rcu_assign_pointer(known->pub.beacon_ies, new->pub.beacon_ies);
tmp->pub.beacon_ies);
/* Override IEs if they were from a beacon before */ /* Override IEs if they were from a beacon before */
if (old == rcu_access_pointer(found->pub.ies)) if (old == rcu_access_pointer(known->pub.ies))
rcu_assign_pointer(found->pub.ies, rcu_assign_pointer(known->pub.ies, new->pub.beacon_ies);
tmp->pub.beacon_ies);
/* Assign beacon IEs to all sub entries */ /* Assign beacon IEs to all sub entries */
list_for_each_entry(bss, &found->hidden_list, list_for_each_entry(bss, &known->hidden_list, hidden_list) {
hidden_list) {
const struct cfg80211_bss_ies *ies; const struct cfg80211_bss_ies *ies;
ies = rcu_access_pointer(bss->pub.beacon_ies); ies = rcu_access_pointer(bss->pub.beacon_ies);
WARN_ON(ies != old); WARN_ON(ies != old);
rcu_assign_pointer(bss->pub.beacon_ies, rcu_assign_pointer(bss->pub.beacon_ies,
tmp->pub.beacon_ies); new->pub.beacon_ies);
} }
if (old) if (old)
kfree_rcu((struct cfg80211_bss_ies *)old, kfree_rcu((struct cfg80211_bss_ies *)old, rcu_head);
rcu_head);
} }
found->pub.beacon_interval = tmp->pub.beacon_interval; known->pub.beacon_interval = new->pub.beacon_interval;
/*
* don't update the signal if beacon was heard on /* don't update the signal if beacon was heard on
* adjacent channel. * adjacent channel.
*/ */
if (signal_valid) if (signal_valid)
found->pub.signal = tmp->pub.signal; known->pub.signal = new->pub.signal;
found->pub.capability = tmp->pub.capability; known->pub.capability = new->pub.capability;
found->ts = tmp->ts; known->ts = new->ts;
found->ts_boottime = tmp->ts_boottime; known->ts_boottime = new->ts_boottime;
found->parent_tsf = tmp->parent_tsf; known->parent_tsf = new->parent_tsf;
found->pub.chains = tmp->pub.chains; known->pub.chains = new->pub.chains;
memcpy(found->pub.chain_signal, tmp->pub.chain_signal, memcpy(known->pub.chain_signal, new->pub.chain_signal,
IEEE80211_MAX_CHAINS); IEEE80211_MAX_CHAINS);
ether_addr_copy(found->parent_bssid, tmp->parent_bssid); ether_addr_copy(known->parent_bssid, new->parent_bssid);
found->pub.max_bssid_indicator = tmp->pub.max_bssid_indicator; known->pub.max_bssid_indicator = new->pub.max_bssid_indicator;
found->pub.bssid_index = tmp->pub.bssid_index; known->pub.bssid_index = new->pub.bssid_index;
return true;
}
/* Returned bss is reference counted and must be cleaned up appropriately. */
struct cfg80211_internal_bss *
cfg80211_bss_update(struct cfg80211_registered_device *rdev,
struct cfg80211_internal_bss *tmp,
bool signal_valid, unsigned long ts)
{
struct cfg80211_internal_bss *found = NULL;
if (WARN_ON(!tmp->pub.channel))
return NULL;
tmp->ts = ts;
spin_lock_bh(&rdev->bss_lock);
if (WARN_ON(!rcu_access_pointer(tmp->pub.ies))) {
spin_unlock_bh(&rdev->bss_lock);
return NULL;
}
found = rb_find_bss(rdev, tmp, BSS_CMP_REGULAR);
if (found) {
if (!cfg80211_update_known_bss(rdev, found, tmp, signal_valid))
goto drop;
} else { } else {
struct cfg80211_internal_bss *new; struct cfg80211_internal_bss *new;
struct cfg80211_internal_bss *hidden; struct cfg80211_internal_bss *hidden;
......
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