diff options
author | Igor Perminov <igor.perminov@inbox.ru> | 2009-08-08 17:55:18 -0400 |
---|---|---|
committer | John W. Linville <linville@tuxdriver.com> | 2009-08-14 09:13:53 -0400 |
commit | 17512dc3b7fc9ff1a60d3748ce87c323df507c3d (patch) | |
tree | 1ad4c98c3c653571552156116ce639c6e0aa2789 /drivers/net/wireless/rt2x00/rt2x00dev.c | |
parent | 66679a65efffffb62dc2650960b3aff758d575f9 (diff) |
rt2x00: Fix for race condition while update beacon
The patch "Implement set_tim callback for all drivers" can cause kernel
oops in rt73usb_write_beacon. The oops is caused by one of the following
race conditions:
* In case of two near calls to set_tim: rt2x00lib_beacondone_iter is
cleaning the beacon skb, whereas rt73usb_write_beacon is still using it.
* In case of two near updates of beacon: first as the result of set_tim
and second as the result of a call from an application (e.g. hostapd).
This patch fixes the race condition by rearranging the update logic and
guarding rt2x00_intf->beacon->skb with a mutex.
Signed-off-by: Igor Perminov <igor.perminov@inbox.ru>
Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'drivers/net/wireless/rt2x00/rt2x00dev.c')
-rw-r--r-- | drivers/net/wireless/rt2x00/rt2x00dev.c | 7 |
1 files changed, 0 insertions, 7 deletions
diff --git a/drivers/net/wireless/rt2x00/rt2x00dev.c b/drivers/net/wireless/rt2x00/rt2x00dev.c index e0348cc22d16..b6676c6722fc 100644 --- a/drivers/net/wireless/rt2x00/rt2x00dev.c +++ b/drivers/net/wireless/rt2x00/rt2x00dev.c | |||
@@ -186,7 +186,6 @@ static void rt2x00lib_intf_scheduled(struct work_struct *work) | |||
186 | static void rt2x00lib_beacondone_iter(void *data, u8 *mac, | 186 | static void rt2x00lib_beacondone_iter(void *data, u8 *mac, |
187 | struct ieee80211_vif *vif) | 187 | struct ieee80211_vif *vif) |
188 | { | 188 | { |
189 | struct rt2x00_dev *rt2x00dev = data; | ||
190 | struct rt2x00_intf *intf = vif_to_intf(vif); | 189 | struct rt2x00_intf *intf = vif_to_intf(vif); |
191 | 190 | ||
192 | if (vif->type != NL80211_IFTYPE_AP && | 191 | if (vif->type != NL80211_IFTYPE_AP && |
@@ -195,12 +194,6 @@ static void rt2x00lib_beacondone_iter(void *data, u8 *mac, | |||
195 | vif->type != NL80211_IFTYPE_WDS) | 194 | vif->type != NL80211_IFTYPE_WDS) |
196 | return; | 195 | return; |
197 | 196 | ||
198 | /* | ||
199 | * Clean up the beacon skb. | ||
200 | */ | ||
201 | rt2x00queue_free_skb(rt2x00dev, intf->beacon->skb); | ||
202 | intf->beacon->skb = NULL; | ||
203 | |||
204 | spin_lock(&intf->lock); | 197 | spin_lock(&intf->lock); |
205 | intf->delayed_flags |= DELAYED_UPDATE_BEACON; | 198 | intf->delayed_flags |= DELAYED_UPDATE_BEACON; |
206 | spin_unlock(&intf->lock); | 199 | spin_unlock(&intf->lock); |