Commit b2e253cf authored by Luis R. Rodriguez's avatar Luis R. Rodriguez Committed by John W. Linville

cfg80211: Fix regulatory bug with multiple cards and delays

When two cards are connected with the same regulatory domain
if CRDA had a delayed response then cfg80211's own set regulatory
domain would still be the world regulatory domain. There was a bug
on cfg80211's logic such that it assumed that once you pegged a
request as the last request it was already the currently set
regulatory domain. This would mean we would race setting a stale
regulatory domain to secondary cards which had the same regulatory
domain since the alpha2 would match.

We fix this by processing each regulatory request atomically,
and only move on to the next one once we get it fully processed.
In the case CRDA is not present we will simply world roam.

This issue is only present when you have a slow system and the
CRDA processing is delayed. Because of this it is not a known
regression.

Without this fix when a delay is present with CRDA the second card
would end up with an intersected regulatory domain and not allow it
to use the channels it really is designed for. When two cards with
two different regulatory domains were inserted you'd end up
rejecting the second card's regulatory domain request.
This fails with mac80211_hswim's regtest=2 (two requests, same alpha2)
and regtest=3 (two requests, different alpha2) module parameter
options.

This was reproduced and tested against mac80211_hwsim using this
CRDA delayer:

       #!/bin/bash
       echo $COUNTRY >> /tmp/log
       sleep 2
       /sbin/crda.orig

And these regulatory tests:

       modprobe mac80211_hwsim regtest=2
       modprobe mac80211_hwsim regtest=3
Reported-by: default avatarMark Mentovai <mark@moxienet.com>
Signed-off-by: default avatarLuis R. Rodriguez <lrodriguez@atheros.com>
Tested-by: default avatarMark Mentovai <mark@moxienet.com>
Tested-by: default avatarBruno Randolf <br1@einfach.org>
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parent b0e2880b
...@@ -43,6 +43,12 @@ enum environment_cap { ...@@ -43,6 +43,12 @@ enum environment_cap {
* @intersect: indicates whether the wireless core should intersect * @intersect: indicates whether the wireless core should intersect
* the requested regulatory domain with the presently set regulatory * the requested regulatory domain with the presently set regulatory
* domain. * domain.
* @processed: indicates whether or not this requests has already been
* processed. When the last request is processed it means that the
* currently regulatory domain set on cfg80211 is updated from
* CRDA and can be used by other regulatory requests. When a
* the last request is not yet processed we must yield until it
* is processed before processing any new requests.
* @country_ie_checksum: checksum of the last processed and accepted * @country_ie_checksum: checksum of the last processed and accepted
* country IE * country IE
* @country_ie_env: lets us know if the AP is telling us we are outdoor, * @country_ie_env: lets us know if the AP is telling us we are outdoor,
...@@ -54,6 +60,7 @@ struct regulatory_request { ...@@ -54,6 +60,7 @@ struct regulatory_request {
enum nl80211_reg_initiator initiator; enum nl80211_reg_initiator initiator;
char alpha2[2]; char alpha2[2];
bool intersect; bool intersect;
bool processed;
enum environment_cap country_ie_env; enum environment_cap country_ie_env;
struct list_head list; struct list_head list;
}; };
......
...@@ -1320,6 +1320,21 @@ static int ignore_request(struct wiphy *wiphy, ...@@ -1320,6 +1320,21 @@ static int ignore_request(struct wiphy *wiphy,
return -EINVAL; return -EINVAL;
} }
static void reg_set_request_processed(void)
{
bool need_more_processing = false;
last_request->processed = true;
spin_lock(&reg_requests_lock);
if (!list_empty(&reg_requests_list))
need_more_processing = true;
spin_unlock(&reg_requests_lock);
if (need_more_processing)
schedule_work(&reg_work);
}
/** /**
* __regulatory_hint - hint to the wireless core a regulatory domain * __regulatory_hint - hint to the wireless core a regulatory domain
* @wiphy: if the hint comes from country information from an AP, this * @wiphy: if the hint comes from country information from an AP, this
...@@ -1395,8 +1410,10 @@ static int __regulatory_hint(struct wiphy *wiphy, ...@@ -1395,8 +1410,10 @@ static int __regulatory_hint(struct wiphy *wiphy,
* have applied the requested regulatory domain before we just * have applied the requested regulatory domain before we just
* inform userspace we have processed the request * inform userspace we have processed the request
*/ */
if (r == -EALREADY) if (r == -EALREADY) {
nl80211_send_reg_change_event(last_request); nl80211_send_reg_change_event(last_request);
reg_set_request_processed();
}
return r; return r;
} }
...@@ -1428,7 +1445,11 @@ static void reg_process_hint(struct regulatory_request *reg_request) ...@@ -1428,7 +1445,11 @@ static void reg_process_hint(struct regulatory_request *reg_request)
wiphy_update_regulatory(wiphy, initiator); wiphy_update_regulatory(wiphy, initiator);
} }
/* Processes regulatory hints, this is all the NL80211_REGDOM_SET_BY_* */ /*
* Processes regulatory hints, this is all the NL80211_REGDOM_SET_BY_*
* Regulatory hints come on a first come first serve basis and we
* must process each one atomically.
*/
static void reg_process_pending_hints(void) static void reg_process_pending_hints(void)
{ {
struct regulatory_request *reg_request; struct regulatory_request *reg_request;
...@@ -1436,19 +1457,30 @@ static void reg_process_pending_hints(void) ...@@ -1436,19 +1457,30 @@ static void reg_process_pending_hints(void)
mutex_lock(&cfg80211_mutex); mutex_lock(&cfg80211_mutex);
mutex_lock(&reg_mutex); mutex_lock(&reg_mutex);
/* When last_request->processed becomes true this will be rescheduled */
if (last_request && !last_request->processed) {
REG_DBG_PRINT("Pending regulatory request, waiting "
"for it to be processed...");
goto out;
}
spin_lock(&reg_requests_lock); spin_lock(&reg_requests_lock);
while (!list_empty(&reg_requests_list)) {
reg_request = list_first_entry(&reg_requests_list,
struct regulatory_request,
list);
list_del_init(&reg_request->list);
if (list_empty(&reg_requests_list)) {
spin_unlock(&reg_requests_lock); spin_unlock(&reg_requests_lock);
reg_process_hint(reg_request); goto out;
spin_lock(&reg_requests_lock);
} }
reg_request = list_first_entry(&reg_requests_list,
struct regulatory_request,
list);
list_del_init(&reg_request->list);
spin_unlock(&reg_requests_lock); spin_unlock(&reg_requests_lock);
reg_process_hint(reg_request);
out:
mutex_unlock(&reg_mutex); mutex_unlock(&reg_mutex);
mutex_unlock(&cfg80211_mutex); mutex_unlock(&cfg80211_mutex);
} }
...@@ -2057,6 +2089,8 @@ int set_regdom(const struct ieee80211_regdomain *rd) ...@@ -2057,6 +2089,8 @@ int set_regdom(const struct ieee80211_regdomain *rd)
nl80211_send_reg_change_event(last_request); nl80211_send_reg_change_event(last_request);
reg_set_request_processed();
mutex_unlock(&reg_mutex); mutex_unlock(&reg_mutex);
return r; return r;
......
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