root/drivers/net/wireless/mediatek/mt76/channel.c
// SPDX-License-Identifier: BSD-3-Clause-Clear
/*
 * Copyright (C) 2024 Felix Fietkau <nbd@nbd.name>
 */
#include "mt76.h"

static struct mt76_vif_link *
mt76_alloc_mlink(struct mt76_dev *dev, struct mt76_vif_data *mvif)
{
        struct mt76_vif_link *mlink;

        mlink = kzalloc(dev->drv->link_data_size, GFP_KERNEL);
        if (!mlink)
                return NULL;

        mlink->mvif = mvif;

        return mlink;
}

static int
mt76_phy_update_channel(struct mt76_phy *phy,
                        struct ieee80211_chanctx_conf *conf)
{
        phy->radar_enabled = conf->radar_enabled;
        phy->main_chandef = conf->def;
        phy->chanctx = (struct mt76_chanctx *)conf->drv_priv;

        return __mt76_set_channel(phy, &phy->main_chandef, false);
}

int mt76_add_chanctx(struct ieee80211_hw *hw,
                     struct ieee80211_chanctx_conf *conf)
{
        struct mt76_chanctx *ctx = (struct mt76_chanctx *)conf->drv_priv;
        struct mt76_phy *phy = hw->priv;
        struct mt76_dev *dev = phy->dev;
        int ret = -EINVAL;

        phy = ctx->phy = dev->band_phys[conf->def.chan->band];
        if (WARN_ON_ONCE(!phy))
                return ret;

        if (dev->scan.phy == phy)
                mt76_abort_scan(dev);

        mutex_lock(&dev->mutex);
        if (!phy->chanctx)
                ret = mt76_phy_update_channel(phy, conf);
        else
                ret = 0;
        mutex_unlock(&dev->mutex);

        return ret;
}
EXPORT_SYMBOL_GPL(mt76_add_chanctx);

void mt76_remove_chanctx(struct ieee80211_hw *hw,
                         struct ieee80211_chanctx_conf *conf)
{
        struct mt76_chanctx *ctx = (struct mt76_chanctx *)conf->drv_priv;
        struct mt76_phy *phy = hw->priv;
        struct mt76_dev *dev = phy->dev;

        phy = ctx->phy;
        if (WARN_ON_ONCE(!phy))
                return;

        if (dev->scan.phy == phy)
                mt76_abort_scan(dev);

        mutex_lock(&dev->mutex);
        if (phy->chanctx == ctx)
                phy->chanctx = NULL;
        mutex_unlock(&dev->mutex);
}
EXPORT_SYMBOL_GPL(mt76_remove_chanctx);

void mt76_change_chanctx(struct ieee80211_hw *hw,
                         struct ieee80211_chanctx_conf *conf,
                         u32 changed)
{
        struct mt76_chanctx *ctx = (struct mt76_chanctx *)conf->drv_priv;
        struct mt76_phy *phy = ctx->phy;
        struct mt76_dev *dev = phy->dev;

        if (!(changed & (IEEE80211_CHANCTX_CHANGE_WIDTH |
                         IEEE80211_CHANCTX_CHANGE_RADAR)))
                return;

        cancel_delayed_work_sync(&phy->mac_work);

        mutex_lock(&dev->mutex);
        mt76_phy_update_channel(phy, conf);
        mutex_unlock(&dev->mutex);
}
EXPORT_SYMBOL_GPL(mt76_change_chanctx);


int mt76_assign_vif_chanctx(struct ieee80211_hw *hw,
                            struct ieee80211_vif *vif,
                            struct ieee80211_bss_conf *link_conf,
                            struct ieee80211_chanctx_conf *conf)
{
        struct mt76_chanctx *ctx = (struct mt76_chanctx *)conf->drv_priv;
        struct mt76_vif_link *mlink = (struct mt76_vif_link *)vif->drv_priv;
        struct mt76_vif_data *mvif = mlink->mvif;
        int link_id = link_conf->link_id;
        struct mt76_phy *phy = ctx->phy;
        struct mt76_dev *dev = phy->dev;
        bool mlink_alloc = false;
        int ret = 0;

        if (dev->scan.vif == vif)
                mt76_abort_scan(dev);

        mutex_lock(&dev->mutex);

        if (vif->type == NL80211_IFTYPE_MONITOR &&
            is_zero_ether_addr(vif->addr))
                goto out;

        mlink = mt76_vif_conf_link(dev, vif, link_conf);
        if (!mlink) {
                mlink = mt76_alloc_mlink(dev, mvif);
                if (!mlink) {
                        ret = -ENOMEM;
                        goto out;
                }
                mlink_alloc = true;
        }

        mlink->ctx = conf;
        ret = dev->drv->vif_link_add(phy, vif, link_conf, mlink);
        if (ret) {
                if (mlink_alloc)
                        kfree(mlink);
                goto out;
        }

