diff options
Diffstat (limited to 'net/mac80211/mlme.c')
| -rw-r--r-- | net/mac80211/mlme.c | 28 |
1 files changed, 25 insertions, 3 deletions
diff --git a/net/mac80211/mlme.c b/net/mac80211/mlme.c index 6b75cb6c6300..a5e5c31c23ab 100644 --- a/net/mac80211/mlme.c +++ b/net/mac80211/mlme.c | |||
| @@ -2248,10 +2248,13 @@ static void ieee80211_rx_bss_put(struct net_device *dev, | |||
| 2248 | struct ieee80211_sta_bss *bss) | 2248 | struct ieee80211_sta_bss *bss) |
| 2249 | { | 2249 | { |
| 2250 | struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); | 2250 | struct ieee80211_local *local = wdev_priv(dev->ieee80211_ptr); |
| 2251 | if (!atomic_dec_and_test(&bss->users)) | 2251 | |
| 2252 | local_bh_disable(); | ||
| 2253 | if (!atomic_dec_and_lock(&bss->users, &local->sta_bss_lock)) { | ||
| 2254 | local_bh_enable(); | ||
| 2252 | return; | 2255 | return; |
| 2256 | } | ||
| 2253 | 2257 | ||
| 2254 | spin_lock_bh(&local->sta_bss_lock); | ||
| 2255 | __ieee80211_rx_bss_hash_del(dev, bss); | 2258 | __ieee80211_rx_bss_hash_del(dev, bss); |
| 2256 | list_del(&bss->list); | 2259 | list_del(&bss->list); |
| 2257 | spin_unlock_bh(&local->sta_bss_lock); | 2260 | spin_unlock_bh(&local->sta_bss_lock); |
| @@ -2709,7 +2712,26 @@ static void ieee80211_rx_bss_info(struct net_device *dev, | |||
| 2709 | bss->wmm_ie_len = elems.wmm_param_len + 2; | 2712 | bss->wmm_ie_len = elems.wmm_param_len + 2; |
| 2710 | } else | 2713 | } else |
| 2711 | bss->wmm_ie_len = 0; | 2714 | bss->wmm_ie_len = 0; |
| 2712 | } else if (!elems.wmm_param && bss->wmm_ie) { | 2715 | } else if (elems.wmm_info && |
| 2716 | (!bss->wmm_ie || bss->wmm_ie_len != elems.wmm_info_len || | ||
| 2717 | memcmp(bss->wmm_ie, elems.wmm_info, elems.wmm_info_len))) { | ||
| 2718 | /* As for certain AP's Fifth bit is not set in WMM IE in | ||
| 2719 | * beacon frames.So while parsing the beacon frame the | ||
| 2720 | * wmm_info structure is used instead of wmm_param. | ||
| 2721 | * wmm_info structure was never used to set bss->wmm_ie. | ||
| 2722 | * This code fixes this problem by copying the WME | ||
| 2723 | * information from wmm_info to bss->wmm_ie and enabling | ||
| 2724 | * n-band association. | ||
| 2725 | */ | ||
| 2726 | kfree(bss->wmm_ie); | ||
| 2727 | bss->wmm_ie = kmalloc(elems.wmm_info_len + 2, GFP_ATOMIC); | ||
| 2728 | if (bss->wmm_ie) { | ||
| 2729 | memcpy(bss->wmm_ie, elems.wmm_info - 2, | ||
| 2730 | elems.wmm_info_len + 2); | ||
| 2731 | bss->wmm_ie_len = elems.wmm_info_len + 2; | ||
| 2732 | } else | ||
| 2733 | bss->wmm_ie_len = 0; | ||
| 2734 | } else if (!elems.wmm_param && !elems.wmm_info && bss->wmm_ie) { | ||
| 2713 | kfree(bss->wmm_ie); | 2735 | kfree(bss->wmm_ie); |
| 2714 | bss->wmm_ie = NULL; | 2736 | bss->wmm_ie = NULL; |
| 2715 | bss->wmm_ie_len = 0; | 2737 | bss->wmm_ie_len = 0; |
