aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDaniel Drake <dsd@laptop.org>2012-08-02 13:41:48 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2012-08-15 15:04:30 -0400
commitb27c59d2c23d5f326a74e02a799cc4d00246165a (patch)
treedb452bc310e388ecea4402710bb0ca2143de286f
parent9f75ebd871f7f0a613fdb4e1231fbd540916872c (diff)
cfg80211: process pending events when unregistering net device
commit 1f6fc43e621167492ed4b7f3b4269c584c3d6ccc upstream. libertas currently calls cfg80211_disconnected() when it is being brought down. This causes an event to be allocated, but since the wdev is already removed from the rdev by the time that the event processing work executes, the event is never processed or freed. http://article.gmane.org/gmane.linux.kernel.wireless.general/95666 Fix this leak, and other possible situations, by processing the event queue when a device is being unregistered. Thanks to Johannes Berg for the suggestion. Signed-off-by: Daniel Drake <dsd@laptop.org> Reviewed-by: Johannes Berg <johannes@sipsolutions.net> Signed-off-by: John W. Linville <linville@tuxdriver.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--net/wireless/core.c5
-rw-r--r--net/wireless/core.h1
-rw-r--r--net/wireless/util.c2
3 files changed, 7 insertions, 1 deletions
diff --git a/net/wireless/core.c b/net/wireless/core.c
index 880dbe2e6f9..498c760a1d2 100644
--- a/net/wireless/core.c
+++ b/net/wireless/core.c
@@ -959,6 +959,11 @@ static int cfg80211_netdev_notifier_call(struct notifier_block * nb,
959 */ 959 */
960 synchronize_rcu(); 960 synchronize_rcu();
961 INIT_LIST_HEAD(&wdev->list); 961 INIT_LIST_HEAD(&wdev->list);
962 /*
963 * Ensure that all events have been processed and
964 * freed.
965 */
966 cfg80211_process_wdev_events(wdev);
962 break; 967 break;
963 case NETDEV_PRE_UP: 968 case NETDEV_PRE_UP:
964 if (!(wdev->wiphy->interface_modes & BIT(wdev->iftype))) 969 if (!(wdev->wiphy->interface_modes & BIT(wdev->iftype)))
diff --git a/net/wireless/core.h b/net/wireless/core.h
index a570ff9214e..83516455a1d 100644
--- a/net/wireless/core.h
+++ b/net/wireless/core.h
@@ -426,6 +426,7 @@ int cfg80211_change_iface(struct cfg80211_registered_device *rdev,
426 struct net_device *dev, enum nl80211_iftype ntype, 426 struct net_device *dev, enum nl80211_iftype ntype,
427 u32 *flags, struct vif_params *params); 427 u32 *flags, struct vif_params *params);
428void cfg80211_process_rdev_events(struct cfg80211_registered_device *rdev); 428void cfg80211_process_rdev_events(struct cfg80211_registered_device *rdev);
429void cfg80211_process_wdev_events(struct wireless_dev *wdev);
429 430
430int cfg80211_can_change_interface(struct cfg80211_registered_device *rdev, 431int cfg80211_can_change_interface(struct cfg80211_registered_device *rdev,
431 struct wireless_dev *wdev, 432 struct wireless_dev *wdev,
diff --git a/net/wireless/util.c b/net/wireless/util.c
index bbcb58e61ed..4d69149eef2 100644
--- a/net/wireless/util.c
+++ b/net/wireless/util.c
@@ -719,7 +719,7 @@ void cfg80211_upload_connect_keys(struct wireless_dev *wdev)
719 wdev->connect_keys = NULL; 719 wdev->connect_keys = NULL;
720} 720}
721 721
722static void cfg80211_process_wdev_events(struct wireless_dev *wdev) 722void cfg80211_process_wdev_events(struct wireless_dev *wdev)
723{ 723{
724 struct cfg80211_event *ev; 724 struct cfg80211_event *ev;
725 unsigned long flags; 725 unsigned long flags;