        if (link_conf != &vif->bss_conf)
                rcu_assign_pointer(mvif->link[link_id], mlink);

out:
        mutex_unlock(&dev->mutex);

        return ret;
}
EXPORT_SYMBOL_GPL(mt76_assign_vif_chanctx);

void mt76_unassign_vif_chanctx(struct ieee80211_hw *hw,
                               struct ieee80211_vif *vif,
                               struct ieee80211_bss_conf *link_conf,
                               struct ieee80211_chanctx_conf *conf)
{
        struct mt76_chanctx *ctx = (struct mt76_chanctx *)conf->drv_priv;
        struct mt76_vif_link *mlink = (struct mt76_vif_link *)vif->drv_priv;
        struct mt76_vif_data *mvif = mlink->mvif;
        int link_id = link_conf->link_id;
        struct mt76_phy *phy = ctx->phy;
        struct mt76_dev *dev = phy->dev;

        if (dev->scan.vif == vif)
                mt76_abort_scan(dev);

        mutex_lock(&dev->mutex);

        if (vif->type == NL80211_IFTYPE_MONITOR &&
            is_zero_ether_addr(vif->addr))
                goto out;

        mlink = mt76_vif_conf_link(dev, vif, link_conf);
        if (!mlink)
                goto out;

        if (mlink != (struct mt76_vif_link *)vif->drv_priv)
                rcu_assign_pointer(mvif->link[link_id], NULL);

        dev->drv->vif_link_remove(phy, vif, link_conf, mlink);
        mlink->ctx = NULL;

        if (mlink != (struct mt76_vif_link *)vif->drv_priv)
                kfree_rcu(mlink, rcu_head);

out:
        mutex_unlock(&dev->mutex);
}
EXPORT_SYMBOL_GPL(mt76_unassign_vif_chanctx);

int mt76_switch_vif_chanctx(struct ieee80211_hw *hw,
                            struct ieee80211_vif_chanctx_switch *vifs,
                            int n_vifs,
                            enum ieee80211_chanctx_switch_mode mode)
{
        struct mt76_chanctx *old_ctx = (struct mt76_chanctx *)vifs->old_ctx->drv_priv;
        struct mt76_chanctx *new_ctx = (struct mt76_chanctx *)vifs->new_ctx->drv_priv;
        struct ieee80211_chanctx_conf *conf = vifs->new_ctx;
        struct mt76_phy *old_phy = old_ctx->phy;
        struct mt76_phy *phy = hw->priv;
        struct mt76_dev *dev = phy->dev;
        struct mt76_vif_link *mlink;
        bool update_chan;
        int i, ret = 0;

        if (mode == CHANCTX_SWMODE_SWAP_CONTEXTS)
                phy = new_ctx->phy = dev->band_phys[conf->def.chan->band];
        else
                phy = new_ctx->phy;
        if (!phy)
                return -EINVAL;

        update_chan = phy->chanctx != new_ctx;
        if (update_chan) {
                if (dev->scan.phy == phy)
                        mt76_abort_scan(dev);

                cancel_delayed_work_sync(&phy->mac_work);
        }

        mutex_lock(&dev->mutex);

        if (mode == CHANCTX_SWMODE_SWAP_CONTEXTS &&
            phy != old_phy && old_phy->chanctx == old_ctx)
                old_phy->chanctx = NULL;

        if (update_chan)
                ret = mt76_phy_update_channel(phy, vifs->new_ctx);

        if (ret)
                goto out;

        if (old_phy == phy)
                goto skip_link_replace;

        for (i = 0; i < n_vifs; i++) {
                mlink = mt76_vif_conf_link(dev, vifs[i].vif, vifs[i].link_conf);
                if (!mlink)
                        continue;

                dev->drv->vif_link_remove(old_phy, vifs[i].vif,
                                          vifs[i].link_conf, mlink);

                ret = dev->drv->vif_link_add(phy, vifs[i].vif,
                                             vifs[i].link_conf, mlink);
                if (ret)
                        goto out;

        }

skip_link_replace:
        for (i = 0; i < n_vifs; i++) {
                mlink = mt76_vif_conf_link(dev, vifs[i].vif, vifs[i].link_conf);
                if (!mlink)
                        continue;

                mlink->ctx = vifs->new_ctx;
        }

out:
        mutex_unlock(&dev->mutex);

        return ret;
}
EXPORT_SYMBOL_GPL(mt76_switch_vif_chanctx);

struct mt76_vif_link *mt76_get_vif_phy_link(struct mt76_phy *phy,
                                            struct ieee80211_vif *vif)
{
        struct mt76_vif_link *mlink = (struct mt76_vif_link *)vif->drv_priv;
        struct mt76_vif_data *mvif = mlink->mvif;
        struct mt76_dev *dev = phy->dev;
        int i, ret;

