diff options
author | Johannes Berg <johannes@sipsolutions.net> | 2007-09-26 09:19:47 -0400 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2007-10-10 19:53:16 -0400 |
commit | 53918994b7c8c3bf0af5f641e1f299856799d883 (patch) | |
tree | 63338872613d3c08473acfb46c8a57490f85350b /net/mac80211/ieee80211.c | |
parent | 50741ae05a4742cae99361f57d84b5f8d33822a4 (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.c | 12 |
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 | ||