aboutsummaryrefslogtreecommitdiffstats
path: root/net/mac80211/ieee80211.c
diff options
context:
space:
mode:
authorJohannes Berg <johannes@sipsolutions.net>2007-09-26 09:19:47 -0400
committerDavid S. Miller <davem@sunset.davemloft.net>2007-10-10 19:53:16 -0400
commit53918994b7c8c3bf0af5f641e1f299856799d883 (patch)
tree63338872613d3c08473acfb46c8a57490f85350b /net/mac80211/ieee80211.c
parent50741ae05a4742cae99361f57d84b5f8d33822a4 (diff)
[PATCH] mac80211: fix iff_promiscs, iff_allmultis race
When we update the counters iff_promiscs and iff_allmultis in struct ieee80211_local we have no common lock held to protect them. The problem is that the update to each counter may not be atomic, so we could end up with iff_promiscs == -1 in unfortunate conditions. To fix it, use atomic_t values. It doesn't matter whether the two counters are updated together atomically or not, if there are two invocations of set_multicast_list we will end up with multiple configure_filter() invocations of which the latter will always be correct. Signed-off-by: Johannes Berg <johannes@sipsolutions.net> Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'net/mac80211/ieee80211.c')
-rw-r--r--net/mac80211/ieee80211.c12
1 files changed, 6 insertions, 6 deletions
diff --git a/net/mac80211/ieee80211.c b/net/mac80211/ieee80211.c
index b1180536c335..2501bff0d15e 100644
--- a/net/mac80211/ieee80211.c
+++ b/net/mac80211/ieee80211.c
@@ -59,10 +59,10 @@ static void ieee80211_configure_filter(struct ieee80211_local *local)
59 unsigned int changed_flags; 59 unsigned int changed_flags;
60 unsigned int new_flags = 0; 60 unsigned int new_flags = 0;
61 61
62 if (local->iff_promiscs) 62 if (atomic_read(&local->iff_promiscs))
63 new_flags |= FIF_PROMISC_IN_BSS; 63 new_flags |= FIF_PROMISC_IN_BSS;
64 64
65 if (local->iff_allmultis) 65 if (atomic_read(&local->iff_allmultis))
66 new_flags |= FIF_ALLMULTI; 66 new_flags |= FIF_ALLMULTI;
67 67
68 if (local->monitors) 68 if (local->monitors)
@@ -521,17 +521,17 @@ static void ieee80211_set_multicast_list(struct net_device *dev)
521 521
522 if (allmulti != sdata_allmulti) { 522 if (allmulti != sdata_allmulti) {
523 if (dev->flags & IFF_ALLMULTI) 523 if (dev->flags & IFF_ALLMULTI)
524 local->iff_allmultis++; 524 atomic_inc(&local->iff_allmultis);
525 else 525 else
526 local->iff_allmultis--; 526 atomic_dec(&local->iff_allmultis);
527 sdata->flags ^= IEEE80211_SDATA_ALLMULTI; 527 sdata->flags ^= IEEE80211_SDATA_ALLMULTI;
528 } 528 }
529 529
530 if (promisc != sdata_promisc) { 530 if (promisc != sdata_promisc) {
531 if (dev->flags & IFF_PROMISC) 531 if (dev->flags & IFF_PROMISC)
532 local->iff_promiscs++; 532 atomic_inc(&local->iff_promiscs);
533 else 533 else
534 local->iff_promiscs--; 534 atomic_dec(&local->iff_promiscs);
535 sdata->flags ^= IEEE80211_SDATA_PROMISC; 535 sdata->flags ^= IEEE80211_SDATA_PROMISC;
536 } 536 }
537 537