        for (i = 0; i < ARRAY_SIZE(mvif->link); i++) {
                mlink = mt76_dereference(mvif->link[i], dev);
                if (!mlink)
                        continue;

                if (mt76_vif_link_phy(mlink) == phy)
                        return mlink;
        }

        if (!dev->drv->vif_link_add)
                return ERR_PTR(-EINVAL);

        mlink = mt76_alloc_mlink(dev, mvif);
        if (!mlink)
                return ERR_PTR(-ENOMEM);

        mlink->offchannel = true;
        ret = dev->drv->vif_link_add(phy, vif, &vif->bss_conf, mlink);
        if (ret) {
                kfree(mlink);
                return ERR_PTR(ret);
        }
        rcu_assign_pointer(mvif->offchannel_link, mlink);

        return mlink;
}

void mt76_put_vif_phy_link(struct mt76_phy *phy, struct ieee80211_vif *vif,
                           struct mt76_vif_link *mlink)
{
        struct mt76_dev *dev = phy->dev;
        struct mt76_vif_data *mvif;

        if (IS_ERR_OR_NULL(mlink) || !mlink->offchannel)
                return;

        mvif = mlink->mvif;

        rcu_assign_pointer(mvif->offchannel_link, NULL);
        dev->drv->vif_link_remove(phy, vif, &vif->bss_conf, mlink);
        kfree(mlink);
}

void mt76_roc_complete(struct mt76_phy *phy)
{
        struct mt76_vif_link *mlink = phy->roc_link;
        struct mt76_dev *dev = phy->dev;

        if (!phy->roc_vif)
                return;

        if (mlink)
                mlink->mvif->roc_phy = NULL;
        if (phy->main_chandef.chan &&
            !test_bit(MT76_MCU_RESET, &dev->phy.state))
                mt76_set_channel(phy, &phy->main_chandef, false);
        mt76_put_vif_phy_link(phy, phy->roc_vif, phy->roc_link);
        phy->roc_vif = NULL;
        phy->roc_link = NULL;
        if (!test_bit(MT76_MCU_RESET, &dev->phy.state))
                ieee80211_remain_on_channel_expired(phy->hw);
}

void mt76_roc_complete_work(struct work_struct *work)
{
        struct mt76_phy *phy = container_of(work, struct mt76_phy, roc_work.work);
        struct mt76_dev *dev = phy->dev;

        mutex_lock(&dev->mutex);
        mt76_roc_complete(phy);
        mutex_unlock(&dev->mutex);
}

void mt76_abort_roc(struct mt76_phy *phy)
{
        struct mt76_dev *dev = phy->dev;

        cancel_delayed_work_sync(&phy->roc_work);

        mutex_lock(&dev->mutex);
        mt76_roc_complete(phy);
        mutex_unlock(&dev->mutex);
}
EXPORT_SYMBOL_GPL(mt76_abort_roc);

int mt76_remain_on_channel(struct ieee80211_hw *hw, struct ieee80211_vif *vif,
                           struct ieee80211_channel *chan, int duration,
                           enum ieee80211_roc_type type)
{
        struct cfg80211_chan_def chandef = {};
        struct mt76_phy *phy = hw->priv;
        struct mt76_dev *dev = phy->dev;
        struct mt76_vif_link *mlink;
        int ret = 0;

        phy = dev->band_phys[chan->band];
        if (!phy)
                return -EINVAL;

        mutex_lock(&dev->mutex);

        if (phy->roc_vif || dev->scan.phy == phy ||
            test_bit(MT76_MCU_RESET, &dev->phy.state)) {
                ret = -EBUSY;
                goto out;
        }

        mlink = mt76_get_vif_phy_link(phy, vif);
        if (IS_ERR(mlink)) {
                ret = PTR_ERR(mlink);
                goto out;
        }

        mlink->mvif->roc_phy = phy;
        phy->roc_vif = vif;
        phy->roc_link = mlink;
        cfg80211_chandef_create(&chandef, chan, NL80211_CHAN_HT20);
        mt76_set_channel(phy, &chandef, true);
        ieee80211_ready_on_channel(hw);
        ieee80211_queue_delayed_work(phy->hw, &phy->roc_work,
                                     msecs_to_jiffies(duration));

out:
        mutex_unlock(&dev->mutex);
        return ret;
}
EXPORT_SYMBOL_GPL(mt76_remain_on_channel);

int mt76_cancel_remain_on_channel(struct ieee80211_hw *hw,
                                  struct ieee80211_vif *vif)
{
        struct mt76_vif_link *mlink = (struct mt76_vif_link *)vif->drv_priv;
        struct mt76_vif_data *mvif = mlink->mvif;
        struct mt76_phy *phy = mvif->roc_phy;

        if (!phy)
                return 0;

        mt76_abort_roc(phy);

        return 0;
}
EXPORT_SYMBOL_GPL(mt76_cancel_remain_on_channel);