diff options
Diffstat (limited to 'net/wireless/reg.c')
-rw-r--r-- | net/wireless/reg.c | 63 |
1 files changed, 16 insertions, 47 deletions
diff --git a/net/wireless/reg.c b/net/wireless/reg.c index 038f8f133c54..dc10071deaaa 100644 --- a/net/wireless/reg.c +++ b/net/wireless/reg.c | |||
@@ -42,7 +42,10 @@ | |||
42 | #include "core.h" | 42 | #include "core.h" |
43 | #include "reg.h" | 43 | #include "reg.h" |
44 | 44 | ||
45 | /* wiphy is set if this request's initiator is REGDOM_SET_BY_DRIVER */ | 45 | /* |
46 | * wiphy is set if this request's initiator is | ||
47 | * REGDOM_SET_BY_COUNTRY_IE or _DRIVER | ||
48 | */ | ||
46 | struct regulatory_request { | 49 | struct regulatory_request { |
47 | struct wiphy *wiphy; | 50 | struct wiphy *wiphy; |
48 | enum reg_set_by initiator; | 51 | enum reg_set_by initiator; |
@@ -298,7 +301,7 @@ static int call_crda(const char *alpha2) | |||
298 | /* This has the logic which determines when a new request | 301 | /* This has the logic which determines when a new request |
299 | * should be ignored. */ | 302 | * should be ignored. */ |
300 | static int ignore_request(struct wiphy *wiphy, enum reg_set_by set_by, | 303 | static int ignore_request(struct wiphy *wiphy, enum reg_set_by set_by, |
301 | char *alpha2, struct ieee80211_regdomain *rd) | 304 | const char *alpha2) |
302 | { | 305 | { |
303 | /* All initial requests are respected */ | 306 | /* All initial requests are respected */ |
304 | if (!last_request) | 307 | if (!last_request) |
@@ -343,22 +346,8 @@ static int ignore_request(struct wiphy *wiphy, enum reg_set_by set_by, | |||
343 | return 1; | 346 | return 1; |
344 | case REGDOM_SET_BY_DRIVER: | 347 | case REGDOM_SET_BY_DRIVER: |
345 | BUG_ON(!wiphy); | 348 | BUG_ON(!wiphy); |
346 | if (last_request->initiator == REGDOM_SET_BY_DRIVER) { | 349 | if (last_request->initiator == REGDOM_SET_BY_DRIVER) |
347 | /* Two separate drivers hinting different things, | ||
348 | * this is possible if you have two devices present | ||
349 | * on a system with different EEPROM regulatory | ||
350 | * readings. XXX: Do intersection, we support only | ||
351 | * the first regulatory hint for now */ | ||
352 | if (last_request->wiphy != wiphy) | ||
353 | return -EALREADY; | ||
354 | if (rd) | ||
355 | return -EALREADY; | ||
356 | /* Driver should not be trying to hint different | ||
357 | * regulatory domains! */ | ||
358 | BUG_ON(!alpha2_equal(alpha2, | ||
359 | cfg80211_regdomain->alpha2)); | ||
360 | return -EALREADY; | 350 | return -EALREADY; |
361 | } | ||
362 | if (last_request->initiator == REGDOM_SET_BY_CORE) | 351 | if (last_request->initiator == REGDOM_SET_BY_CORE) |
363 | return 0; | 352 | return 0; |
364 | /* XXX: Handle intersection, and add the | 353 | /* XXX: Handle intersection, and add the |
@@ -557,40 +546,32 @@ void wiphy_update_regulatory(struct wiphy *wiphy, enum reg_set_by setby) | |||
557 | 546 | ||
558 | /* Caller must hold &cfg80211_drv_mutex */ | 547 | /* Caller must hold &cfg80211_drv_mutex */ |
559 | int __regulatory_hint(struct wiphy *wiphy, enum reg_set_by set_by, | 548 | int __regulatory_hint(struct wiphy *wiphy, enum reg_set_by set_by, |
560 | const char *alpha2, struct ieee80211_regdomain *rd) | 549 | const char *alpha2) |
561 | { | 550 | { |
562 | struct regulatory_request *request; | 551 | struct regulatory_request *request; |
563 | char *rd_alpha2; | ||
564 | int r = 0; | 552 | int r = 0; |
565 | 553 | ||
566 | r = ignore_request(wiphy, set_by, (char *) alpha2, rd); | 554 | r = ignore_request(wiphy, set_by, alpha2); |
567 | if (r) | 555 | if (r) |
568 | return r; | 556 | return r; |
569 | 557 | ||
570 | if (rd) | ||
571 | rd_alpha2 = rd->alpha2; | ||
572 | else | ||
573 | rd_alpha2 = (char *) alpha2; | ||
574 | |||
575 | switch (set_by) { | 558 | switch (set_by) { |
576 | case REGDOM_SET_BY_CORE: | 559 | case REGDOM_SET_BY_CORE: |
577 | case REGDOM_SET_BY_COUNTRY_IE: | 560 | case REGDOM_SET_BY_COUNTRY_IE: |
578 | case REGDOM_SET_BY_DRIVER: | 561 | case REGDOM_SET_BY_DRIVER: |
579 | case REGDOM_SET_BY_USER: | 562 | case REGDOM_SET_BY_USER: |
580 | request = kzalloc(sizeof(struct regulatory_request), | 563 | request = kzalloc(sizeof(struct regulatory_request), |
581 | GFP_KERNEL); | 564 | GFP_KERNEL); |
582 | if (!request) | 565 | if (!request) |
583 | return -ENOMEM; | 566 | return -ENOMEM; |
584 | 567 | ||
585 | request->alpha2[0] = rd_alpha2[0]; | 568 | request->alpha2[0] = alpha2[0]; |
586 | request->alpha2[1] = rd_alpha2[1]; | 569 | request->alpha2[1] = alpha2[1]; |
587 | request->initiator = set_by; | 570 | request->initiator = set_by; |
588 | request->wiphy = wiphy; | 571 | request->wiphy = wiphy; |
589 | 572 | ||
590 | kfree(last_request); | 573 | kfree(last_request); |
591 | last_request = request; | 574 | last_request = request; |
592 | if (rd) | ||
593 | break; | ||
594 | r = call_crda(alpha2); | 575 | r = call_crda(alpha2); |
595 | #ifndef CONFIG_WIRELESS_OLD_REGULATORY | 576 | #ifndef CONFIG_WIRELESS_OLD_REGULATORY |
596 | if (r) | 577 | if (r) |
@@ -605,25 +586,13 @@ int __regulatory_hint(struct wiphy *wiphy, enum reg_set_by set_by, | |||
605 | return r; | 586 | return r; |
606 | } | 587 | } |
607 | 588 | ||
608 | int regulatory_hint(struct wiphy *wiphy, const char *alpha2, | 589 | void regulatory_hint(struct wiphy *wiphy, const char *alpha2) |
609 | struct ieee80211_regdomain *rd) | ||
610 | { | 590 | { |
611 | int r; | 591 | BUG_ON(!alpha2); |
612 | BUG_ON(!rd && !alpha2); | ||
613 | 592 | ||
614 | mutex_lock(&cfg80211_drv_mutex); | 593 | mutex_lock(&cfg80211_drv_mutex); |
615 | 594 | __regulatory_hint(wiphy, REGDOM_SET_BY_DRIVER, alpha2); | |
616 | r = __regulatory_hint(wiphy, REGDOM_SET_BY_DRIVER, alpha2, rd); | ||
617 | if (r || !rd) | ||
618 | goto unlock_and_exit; | ||
619 | |||
620 | /* If the driver passed a regulatory domain we skipped asking | ||
621 | * userspace for one so we can now go ahead and set it */ | ||
622 | r = set_regdom(rd); | ||
623 | |||
624 | unlock_and_exit: | ||
625 | mutex_unlock(&cfg80211_drv_mutex); | 595 | mutex_unlock(&cfg80211_drv_mutex); |
626 | return r; | ||
627 | } | 596 | } |
628 | EXPORT_SYMBOL(regulatory_hint); | 597 | EXPORT_SYMBOL(regulatory_hint); |
629 | 598 | ||
@@ -792,11 +761,11 @@ int regulatory_init(void) | |||
792 | * that is not a valid ISO / IEC 3166 alpha2 */ | 761 | * that is not a valid ISO / IEC 3166 alpha2 */ |
793 | if (ieee80211_regdom[0] != 'E' || ieee80211_regdom[1] != 'U') | 762 | if (ieee80211_regdom[0] != 'E' || ieee80211_regdom[1] != 'U') |
794 | err = __regulatory_hint(NULL, REGDOM_SET_BY_CORE, | 763 | err = __regulatory_hint(NULL, REGDOM_SET_BY_CORE, |
795 | ieee80211_regdom, NULL); | 764 | ieee80211_regdom); |
796 | #else | 765 | #else |
797 | cfg80211_regdomain = cfg80211_world_regdom; | 766 | cfg80211_regdomain = cfg80211_world_regdom; |
798 | 767 | ||
799 | err = __regulatory_hint(NULL, REGDOM_SET_BY_CORE, "00", NULL); | 768 | err = __regulatory_hint(NULL, REGDOM_SET_BY_CORE, "00"); |
800 | if (err) | 769 | if (err) |
801 | printk(KERN_ERR "cfg80211: calling CRDA failed - " | 770 | printk(KERN_ERR "cfg80211: calling CRDA failed - " |
802 | "unable to update world regulatory domain, " | 771 | "unable to update world regulatory domain, " |