iwlwifi: mvm: continue (with error) CSA on GO time event failure
authorJohannes Berg <johannes.berg@intel.com>
Tue, 10 Mar 2015 13:44:00 +0000 (14:44 +0100)
committerEmmanuel Grumbach <emmanuel.grumbach@intel.com>
Wed, 18 Mar 2015 06:40:02 +0000 (08:40 +0200)
If, on a GO, the CSA time event fails to be scheduled, continue the
flow towards mac80211's state machine so it doesn't get stuck, but
report an error later on the post switch which will cause mac80211
to tear down the operation. This ensures nothing gets stuck due to
the scheduling failure.

Signed-off-by: Johannes Berg <johannes.berg@intel.com>
Signed-off-by: Emmanuel Grumbach <emmanuel.grumbach@intel.com>
drivers/net/wireless/iwlwifi/mvm/mac80211.c
drivers/net/wireless/iwlwifi/mvm/mvm.h
drivers/net/wireless/iwlwifi/mvm/time-event.c

index 302c8cc50f25a695c40ce59555f8a05bf58cae2d..5f26557e4197cb6c78e9799fb6a27482cd8491c0 100644 (file)
@@ -3646,6 +3646,8 @@ static int iwl_mvm_pre_channel_switch(struct ieee80211_hw *hw,
 
        mutex_lock(&mvm->mutex);
 
+       mvmvif->csa_failed = false;
+
        IWL_DEBUG_MAC80211(mvm, "pre CSA to freq %d\n",
                           chsw->chandef.center_freq1);
 
@@ -3721,6 +3723,12 @@ static int iwl_mvm_post_channel_switch(struct ieee80211_hw *hw,
 
        mutex_lock(&mvm->mutex);
 
+       if (mvmvif->csa_failed) {
+               mvmvif->csa_failed = false;
+               ret = -EIO;
+               goto out_unlock;
+       }
+
        if (vif->type == NL80211_IFTYPE_STATION) {
                struct iwl_mvm_sta *mvmsta;
 
index 432265eb0dd71dfac6d19296811416a62d5835de..4903dec7048a64e2aaecf9d4549a3c665bf867c4 100644 (file)
@@ -354,6 +354,7 @@ struct iwl_mvm_vif_bf_data {
  * @beacon_stats: beacon statistics, containing the # of received beacons,
  *     # of received beacons accumulated over FW restart, and the current
  *     average signal of beacons retrieved from the firmware
+ * @csa_failed: CSA failed to schedule time event, report an error later
  */
 struct iwl_mvm_vif {
        struct iwl_mvm *mvm;
@@ -433,6 +434,7 @@ struct iwl_mvm_vif {
 
        /* Indicates that CSA countdown may be started */
        bool csa_countdown;
+       bool csa_failed;
 };
 
 static inline struct iwl_mvm_vif *
index 8d179ab67cc237026e8653d18e791d826d9ee0e4..98c8d70c90d2b8e6f6a5f48daae01280e44165bf 100644 (file)
@@ -196,19 +196,23 @@ iwl_mvm_te_handle_notify_csa(struct iwl_mvm *mvm,
                             struct iwl_mvm_time_event_data *te_data,
                             struct iwl_time_event_notif *notif)
 {
-       if (!le32_to_cpu(notif->status)) {
-               if (te_data->vif->type == NL80211_IFTYPE_STATION)
-                       ieee80211_connection_loss(te_data->vif);
+       struct ieee80211_vif *vif = te_data->vif;
+       struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
+
+       if (!notif->status)
                IWL_DEBUG_TE(mvm, "CSA time event failed to start\n");
-               iwl_mvm_te_clear_data(mvm, te_data);
-               return;
-       }
 
        switch (te_data->vif->type) {
        case NL80211_IFTYPE_AP:
+               if (!notif->status)
+                       mvmvif->csa_failed = true;
                iwl_mvm_csa_noa_start(mvm);
                break;
        case NL80211_IFTYPE_STATION:
+               if (!notif->status) {
+                       ieee80211_connection_loss(te_data->vif);
+                       break;
+               }
                iwl_mvm_csa_client_absent(mvm, te_data->vif);
                ieee80211_chswitch_done(te_data->vif, true);
                break;