mac80211: destroy virtual monitor interface across suspend
authorJohannes Berg <johannes.berg@intel.com>
Wed, 27 Mar 2013 22:20:27 +0000 (23:20 +0100)
committerJohannes Berg <johannes.berg@intel.com>
Mon, 8 Apr 2013 07:17:00 +0000 (09:17 +0200)
It has to be removed from the driver, but completely
destroying it helps handle unplug of a device during
suspend since then the channel context handling etc.
doesn't have to happen later when it's removed.

Signed-off-by: Johannes Berg <johannes.berg@intel.com>
net/mac80211/ieee80211_i.h
net/mac80211/iface.c
net/mac80211/pm.c
net/mac80211/util.c

index 693c1812b7bc945c16e852385a1c1df9d227aa91..55fb382a81998efb36e6a4e5703955b412a193cd 100644 (file)
@@ -1336,6 +1336,8 @@ void ieee80211_adjust_monitor_flags(struct ieee80211_sub_if_data *sdata,
                                    const int offset);
 int ieee80211_do_open(struct wireless_dev *wdev, bool coming_up);
 void ieee80211_sdata_stop(struct ieee80211_sub_if_data *sdata);
+int ieee80211_add_virtual_monitor(struct ieee80211_local *local);
+void ieee80211_del_virtual_monitor(struct ieee80211_local *local);
 
 bool __ieee80211_recalc_txpower(struct ieee80211_sub_if_data *sdata);
 void ieee80211_recalc_txpower(struct ieee80211_sub_if_data *sdata);
index 75b322f9d9450e0ceeb5ceed5a51b6f720585cc1..d0d5f20f1ec40dee41ed27aec1575bb015969c28 100644 (file)
@@ -346,7 +346,7 @@ static void ieee80211_set_default_queues(struct ieee80211_sub_if_data *sdata)
        sdata->vif.cab_queue = IEEE80211_INVAL_HW_QUEUE;
 }
 
-static int ieee80211_add_virtual_monitor(struct ieee80211_local *local)
+int ieee80211_add_virtual_monitor(struct ieee80211_local *local)
 {
        struct ieee80211_sub_if_data *sdata;
        int ret = 0;
@@ -400,7 +400,7 @@ static int ieee80211_add_virtual_monitor(struct ieee80211_local *local)
        return ret;
 }
 
-static void ieee80211_del_virtual_monitor(struct ieee80211_local *local)
+void ieee80211_del_virtual_monitor(struct ieee80211_local *local)
 {
        struct ieee80211_sub_if_data *sdata;
 
index b98d927dd0f38a47a01b4d4e7838ba1f11be6163..d1c021b62fe5abbe1609f25546895f5db0ed91bc 100644 (file)
@@ -21,6 +21,8 @@ int __ieee80211_suspend(struct ieee80211_hw *hw, struct cfg80211_wowlan *wowlan)
 
        ieee80211_roc_purge(local, NULL);
 
+       ieee80211_del_virtual_monitor(local);
+
        if (hw->flags & IEEE80211_HW_AMPDU_AGGREGATION) {
                mutex_lock(&local->sta_mtx);
                list_for_each_entry(sta, &local->sta_list, list) {
@@ -103,10 +105,6 @@ int __ieee80211_suspend(struct ieee80211_hw *hw, struct cfg80211_wowlan *wowlan)
                drv_remove_interface(local, sdata);
        }
 
-       sdata = rtnl_dereference(local->monitor_sdata);
-       if (sdata)
-               drv_remove_interface(local, sdata);
-
        /*
         * We disconnected on all interfaces before suspend, all channel
         * contexts should be released.
index f9581c6378aeb8317549ccc8d2fbc19bda37e08f..43465b6e477805d8760ca9a27daeec056ffc6f69 100644 (file)
@@ -1461,6 +1461,8 @@ int ieee80211_reconfig(struct ieee80211_local *local)
        /* add interfaces */
        sdata = rtnl_dereference(local->monitor_sdata);
        if (sdata) {
+               /* in HW restart it exists already */
+               WARN_ON(local->resuming);
                res = drv_add_interface(local, sdata);
                if (WARN_ON(res)) {
                        rcu_assign_pointer(local->monitor_sdata, NULL);
@@ -1650,6 +1652,9 @@ int ieee80211_reconfig(struct ieee80211_local *local)
        local->in_reconfig = false;
        barrier();
 
+       if (local->monitors == local->open_count && local->monitors > 0)
+               ieee80211_add_virtual_monitor(local);
+
        /*
         * Clear the WLAN_STA_BLOCK_BA flag so new aggregation
         * sessions can be established after a resume.