wifi: iwlwifi: mvm: implement link change ops
Implement the link change ops for links and stations. Note that the stations one is empty for now as we only have support for a single link so far, and then the stations are created with the first link as deflink by mac80211, so right now we don't really need anything. Signed-off-by: Johannes Berg <johannes.berg@intel.com> Signed-off-by: Gregory Greenman <gregory.greenman@intel.com> Link: https://lore.kernel.org/r/20230328104949.6186c5a37e99.Ifd00d3ee93356ddef273aa18f1e081cd8f2c84ae@changeid Signed-off-by: Johannes Berg <johannes.berg@intel.com>
This commit is contained in:
parent
79faae3a40
commit
bf976c814c
@ -3,6 +3,7 @@
|
||||
* Copyright (C) 2022 - 2023 Intel Corporation
|
||||
*/
|
||||
#include "mvm.h"
|
||||
#include "time-event.h"
|
||||
|
||||
static u32 iwl_mvm_get_free_fw_link_id(struct iwl_mvm *mvm,
|
||||
struct iwl_mvm_vif *mvm_vif)
|
||||
@ -108,6 +109,28 @@ int iwl_mvm_link_changed(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
mvmvif->fw_active_links_num >= IWL_MVM_FW_MAX_ACTIVE_LINKS_NUM)
|
||||
return -EINVAL;
|
||||
|
||||
if (changes & LINK_CONTEXT_MODIFY_ACTIVE) {
|
||||
/* When activating a link, phy context should be valid;
|
||||
* when deactivating a link, it also should be valid since
|
||||
* the link was active before. So, do nothing in this case.
|
||||
* Since a link is added first with FW_CTXT_INVALID, then we
|
||||
* can get here in case it's removed before it was activated.
|
||||
*/
|
||||
if (!link_info->phy_ctxt)
|
||||
return 0;
|
||||
|
||||
/* Catch early if driver tries to activate or deactivate a link
|
||||
* twice.
|
||||
*/
|
||||
WARN_ON_ONCE(active == link_info->active);
|
||||
|
||||
/* When deactivating a link session protection should
|
||||
* be stopped
|
||||
*/
|
||||
if (!active && vif->type == NL80211_IFTYPE_STATION)
|
||||
iwl_mvm_stop_session_protection(mvm, vif);
|
||||
}
|
||||
|
||||
cmd.link_id = cpu_to_le32(link_info->fw_link_id);
|
||||
|
||||
/* The phy_id, link address and listen_lmac can be modified only until
|
||||
@ -248,3 +271,23 @@ int iwl_mvm_remove_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* link should be deactivated before removal, so in most cases we need to
|
||||
* perform these two operations together
|
||||
*/
|
||||
int iwl_mvm_disable_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
struct ieee80211_bss_conf *link_conf)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = iwl_mvm_link_changed(mvm, vif, link_conf,
|
||||
LINK_CONTEXT_MODIFY_ACTIVE, false);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = iwl_mvm_remove_link(mvm, vif, link_conf);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
@ -983,6 +983,7 @@ static void iwl_mvm_cleanup_iterator(void *data, u8 *mac,
|
||||
{
|
||||
struct iwl_mvm *mvm = data;
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_probe_resp_data *probe_data;
|
||||
unsigned int link_id;
|
||||
|
||||
mvmvif->uploaded = false;
|
||||
@ -993,13 +994,19 @@ static void iwl_mvm_cleanup_iterator(void *data, u8 *mac,
|
||||
|
||||
memset(&mvmvif->bf_data, 0, sizeof(mvmvif->bf_data));
|
||||
|
||||
mvmvif->fw_active_links_num = 0;
|
||||
for_each_mvm_vif_valid_link(mvmvif, link_id) {
|
||||
mvmvif->link[link_id]->ap_sta_id = IWL_MVM_INVALID_STA;
|
||||
mvmvif->link[link_id]->fw_link_id = IWL_MVM_FW_LINK_ID_INVALID;
|
||||
mvmvif->link[link_id]->phy_ctxt = NULL;
|
||||
memset(&mvmvif->link[link_id]->probe_resp_data, 0,
|
||||
sizeof(mvmvif->link[link_id]->probe_resp_data));
|
||||
mvmvif->link[link_id]->active = 0;
|
||||
}
|
||||
|
||||
probe_data = rcu_dereference_protected(mvmvif->deflink.probe_resp_data,
|
||||
lockdep_is_held(&mvm->mutex));
|
||||
if (probe_data)
|
||||
kfree_rcu(probe_data, rcu_head);
|
||||
RCU_INIT_POINTER(mvmvif->deflink.probe_resp_data, NULL);
|
||||
}
|
||||
|
||||
static void iwl_mvm_restart_cleanup(struct iwl_mvm *mvm)
|
||||
@ -1455,7 +1462,6 @@ static bool iwl_mvm_mac_add_interface_common(struct iwl_mvm *mvm,
|
||||
lockdep_assert_held(&mvm->mutex);
|
||||
|
||||
mvmvif->mvm = mvm;
|
||||
RCU_INIT_POINTER(mvmvif->deflink.probe_resp_data, NULL);
|
||||
|
||||
/* the first link always points to the default one */
|
||||
mvmvif->link[0] = &mvmvif->deflink;
|
||||
|
@ -14,7 +14,6 @@ static int iwl_mvm_mld_mac_add_interface(struct ieee80211_hw *hw,
|
||||
mutex_lock(&mvm->mutex);
|
||||
|
||||
mvmvif->mvm = mvm;
|
||||
RCU_INIT_POINTER(mvmvif->deflink.probe_resp_data, NULL);
|
||||
|
||||
/* Not much to do here. The stack will not allow interface
|
||||
* types or combinations that we didn't advertise, so we
|
||||
@ -35,8 +34,10 @@ static int iwl_mvm_mld_mac_add_interface(struct ieee80211_hw *hw,
|
||||
|
||||
mvmvif->features |= hw->netdev_features;
|
||||
|
||||
/* the first link always points to the default one */
|
||||
/* reset deflink MLO parameters */
|
||||
mvmvif->deflink.fw_link_id = IWL_MVM_FW_LINK_ID_INVALID;
|
||||
mvmvif->deflink.active = 0;
|
||||
/* the first link always points to the default one */
|
||||
mvmvif->link[0] = &mvmvif->deflink;
|
||||
|
||||
ret = iwl_mvm_mld_mac_ctxt_add(mvm, vif);
|
||||
@ -119,10 +120,7 @@ static int iwl_mvm_mld_mac_add_interface(struct ieee80211_hw *hw,
|
||||
goto out_unlock;
|
||||
|
||||
out_remove_link:
|
||||
/* Link needs to be deactivated before removal */
|
||||
iwl_mvm_link_changed(mvm, vif, &vif->bss_conf,
|
||||
LINK_CONTEXT_MODIFY_ACTIVE, false);
|
||||
iwl_mvm_remove_link(mvm, vif, &vif->bss_conf);
|
||||
iwl_mvm_disable_link(mvm, vif, &vif->bss_conf);
|
||||
out_unref_phy:
|
||||
iwl_mvm_phy_ctxt_unref(mvm, mvmvif->deflink.phy_ctxt);
|
||||
out_free_bf:
|
||||
@ -198,14 +196,11 @@ static void iwl_mvm_mld_mac_remove_interface(struct ieee80211_hw *hw,
|
||||
|
||||
/* P2P device uses only one link */
|
||||
iwl_mvm_mld_rm_bcast_sta(mvm, vif, &vif->bss_conf);
|
||||
/* Link needs to be deactivated before removal */
|
||||
iwl_mvm_link_changed(mvm, vif, &vif->bss_conf,
|
||||
LINK_CONTEXT_MODIFY_ACTIVE, false);
|
||||
iwl_mvm_remove_link(mvm, vif, &vif->bss_conf);
|
||||
iwl_mvm_disable_link(mvm, vif, &vif->bss_conf);
|
||||
iwl_mvm_phy_ctxt_unref(mvm, mvmvif->deflink.phy_ctxt);
|
||||
mvmvif->deflink.phy_ctxt = NULL;
|
||||
} else {
|
||||
iwl_mvm_remove_link(mvm, vif, &vif->bss_conf);
|
||||
iwl_mvm_disable_link(mvm, vif, &vif->bss_conf);
|
||||
}
|
||||
|
||||
iwl_mvm_mld_mac_ctxt_remove(mvm, vif);
|
||||
@ -359,7 +354,10 @@ static int iwl_mvm_mld_start_ap_ibss(struct ieee80211_hw *hw,
|
||||
if (ret)
|
||||
goto out_unlock;
|
||||
|
||||
ret = iwl_mvm_link_changed(mvm, vif, link_conf, LINK_CONTEXT_MODIFY_ALL,
|
||||
/* the link should be already activated when assigning chan context */
|
||||
ret = iwl_mvm_link_changed(mvm, vif, link_conf,
|
||||
LINK_CONTEXT_MODIFY_ALL &
|
||||
~LINK_CONTEXT_MODIFY_ACTIVE,
|
||||
true);
|
||||
if (ret)
|
||||
goto out_unlock;
|
||||
@ -839,6 +837,126 @@ static int iwl_mvm_mld_roc(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
|
||||
|
||||
return iwl_mvm_roc_common(hw, vif, channel, duration, type, &ops);
|
||||
}
|
||||
|
||||
static int
|
||||
iwl_mvm_mld_change_vif_links(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
u16 old_links, u16 new_links,
|
||||
struct ieee80211_bss_conf *old[IEEE80211_MLD_MAX_NUM_LINKS])
|
||||
{
|
||||
struct iwl_mvm_vif_link_info *new_link[IEEE80211_MLD_MAX_NUM_LINKS] = {};
|
||||
struct iwl_mvm_vif *mvmvif = iwl_mvm_vif_from_mac80211(vif);
|
||||
struct iwl_mvm *mvm = IWL_MAC80211_GET_MVM(hw);
|
||||
u16 removed = old_links & ~new_links;
|
||||
u16 added = new_links & ~old_links;
|
||||
int err, i;
|
||||
|
||||
if (hweight16(new_links) > 2) {
|
||||
return -EOPNOTSUPP;
|
||||
} else if (hweight16(new_links) > 1) {
|
||||
unsigned int n_active = 0;
|
||||
|
||||
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++) {
|
||||
struct ieee80211_bss_conf *link_conf;
|
||||
|
||||
link_conf = link_conf_dereference_protected(vif, i);
|
||||
if (link_conf &&
|
||||
rcu_access_pointer(link_conf->chanctx_conf))
|
||||
n_active++;
|
||||
}
|
||||
|
||||
if (n_active > 1)
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++) {
|
||||
int r;
|
||||
|
||||
if (test_bit(IWL_MVM_STATUS_IN_HW_RESTART, &mvm->status))
|
||||
break;
|
||||
|
||||
if (!(added & BIT(i)))
|
||||
continue;
|
||||
new_link[i] = kzalloc(sizeof(*new_link[i]), GFP_KERNEL);
|
||||
if (!new_link[i]) {
|
||||
err = -ENOMEM;
|
||||
goto free;
|
||||
}
|
||||
|
||||
new_link[i]->bcast_sta.sta_id = IWL_MVM_INVALID_STA;
|
||||
new_link[i]->mcast_sta.sta_id = IWL_MVM_INVALID_STA;
|
||||
new_link[i]->ap_sta_id = IWL_MVM_INVALID_STA;
|
||||
new_link[i]->fw_link_id = IWL_MVM_FW_LINK_ID_INVALID;
|
||||
|
||||
for (r = 0; r < NUM_IWL_MVM_SMPS_REQ; r++)
|
||||
new_link[i]->smps_requests[r] =
|
||||
IEEE80211_SMPS_AUTOMATIC;
|
||||
}
|
||||
|
||||
mutex_lock(&mvm->mutex);
|
||||
|
||||
if (old_links == 0) {
|
||||
err = iwl_mvm_disable_link(mvm, vif, &vif->bss_conf);
|
||||
if (err)
|
||||
goto out_err;
|
||||
mvmvif->link[0] = NULL;
|
||||
}
|
||||
|
||||
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++) {
|
||||
if (removed & BIT(i)) {
|
||||
struct ieee80211_bss_conf *link_conf = old[i];
|
||||
|
||||
err = iwl_mvm_disable_link(mvm, vif, link_conf);
|
||||
if (err)
|
||||
goto out_err;
|
||||
kfree(mvmvif->link[i]);
|
||||
mvmvif->link[i] = NULL;
|
||||
}
|
||||
|
||||
if (added & BIT(i)) {
|
||||
struct ieee80211_bss_conf *link_conf;
|
||||
|
||||
/* FIXME: allow use of sdata_dereference()? */
|
||||
link_conf = rcu_dereference_protected(vif->link_conf[i],
|
||||
1);
|
||||
if (WARN_ON(!link_conf))
|
||||
continue;
|
||||
|
||||
if (!test_bit(IWL_MVM_STATUS_IN_HW_RESTART,
|
||||
&mvm->status))
|
||||
mvmvif->link[i] = new_link[i];
|
||||
new_link[i] = NULL;
|
||||
err = iwl_mvm_add_link(mvm, vif, link_conf);
|
||||
if (err)
|
||||
goto out_err;
|
||||
}
|
||||
}
|
||||
|
||||
err = 0;
|
||||
if (new_links == 0) {
|
||||
mvmvif->link[0] = &mvmvif->deflink;
|
||||
err = iwl_mvm_add_link(mvm, vif, &vif->bss_conf);
|
||||
}
|
||||
|
||||
out_err:
|
||||
/* we really don't have a good way to roll back here ... */
|
||||
mutex_unlock(&mvm->mutex);
|
||||
|
||||
free:
|
||||
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++)
|
||||
kfree(new_link[i]);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int
|
||||
iwl_mvm_mld_change_sta_links(struct ieee80211_hw *hw,
|
||||
struct ieee80211_vif *vif,
|
||||
struct ieee80211_sta *sta,
|
||||
u16 old_links, u16 new_links)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
const struct ieee80211_ops iwl_mvm_mld_hw_ops = {
|
||||
.tx = iwl_mvm_mac_tx,
|
||||
.wake_tx_queue = iwl_mvm_mac_wake_tx_queue,
|
||||
@ -928,4 +1046,7 @@ const struct ieee80211_ops iwl_mvm_mld_hw_ops = {
|
||||
.sta_add_debugfs = iwl_mvm_sta_add_debugfs,
|
||||
#endif
|
||||
.set_hw_timestamp = iwl_mvm_set_hw_timestamp,
|
||||
|
||||
.change_vif_links = iwl_mvm_mld_change_vif_links,
|
||||
.change_sta_links = iwl_mvm_mld_change_sta_links,
|
||||
};
|
||||
|
@ -1824,6 +1824,8 @@ int iwl_mvm_link_changed(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
u32 changes, bool active);
|
||||
int iwl_mvm_remove_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
struct ieee80211_bss_conf *link_conf);
|
||||
int iwl_mvm_disable_link(struct iwl_mvm *mvm, struct ieee80211_vif *vif,
|
||||
struct ieee80211_bss_conf *link_conf);
|
||||
|
||||
/* AP and IBSS */
|
||||
bool iwl_mvm_start_ap_ibss_common(struct ieee80211_hw *hw,
|
||||
|
Loading…
x
Reference in New Issue
Block a user