Currently, the sequence goes like this (among others):
1. flush all stations (including the AP ones) -> this will tell the
   drivers to remove the stations
2. notify the driver the vif is not associated.
Which means that in between 1 and 2, the state is that the vif is
associated, but there is no AP station, which makes no sense, and may be
problematic for some drivers (for example iwlwifi)
Change the sequence to:
1. flush the TDLS stations
2. move the AP station to IEEE80211_STA_NONE
3. notify the driver about the vif being unassociated
4. flush the AP station
In order to not break other drivers, add a vif flag to indicate whether
the driver wants to new sequence or not. If the flag is not set, then
things will be done in the old sequence.
Signed-off-by: Miri Korenblit <miriam.rachel.korenblit@intel.com>
Reviewed-by: Johannes Berg <johannes.berg@intel.com>
Link: https://patch.msgid.link/20241224192322.996ad1be6cb3.I7815d33415aa1d65c0120b54be7a15a45388f807@changeid
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
        mvmvif->deflink.active = 0;
        mvmvif->link[0] = &mvmvif->deflink;
 
+       vif->driver_flags = IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC;
+
        ret = iwl_mvm_set_link_mapping(mvm, vif, &vif->bss_conf);
        if (ret)
                goto out;
                                            &mvm->status),
                                  "Failed to update SF upon disassociation\n");
 
-                       /*
-                        * If we get an assert during the connection (after the
-                        * station has been added, but before the vif is set
-                        * to associated), mac80211 will re-add the station and
-                        * then configure the vif. Since the vif is not
-                        * associated, we would remove the station here and
-                        * this would fail the recovery.
-                        */
-                       if (!test_bit(IWL_MVM_STATUS_IN_HW_RESTART,
-                                     &mvm->status)) {
-                               /* first remove remaining keys */
-                               iwl_mvm_sec_key_remove_ap(mvm, vif,
-                                                         &mvmvif->deflink, 0);
-
-                               /*
-                                * Remove AP station now that
-                                * the MAC is unassoc
-                                */
-                               ret = iwl_mvm_rm_sta_id(mvm, vif,
-                                                       mvmvif->deflink.ap_sta_id);
-                               if (ret)
-                                       IWL_ERR(mvm,
-                                               "failed to remove AP station\n");
-
-                               mvmvif->deflink.ap_sta_id = IWL_INVALID_STA;
-                       }
-
                        /* remove quota for this interface */
                        ret = iwl_mvm_update_quotas(mvm, false, NULL);
                        if (ret)
 
 
        mvmvif->mvm = mvm;
 
+       vif->driver_flags |= IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC;
+
        /* Not much to do here. The stack will not allow interface
         * types or combinations that we didn't advertise, so we
         * don't really have to check the types.
        return false;
 }
 
-static void iwl_mvm_mld_vif_delete_all_stas(struct iwl_mvm *mvm,
-                                           struct ieee80211_vif *vif)
-{
-       struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
-       int i, ret;
-
-       if (test_bit(IWL_MVM_STATUS_IN_HW_RESTART, &mvm->status))
-               return;
-
-       for_each_mvm_vif_valid_link(mvmvif, i) {
-               struct iwl_mvm_vif_link_info *link = mvmvif->link[i];
-
-               if (!link)
-                       continue;
-
-               iwl_mvm_sec_key_remove_ap(mvm, vif, link, i);
-               ret = iwl_mvm_mld_rm_sta_id(mvm, link->ap_sta_id);
-               if (ret)
-                       IWL_ERR(mvm, "failed to remove AP station\n");
-
-               link->ap_sta_id = IWL_INVALID_STA;
-       }
-}
-
 static void iwl_mvm_mld_vif_cfg_changed_station(struct iwl_mvm *mvm,
                                                struct ieee80211_vif *vif,
                                                u64 changes)
                                  !test_bit(IWL_MVM_STATUS_HW_RESTART_REQUESTED,
                                            &mvm->status),
                                  "Failed to update SF upon disassociation\n");
-
-                       /* If we get an assert during the connection (after the
-                        * station has been added, but before the vif is set
-                        * to associated), mac80211 will re-add the station and
-                        * then configure the vif. Since the vif is not
-                        * associated, we would remove the station here and
-                        * this would fail the recovery.
-                        */
-                       iwl_mvm_mld_vif_delete_all_stas(mvm, vif);
                }
 
                iwl_mvm_bss_info_changed_station_assoc(mvm, vif, changes);
 
 
        if (vif->type == NL80211_IFTYPE_STATION &&
            mvm_link->ap_sta_id == sta_id) {
-               /* if associated - we can't remove the AP STA now */
-               if (vif->cfg.assoc)
-                       return true;
-
                /* first remove remaining keys */
                iwl_mvm_sec_key_remove_ap(mvm, vif, mvm_link,
                                          link_sta->link_id);
 
-               /* unassoc - go ahead - remove the AP STA now */
                mvm_link->ap_sta_id = IWL_INVALID_STA;
        }
 
 
  *     operation on this interface and request a channel context without
  *     the AP definition. Use this e.g. because the device is able to
  *     handle OFDMA (downlink and trigger for uplink) on a per-AP basis.
+ * @IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC: indicates that the AP sta should
+ *     be removed only after setting the vif as unassociated, and not the
+ *     opposite. Only relevant for STA vifs.
  */
 enum ieee80211_vif_flags {
        IEEE80211_VIF_BEACON_FILTER             = BIT(0),
        IEEE80211_VIF_GET_NOA_UPDATE            = BIT(3),
        IEEE80211_VIF_EML_ACTIVE                = BIT(4),
        IEEE80211_VIF_IGNORE_OFDMA_WIDER_BW     = BIT(5),
+       IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC  = BIT(6),
 };
 
 
 
 {
        struct ieee80211_if_managed *ifmgd = &sdata->u.mgd;
        struct ieee80211_local *local = sdata->local;
+       struct sta_info *ap_sta = sta_info_get(sdata, sdata->vif.cfg.ap_addr);
        unsigned int link_id;
        u64 changed = 0;
        struct ieee80211_prep_tx_info info = {
 
        lockdep_assert_wiphy(local->hw.wiphy);
 
+       if (WARN_ON(!ap_sta))
+               return;
+
        if (WARN_ON_ONCE(tx && !frame_buf))
                return;
 
 
        sdata->vif.cfg.ssid_len = 0;
 
-       /* remove AP and TDLS peers */
-       sta_info_flush(sdata, -1);
+       /* Remove TDLS peers */
+       __sta_info_flush(sdata, false, -1, ap_sta);
+
+       if (sdata->vif.driver_flags & IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC) {
+               /* Only move the AP state */
+               sta_info_move_state(ap_sta, IEEE80211_STA_NONE);
+       } else {
+               /* Remove AP peer */
+               sta_info_flush(sdata, -1);
+       }
 
        /* finally reset all BSS / config parameters */
        if (!ieee80211_vif_is_mld(&sdata->vif))
                ieee80211_vif_cfg_change_notify(sdata, changed);
        }
 
+       if (sdata->vif.driver_flags & IEEE80211_VIF_REMOVE_AP_AFTER_DISASSOC) {
+               /*
+                * After notifying the driver about the disassoc,
+                * remove the ap sta.
+                */
+               sta_info_flush(sdata, -1);
+       }
+
        /* disassociated - set to defaults now */
        ieee80211_set_wmm_default(&sdata->deflink, false, false);