2022-09-02 16:12:37 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
/*
|
|
|
|
* MLO link handling
|
|
|
|
*
|
2024-01-29 19:34:49 +01:00
|
|
|
* Copyright (C) 2022-2024 Intel Corporation
|
2022-09-02 16:12:37 +02:00
|
|
|
*/
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <net/mac80211.h>
|
|
|
|
#include "ieee80211_i.h"
|
|
|
|
#include "driver-ops.h"
|
2022-09-02 16:12:56 +02:00
|
|
|
#include "key.h"
|
2023-03-01 12:09:27 +02:00
|
|
|
#include "debugfs_netdev.h"
|
2022-09-02 16:12:37 +02:00
|
|
|
|
|
|
|
void ieee80211_link_setup(struct ieee80211_link_data *link)
|
|
|
|
{
|
|
|
|
if (link->sdata->vif.type == NL80211_IFTYPE_STATION)
|
|
|
|
ieee80211_mgd_setup_link(link);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ieee80211_link_init(struct ieee80211_sub_if_data *sdata,
|
|
|
|
int link_id,
|
|
|
|
struct ieee80211_link_data *link,
|
|
|
|
struct ieee80211_bss_conf *link_conf)
|
|
|
|
{
|
|
|
|
bool deflink = link_id < 0;
|
|
|
|
|
|
|
|
if (link_id < 0)
|
|
|
|
link_id = 0;
|
|
|
|
|
|
|
|
rcu_assign_pointer(sdata->vif.link_conf[link_id], link_conf);
|
|
|
|
rcu_assign_pointer(sdata->link[link_id], link);
|
|
|
|
|
|
|
|
link->sdata = sdata;
|
|
|
|
link->link_id = link_id;
|
|
|
|
link->conf = link_conf;
|
|
|
|
link_conf->link_id = link_id;
|
2023-03-01 12:09:25 +02:00
|
|
|
link_conf->vif = &sdata->vif;
|
2024-10-07 15:00:52 +03:00
|
|
|
link->ap_power_level = IEEE80211_UNSET_POWER_LEVEL;
|
|
|
|
link->user_power_level = sdata->local->user_power_level;
|
|
|
|
link_conf->txpower = INT_MIN;
|
2022-09-02 16:12:37 +02:00
|
|
|
|
2024-05-06 21:54:49 +02:00
|
|
|
wiphy_work_init(&link->csa.finalize_work,
|
2023-08-28 13:59:50 +02:00
|
|
|
ieee80211_csa_finalize_work);
|
2023-08-28 13:59:51 +02:00
|
|
|
wiphy_work_init(&link->color_change_finalize_work,
|
|
|
|
ieee80211_color_change_finalize_work);
|
2024-09-24 21:28:05 +02:00
|
|
|
wiphy_delayed_work_init(&link->color_collision_detect_work,
|
|
|
|
ieee80211_color_collision_detection_work);
|
2022-09-02 16:12:37 +02:00
|
|
|
INIT_LIST_HEAD(&link->assigned_chanctx_list);
|
|
|
|
INIT_LIST_HEAD(&link->reserved_chanctx_list);
|
2024-09-06 12:14:19 +05:30
|
|
|
wiphy_delayed_work_init(&link->dfs_cac_timer_work,
|
|
|
|
ieee80211_dfs_cac_timer_work);
|
2022-09-02 16:12:37 +02:00
|
|
|
|
|
|
|
if (!deflink) {
|
|
|
|
switch (sdata->vif.type) {
|
|
|
|
case NL80211_IFTYPE_AP:
|
|
|
|
ether_addr_copy(link_conf->addr,
|
|
|
|
sdata->wdev.links[link_id].addr);
|
|
|
|
link_conf->bssid = link_conf->addr;
|
|
|
|
WARN_ON(!(sdata->wdev.valid_links & BIT(link_id)));
|
|
|
|
break;
|
|
|
|
case NL80211_IFTYPE_STATION:
|
|
|
|
/* station sets the bssid in ieee80211_mgd_setup_link */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
WARN_ON(1);
|
|
|
|
}
|
2023-03-01 12:09:27 +02:00
|
|
|
|
|
|
|
ieee80211_link_debugfs_add(link);
|
2022-09-02 16:12:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ieee80211_link_stop(struct ieee80211_link_data *link)
|
|
|
|
{
|
|
|
|
if (link->sdata->vif.type == NL80211_IFTYPE_STATION)
|
|
|
|
ieee80211_mgd_stop_link(link);
|
|
|
|
|
2024-09-24 21:28:05 +02:00
|
|
|
wiphy_delayed_work_cancel(link->sdata->local->hw.wiphy,
|
|
|
|
&link->color_collision_detect_work);
|
2024-07-05 15:43:26 +08:00
|
|
|
wiphy_work_cancel(link->sdata->local->hw.wiphy,
|
|
|
|
&link->color_change_finalize_work);
|
2024-01-30 19:39:17 +05:30
|
|
|
wiphy_work_cancel(link->sdata->local->hw.wiphy,
|
2024-05-06 21:54:49 +02:00
|
|
|
&link->csa.finalize_work);
|
2024-09-06 12:14:24 +05:30
|
|
|
|
|
|
|
if (link->sdata->wdev.links[link->link_id].cac_started) {
|
|
|
|
wiphy_delayed_work_cancel(link->sdata->local->hw.wiphy,
|
|
|
|
&link->dfs_cac_timer_work);
|
|
|
|
cfg80211_cac_event(link->sdata->dev,
|
|
|
|
&link->conf->chanreq.oper,
|
|
|
|
NL80211_RADAR_CAC_ABORTED,
|
|
|
|
GFP_KERNEL, link->link_id);
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:12:37 +02:00
|
|
|
ieee80211_link_release_channel(link);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct link_container {
|
|
|
|
struct ieee80211_link_data data;
|
|
|
|
struct ieee80211_bss_conf conf;
|
|
|
|
};
|
|
|
|
|
2022-09-02 16:12:42 +02:00
|
|
|
static void ieee80211_tear_down_links(struct ieee80211_sub_if_data *sdata,
|
|
|
|
struct link_container **links, u16 mask)
|
2022-09-02 16:12:37 +02:00
|
|
|
{
|
2022-09-02 16:12:42 +02:00
|
|
|
struct ieee80211_link_data *link;
|
2022-09-02 16:12:37 +02:00
|
|
|
LIST_HEAD(keys);
|
|
|
|
unsigned int link_id;
|
|
|
|
|
|
|
|
for (link_id = 0; link_id < IEEE80211_MLD_MAX_NUM_LINKS; link_id++) {
|
2022-09-02 16:12:42 +02:00
|
|
|
if (!(mask & BIT(link_id)))
|
|
|
|
continue;
|
|
|
|
link = &links[link_id]->data;
|
|
|
|
if (link_id == 0 && !link)
|
|
|
|
link = &sdata->deflink;
|
|
|
|
if (WARN_ON(!link))
|
2022-09-02 16:12:37 +02:00
|
|
|
continue;
|
2022-09-02 16:12:42 +02:00
|
|
|
ieee80211_remove_link_keys(link, &keys);
|
2023-03-01 12:09:27 +02:00
|
|
|
ieee80211_link_debugfs_remove(link);
|
2022-09-02 16:12:42 +02:00
|
|
|
ieee80211_link_stop(link);
|
2022-09-02 16:12:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
synchronize_rcu();
|
|
|
|
|
|
|
|
ieee80211_free_key_list(sdata->local, &keys);
|
2022-09-02 16:12:42 +02:00
|
|
|
}
|
2022-09-02 16:12:37 +02:00
|
|
|
|
2022-09-02 16:12:42 +02:00
|
|
|
static void ieee80211_free_links(struct ieee80211_sub_if_data *sdata,
|
|
|
|
struct link_container **links)
|
|
|
|
{
|
|
|
|
unsigned int link_id;
|
|
|
|
|
|
|
|
for (link_id = 0; link_id < IEEE80211_MLD_MAX_NUM_LINKS; link_id++)
|
2022-09-02 16:12:37 +02:00
|
|
|
kfree(links[link_id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ieee80211_check_dup_link_addrs(struct ieee80211_sub_if_data *sdata)
|
|
|
|
{
|
|
|
|
unsigned int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++) {
|
|
|
|
struct ieee80211_link_data *link1;
|
|
|
|
|
|
|
|
link1 = sdata_dereference(sdata->link[i], sdata);
|
|
|
|
if (!link1)
|
|
|
|
continue;
|
|
|
|
for (j = i + 1; j < IEEE80211_MLD_MAX_NUM_LINKS; j++) {
|
|
|
|
struct ieee80211_link_data *link2;
|
|
|
|
|
|
|
|
link2 = sdata_dereference(sdata->link[j], sdata);
|
|
|
|
if (!link2)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (ether_addr_equal(link1->conf->addr,
|
|
|
|
link2->conf->addr))
|
|
|
|
return -EALREADY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:12:42 +02:00
|
|
|
static void ieee80211_set_vif_links_bitmaps(struct ieee80211_sub_if_data *sdata,
|
2023-06-08 16:36:11 +03:00
|
|
|
u16 valid_links, u16 dormant_links)
|
2022-09-02 16:12:42 +02:00
|
|
|
{
|
2023-06-08 16:36:11 +03:00
|
|
|
sdata->vif.valid_links = valid_links;
|
|
|
|
sdata->vif.dormant_links = dormant_links;
|
|
|
|
|
|
|
|
if (!valid_links ||
|
|
|
|
WARN((~valid_links & dormant_links) ||
|
|
|
|
!(valid_links & ~dormant_links),
|
|
|
|
"Invalid links: valid=0x%x, dormant=0x%x",
|
|
|
|
valid_links, dormant_links)) {
|
2022-09-02 16:12:42 +02:00
|
|
|
sdata->vif.active_links = 0;
|
2023-06-08 16:36:11 +03:00
|
|
|
sdata->vif.dormant_links = 0;
|
2022-09-02 16:12:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (sdata->vif.type) {
|
|
|
|
case NL80211_IFTYPE_AP:
|
|
|
|
/* in an AP all links are always active */
|
2023-06-08 16:36:11 +03:00
|
|
|
sdata->vif.active_links = valid_links;
|
|
|
|
|
|
|
|
/* AP links are not expected to be disabled */
|
|
|
|
WARN_ON(dormant_links);
|
2022-09-02 16:12:42 +02:00
|
|
|
break;
|
|
|
|
case NL80211_IFTYPE_STATION:
|
|
|
|
if (sdata->vif.active_links)
|
|
|
|
break;
|
2023-06-08 16:36:11 +03:00
|
|
|
sdata->vif.active_links = valid_links & ~dormant_links;
|
|
|
|
WARN_ON(hweight16(sdata->vif.active_links) > 1);
|
2022-09-02 16:12:42 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
WARN_ON(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:12:37 +02:00
|
|
|
static int ieee80211_vif_update_links(struct ieee80211_sub_if_data *sdata,
|
|
|
|
struct link_container **to_free,
|
2023-06-08 16:36:11 +03:00
|
|
|
u16 new_links, u16 dormant_links)
|
2022-09-02 16:12:37 +02:00
|
|
|
{
|
|
|
|
u16 old_links = sdata->vif.valid_links;
|
2022-09-02 16:12:42 +02:00
|
|
|
u16 old_active = sdata->vif.active_links;
|
2022-09-02 16:12:37 +02:00
|
|
|
unsigned long add = new_links & ~old_links;
|
|
|
|
unsigned long rem = old_links & ~new_links;
|
|
|
|
unsigned int link_id;
|
|
|
|
int ret;
|
|
|
|
struct link_container *links[IEEE80211_MLD_MAX_NUM_LINKS] = {}, *link;
|
|
|
|
struct ieee80211_bss_conf *old[IEEE80211_MLD_MAX_NUM_LINKS];
|
|
|
|
struct ieee80211_link_data *old_data[IEEE80211_MLD_MAX_NUM_LINKS];
|
|
|
|
bool use_deflink = old_links == 0; /* set for error case */
|
|
|
|
|
2023-08-29 12:18:56 +02:00
|
|
|
lockdep_assert_wiphy(sdata->local->hw.wiphy);
|
2022-09-02 16:12:37 +02:00
|
|
|
|
|
|
|
memset(to_free, 0, sizeof(links));
|
|
|
|
|
2023-09-28 17:35:31 +03:00
|
|
|
if (old_links == new_links && dormant_links == sdata->vif.dormant_links)
|
2022-09-02 16:12:37 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* if there were no old links, need to clear the pointers to deflink */
|
|
|
|
if (!old_links)
|
|
|
|
rem |= BIT(0);
|
|
|
|
|
|
|
|
/* allocate new link structures first */
|
|
|
|
for_each_set_bit(link_id, &add, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
link = kzalloc(sizeof(*link), GFP_KERNEL);
|
|
|
|
if (!link) {
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto free;
|
|
|
|
}
|
|
|
|
links[link_id] = link;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* keep track of the old pointers for the driver */
|
|
|
|
BUILD_BUG_ON(sizeof(old) != sizeof(sdata->vif.link_conf));
|
|
|
|
memcpy(old, sdata->vif.link_conf, sizeof(old));
|
|
|
|
/* and for us in error cases */
|
|
|
|
BUILD_BUG_ON(sizeof(old_data) != sizeof(sdata->link));
|
|
|
|
memcpy(old_data, sdata->link, sizeof(old_data));
|
|
|
|
|
|
|
|
/* grab old links to free later */
|
|
|
|
for_each_set_bit(link_id, &rem, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
if (rcu_access_pointer(sdata->link[link_id]) != &sdata->deflink) {
|
|
|
|
/*
|
|
|
|
* we must have allocated the data through this path so
|
|
|
|
* we know we can free both at the same time
|
|
|
|
*/
|
|
|
|
to_free[link_id] = container_of(rcu_access_pointer(sdata->link[link_id]),
|
|
|
|
typeof(*links[link_id]),
|
|
|
|
data);
|
|
|
|
}
|
|
|
|
|
|
|
|
RCU_INIT_POINTER(sdata->link[link_id], NULL);
|
|
|
|
RCU_INIT_POINTER(sdata->vif.link_conf[link_id], NULL);
|
|
|
|
}
|
|
|
|
|
2023-09-28 17:35:26 +03:00
|
|
|
if (!old_links)
|
|
|
|
ieee80211_debugfs_recreate_netdev(sdata, true);
|
|
|
|
|
2022-09-02 16:12:37 +02:00
|
|
|
/* link them into data structures */
|
|
|
|
for_each_set_bit(link_id, &add, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
WARN_ON(!use_deflink &&
|
|
|
|
rcu_access_pointer(sdata->link[link_id]) == &sdata->deflink);
|
|
|
|
|
|
|
|
link = links[link_id];
|
|
|
|
ieee80211_link_init(sdata, link_id, &link->data, &link->conf);
|
|
|
|
ieee80211_link_setup(&link->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_links == 0)
|
|
|
|
ieee80211_link_init(sdata, -1, &sdata->deflink,
|
|
|
|
&sdata->vif.bss_conf);
|
|
|
|
|
|
|
|
ret = ieee80211_check_dup_link_addrs(sdata);
|
|
|
|
if (!ret) {
|
2022-09-02 16:12:42 +02:00
|
|
|
/* for keys we will not be able to undo this */
|
|
|
|
ieee80211_tear_down_links(sdata, to_free, rem);
|
|
|
|
|
2023-06-08 16:36:11 +03:00
|
|
|
ieee80211_set_vif_links_bitmaps(sdata, new_links, dormant_links);
|
2022-09-02 16:12:42 +02:00
|
|
|
|
2022-09-02 16:12:37 +02:00
|
|
|
/* tell the driver */
|
|
|
|
ret = drv_change_vif_links(sdata->local, sdata,
|
2022-09-02 16:12:42 +02:00
|
|
|
old_links & old_active,
|
|
|
|
new_links & sdata->vif.active_links,
|
2022-09-02 16:12:37 +02:00
|
|
|
old);
|
2023-09-28 17:35:26 +03:00
|
|
|
if (!new_links)
|
|
|
|
ieee80211_debugfs_recreate_netdev(sdata, false);
|
2022-09-02 16:12:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
/* restore config */
|
|
|
|
memcpy(sdata->link, old_data, sizeof(old_data));
|
|
|
|
memcpy(sdata->vif.link_conf, old, sizeof(old));
|
2023-06-08 16:36:11 +03:00
|
|
|
ieee80211_set_vif_links_bitmaps(sdata, old_links, dormant_links);
|
2022-09-02 16:12:37 +02:00
|
|
|
/* and free (only) the newly allocated links */
|
|
|
|
memset(to_free, 0, sizeof(links));
|
|
|
|
goto free;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* use deflink/bss_conf again if and only if there are no more links */
|
|
|
|
use_deflink = new_links == 0;
|
|
|
|
|
|
|
|
goto deinit;
|
|
|
|
free:
|
|
|
|
/* if we failed during allocation, only free all */
|
|
|
|
for (link_id = 0; link_id < IEEE80211_MLD_MAX_NUM_LINKS; link_id++) {
|
|
|
|
kfree(links[link_id]);
|
|
|
|
links[link_id] = NULL;
|
|
|
|
}
|
|
|
|
deinit:
|
|
|
|
if (use_deflink)
|
|
|
|
ieee80211_link_init(sdata, -1, &sdata->deflink,
|
|
|
|
&sdata->vif.bss_conf);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ieee80211_vif_set_links(struct ieee80211_sub_if_data *sdata,
|
2023-06-08 16:36:11 +03:00
|
|
|
u16 new_links, u16 dormant_links)
|
2022-09-02 16:12:37 +02:00
|
|
|
{
|
|
|
|
struct link_container *links[IEEE80211_MLD_MAX_NUM_LINKS];
|
|
|
|
int ret;
|
|
|
|
|
2023-06-08 16:36:11 +03:00
|
|
|
ret = ieee80211_vif_update_links(sdata, links, new_links,
|
|
|
|
dormant_links);
|
2022-09-02 16:12:37 +02:00
|
|
|
ieee80211_free_links(sdata, links);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
static int _ieee80211_set_active_links(struct ieee80211_sub_if_data *sdata,
|
|
|
|
u16 active_links)
|
|
|
|
{
|
|
|
|
struct ieee80211_bss_conf *link_confs[IEEE80211_MLD_MAX_NUM_LINKS];
|
|
|
|
struct ieee80211_local *local = sdata->local;
|
|
|
|
u16 old_active = sdata->vif.active_links;
|
|
|
|
unsigned long rem = old_active & ~active_links;
|
|
|
|
unsigned long add = active_links & ~old_active;
|
|
|
|
struct sta_info *sta;
|
|
|
|
unsigned int link_id;
|
|
|
|
int ret, i;
|
|
|
|
|
|
|
|
if (!ieee80211_sdata_running(sdata))
|
|
|
|
return -ENETDOWN;
|
|
|
|
|
|
|
|
if (sdata->vif.type != NL80211_IFTYPE_STATION)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2023-06-08 16:36:08 +03:00
|
|
|
if (active_links & ~ieee80211_vif_usable_links(&sdata->vif))
|
2022-09-02 16:12:56 +02:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
/* nothing to do */
|
|
|
|
if (old_active == active_links)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (i = 0; i < IEEE80211_MLD_MAX_NUM_LINKS; i++)
|
|
|
|
link_confs[i] = sdata_dereference(sdata->vif.link_conf[i],
|
|
|
|
sdata);
|
|
|
|
|
|
|
|
if (add) {
|
|
|
|
sdata->vif.active_links |= active_links;
|
|
|
|
ret = drv_change_vif_links(local, sdata,
|
|
|
|
old_active,
|
|
|
|
sdata->vif.active_links,
|
|
|
|
link_confs);
|
|
|
|
if (ret) {
|
|
|
|
sdata->vif.active_links = old_active;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for_each_set_bit(link_id, &rem, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
struct ieee80211_link_data *link;
|
|
|
|
|
|
|
|
link = sdata_dereference(sdata->link[link_id], sdata);
|
|
|
|
|
2024-02-28 09:55:48 +01:00
|
|
|
ieee80211_teardown_tdls_peers(link);
|
2022-09-02 16:12:56 +02:00
|
|
|
|
2024-03-20 09:13:59 +02:00
|
|
|
__ieee80211_link_release_channel(link, true);
|
2024-04-15 11:27:11 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If CSA is (still) active while the link is deactivated,
|
|
|
|
* just schedule the channel switch work for the time we
|
|
|
|
* had previously calculated, and we'll take the process
|
|
|
|
* from there.
|
|
|
|
*/
|
|
|
|
if (link->conf->csa_active)
|
|
|
|
wiphy_delayed_work_queue(local->hw.wiphy,
|
2024-05-06 21:54:49 +02:00
|
|
|
&link->u.mgd.csa.switch_work,
|
|
|
|
link->u.mgd.csa.time -
|
2024-04-15 11:27:11 +03:00
|
|
|
jiffies);
|
2022-09-02 16:12:56 +02:00
|
|
|
}
|
|
|
|
|
wifi: mac80211: re-order assigning channel in activate links
The current flow in _ieee80211_set_active_links() does not align with the
operational requirements of drivers that groups multiple hardware
under a single wiphy. These drivers (e.g ath12k) rely on channel
assignment to determine the appropriate hardware for each link. Without
this, the drivers cannot correctly establish the link interface.
Currently in _ieee80211_set_active_links(), after calling
drv_change_vif_links() on the driver, the state of all connected stations
is updated via drv_change_sta_links(). This is followed by handling keys
in the links, and finally, assigning the channel to the links.
Consequently, drv_change_sta_links() prompts drivers to create the station
entry at their level and within their firmware. However, since channels
have not yet been assigned to links at this stage, drivers have not
created the necessary link interface for establishing link stations,
leading to failures in activating the links.
Therefore, re-order the logic so that after drv_change_vif_links() and
removing the old links, channels are assigned to newly added links.
Following this, the flow proceeds to station handling.
Signed-off-by: Aditya Kumar Singh <quic_adisi@quicinc.com>
Link: https://patch.msgid.link/20241001085034.2745669-1-quic_adisi@quicinc.com
[Johannes: fix iwlwifi to deal with the changes]
Reviewed-by: Miriam Rachel Korenblit <miriam.rachel.korenblit@intel.com>
Signed-off-by: Johannes Berg <johannes.berg@intel.com>
2024-10-01 14:20:34 +05:30
|
|
|
for_each_set_bit(link_id, &add, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
struct ieee80211_link_data *link;
|
|
|
|
|
|
|
|
link = sdata_dereference(sdata->link[link_id], sdata);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This call really should not fail. Unfortunately, it appears
|
|
|
|
* that this may happen occasionally with some drivers. Should
|
|
|
|
* it happen, we are stuck in a bad place as going backwards is
|
|
|
|
* not really feasible.
|
|
|
|
*
|
|
|
|
* So lets just tell link_use_channel that it must not fail to
|
|
|
|
* assign the channel context (from mac80211's perspective) and
|
|
|
|
* assume the driver is going to trigger a recovery flow if it
|
|
|
|
* had a failure.
|
|
|
|
* That really is not great nor guaranteed to work. But at least
|
|
|
|
* the internal mac80211 state remains consistent and there is
|
|
|
|
* a chance that we can recover.
|
|
|
|
*/
|
|
|
|
ret = _ieee80211_link_use_channel(link,
|
|
|
|
&link->conf->chanreq,
|
|
|
|
IEEE80211_CHANCTX_SHARED,
|
|
|
|
true);
|
|
|
|
WARN_ON_ONCE(ret);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* inform about the link info changed parameters after all
|
|
|
|
* stations are also added
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
list_for_each_entry(sta, &local->sta_list, list) {
|
|
|
|
if (sdata != sta->sdata)
|
|
|
|
continue;
|
2022-10-05 13:52:26 +02:00
|
|
|
|
|
|
|
/* this is very temporary, but do it anyway */
|
|
|
|
__ieee80211_sta_recalc_aggregates(sta,
|
|
|
|
old_active | active_links);
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
ret = drv_change_sta_links(local, sdata, &sta->sta,
|
|
|
|
old_active,
|
|
|
|
old_active | active_links);
|
|
|
|
WARN_ON_ONCE(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = ieee80211_key_switch_links(sdata, rem, add);
|
|
|
|
WARN_ON_ONCE(ret);
|
|
|
|
|
|
|
|
list_for_each_entry(sta, &local->sta_list, list) {
|
|
|
|
if (sdata != sta->sdata)
|
|
|
|
continue;
|
2022-10-05 13:52:26 +02:00
|
|
|
|
|
|
|
__ieee80211_sta_recalc_aggregates(sta, active_links);
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
ret = drv_change_sta_links(local, sdata, &sta->sta,
|
|
|
|
old_active | active_links,
|
|
|
|
active_links);
|
|
|
|
WARN_ON_ONCE(ret);
|
2022-10-05 13:52:26 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Do it again, just in case - the driver might very
|
|
|
|
* well have called ieee80211_sta_recalc_aggregates()
|
|
|
|
* from there when filling in the new links, which
|
|
|
|
* would set it wrong since the vif's active links are
|
|
|
|
* not switched yet...
|
|
|
|
*/
|
|
|
|
__ieee80211_sta_recalc_aggregates(sta, active_links);
|
2022-09-02 16:12:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for_each_set_bit(link_id, &add, IEEE80211_MLD_MAX_NUM_LINKS) {
|
|
|
|
struct ieee80211_link_data *link;
|
|
|
|
|
|
|
|
link = sdata_dereference(sdata->link[link_id], sdata);
|
|
|
|
|
2023-06-08 16:35:59 +03:00
|
|
|
ieee80211_mgd_set_link_qos_params(link);
|
2022-09-02 16:12:56 +02:00
|
|
|
ieee80211_link_info_change_notify(sdata, link,
|
|
|
|
BSS_CHANGED_ERP_CTS_PROT |
|
|
|
|
BSS_CHANGED_ERP_PREAMBLE |
|
|
|
|
BSS_CHANGED_ERP_SLOT |
|
|
|
|
BSS_CHANGED_HT |
|
|
|
|
BSS_CHANGED_BASIC_RATES |
|
|
|
|
BSS_CHANGED_BSSID |
|
|
|
|
BSS_CHANGED_CQM |
|
|
|
|
BSS_CHANGED_QOS |
|
|
|
|
BSS_CHANGED_TXPOWER |
|
|
|
|
BSS_CHANGED_BANDWIDTH |
|
|
|
|
BSS_CHANGED_TWT |
|
|
|
|
BSS_CHANGED_HE_OBSS_PD |
|
|
|
|
BSS_CHANGED_HE_BSS_COLOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
old_active = sdata->vif.active_links;
|
|
|
|
sdata->vif.active_links = active_links;
|
|
|
|
|
|
|
|
if (rem) {
|
|
|
|
ret = drv_change_vif_links(local, sdata, old_active,
|
|
|
|
active_links, link_confs);
|
|
|
|
WARN_ON_ONCE(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-08-29 12:18:56 +02:00
|
|
|
int ieee80211_set_active_links(struct ieee80211_vif *vif, u16 active_links)
|
2022-09-02 16:12:56 +02:00
|
|
|
{
|
|
|
|
struct ieee80211_sub_if_data *sdata = vif_to_sdata(vif);
|
|
|
|
struct ieee80211_local *local = sdata->local;
|
|
|
|
u16 old_active;
|
|
|
|
int ret;
|
|
|
|
|
2023-08-28 14:00:01 +02:00
|
|
|
lockdep_assert_wiphy(local->hw.wiphy);
|
2023-08-28 14:00:02 +02:00
|
|
|
|
2024-01-29 20:00:53 +01:00
|
|
|
if (WARN_ON(!active_links))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2024-03-20 09:14:03 +02:00
|
|
|
old_active = sdata->vif.active_links;
|
|
|
|
if (old_active == active_links)
|
|
|
|
return 0;
|
|
|
|
|
2023-12-20 13:41:45 +02:00
|
|
|
if (!drv_can_activate_links(local, sdata, active_links))
|
|
|
|
return -EINVAL;
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
if (old_active & active_links) {
|
|
|
|
/*
|
|
|
|
* if there's at least one link that stays active across
|
|
|
|
* the change then switch to it (to those) first, and
|
|
|
|
* then enable the additional links
|
|
|
|
*/
|
|
|
|
ret = _ieee80211_set_active_links(sdata,
|
|
|
|
old_active & active_links);
|
|
|
|
if (!ret)
|
|
|
|
ret = _ieee80211_set_active_links(sdata, active_links);
|
|
|
|
} else {
|
|
|
|
/* otherwise switch directly */
|
|
|
|
ret = _ieee80211_set_active_links(sdata, active_links);
|
|
|
|
}
|
2023-05-11 13:13:21 +02:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2022-09-02 16:12:56 +02:00
|
|
|
EXPORT_SYMBOL_GPL(ieee80211_set_active_links);
|
|
|
|
|
|
|
|
void ieee80211_set_active_links_async(struct ieee80211_vif *vif,
|
|
|
|
u16 active_links)
|
|
|
|
{
|
|
|
|
struct ieee80211_sub_if_data *sdata = vif_to_sdata(vif);
|
|
|
|
|
2024-01-29 20:00:53 +01:00
|
|
|
if (WARN_ON(!active_links))
|
|
|
|
return;
|
|
|
|
|
2022-09-02 16:12:56 +02:00
|
|
|
if (!ieee80211_sdata_running(sdata))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (sdata->vif.type != NL80211_IFTYPE_STATION)
|
|
|
|
return;
|
|
|
|
|
2023-06-08 16:36:08 +03:00
|
|
|
if (active_links & ~ieee80211_vif_usable_links(&sdata->vif))
|
2022-09-02 16:12:56 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* nothing to do */
|
|
|
|
if (sdata->vif.active_links == active_links)
|
|
|
|
return;
|
|
|
|
|
|
|
|
sdata->desired_active_links = active_links;
|
2023-08-28 13:59:43 +02:00
|
|
|
wiphy_work_queue(sdata->local->hw.wiphy, &sdata->activate_links_work);
|
2022-09-02 16:12:56 +02:00
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(ieee80211_set_active_links_async);
|