root/drivers/net/wireless/mediatek/mt76/mt7925/regd.c
// SPDX-License-Identifier: BSD-3-Clause-Clear
/* Copyright (C) 2025 MediaTek Inc. */

#include "mt7925.h"
#include "regd.h"
#include "mcu.h"

static bool mt7925_disable_clc;
module_param_named(disable_clc, mt7925_disable_clc, bool, 0644);
MODULE_PARM_DESC(disable_clc, "disable CLC support");

bool mt7925_regd_clc_supported(struct mt792x_dev *dev)
{
        if (mt7925_disable_clc ||
            mt76_is_usb(&dev->mt76))
                return false;

        return true;
}

void mt7925_regd_be_ctrl(struct mt792x_dev *dev, u8 *alpha2)
{
        struct mt792x_phy *phy = &dev->phy;
        struct mt7925_clc_rule_v2 *rule;
        struct mt7925_clc *clc;
        bool old = dev->has_eht, new = true;
        u32 mtcl_conf = mt792x_acpi_get_mtcl_conf(&dev->phy, alpha2);
        u8 *pos;

        if (mtcl_conf != MT792X_ACPI_MTCL_INVALID &&
            (((mtcl_conf >> 4) & 0x3) == 0)) {
                new = false;
                goto out;
        }

        if (!phy->clc[MT792x_CLC_BE_CTRL])
                goto out;

        clc = (struct mt7925_clc *)phy->clc[MT792x_CLC_BE_CTRL];
        pos = clc->data;

        while (1) {
                rule = (struct mt7925_clc_rule_v2 *)pos;

                if (rule->alpha2[0] == alpha2[0] &&
                    rule->alpha2[1] == alpha2[1]) {
                        new = false;
                        break;
                }

                /* Check the last one */
                if (rule->flag & BIT(0))
                        break;

                pos += sizeof(*rule);
        }

out:
        if (old == new)
                return;

        dev->has_eht = new;
        mt7925_set_stream_he_eht_caps(phy);
}

static void
mt7925_regd_channel_update(struct wiphy *wiphy, struct mt792x_dev *dev)
{
#define IS_UNII_INVALID(idx, sfreq, efreq, cfreq) \
        (!(dev->phy.clc_chan_conf & BIT(idx)) && (cfreq) >= (sfreq) && (cfreq) <= (efreq))
#define MT7925_UNII_59G_IS_VALID        0x1
#define MT7925_UNII_6G_IS_VALID 0x1e
        struct ieee80211_supported_band *sband;
        struct mt76_dev *mdev = &dev->mt76;
        struct ieee80211_channel *ch;
        u32 mtcl_conf = mt792x_acpi_get_mtcl_conf(&dev->phy, mdev->alpha2);
        int i;

        if (mtcl_conf != MT792X_ACPI_MTCL_INVALID) {
                if ((mtcl_conf & 0x3) == 0)
                        dev->phy.clc_chan_conf &= ~MT7925_UNII_59G_IS_VALID;
                if (((mtcl_conf >> 2) & 0x3) == 0)
                        dev->phy.clc_chan_conf &= ~MT7925_UNII_6G_IS_VALID;
        }

        sband = wiphy->bands[NL80211_BAND_2GHZ];
        if (!sband)
                return;

        for (i = 0; i < sband->n_channels; i++) {
                ch = &sband->channels[i];

                if (!dev->has_eht)
                        ch->flags |= IEEE80211_CHAN_NO_EHT;
        }

        sband = wiphy->bands[NL80211_BAND_5GHZ];
        if (!sband)
                return;

        for (i = 0; i < sband->n_channels; i++) {
                ch = &sband->channels[i];

                /* UNII-4 */
                if (IS_UNII_INVALID(0, 5845, 5925, ch->center_freq))
                        ch->flags |= IEEE80211_CHAN_DISABLED;

                if (!dev->has_eht)
                        ch->flags |= IEEE80211_CHAN_NO_EHT;
        }

        sband = wiphy->bands[NL80211_BAND_6GHZ];
        if (!sband)
                return;

        for (i = 0; i < sband->n_channels; i++) {
                ch = &sband->channels[i];

                /* UNII-5/6/7/8 */
                if (IS_UNII_INVALID(1, 5925, 6425, ch->center_freq) ||
                    IS_UNII_INVALID(2, 6425, 6525, ch->center_freq) ||
                    IS_UNII_INVALID(3, 6525, 6875, ch->center_freq) ||
                    IS_UNII_INVALID(4, 6875, 7125, ch->center_freq))
                        ch->flags |= IEEE80211_CHAN_DISABLED;

                if (!dev->has_eht)
                        ch->flags |= IEEE80211_CHAN_NO_EHT;
        }
}

int mt7925_mcu_regd_update(struct mt792x_dev *dev, u8 *alpha2,
                           enum environment_cap country_ie_env)
{
        struct ieee80211_hw *hw = mt76_hw(dev);
        struct wiphy *wiphy = hw->wiphy;
        int ret = 0;

        dev->regd_in_progress = true;

        mt792x_mutex_acquire(dev);
        if (!dev->regd_change)
                goto err;

        ret = mt7925_mcu_set_clc(dev, alpha2, country_ie_env);
        if (ret < 0)
                goto err;

        mt7925_regd_be_ctrl(dev, alpha2);
        mt7925_regd_channel_update(wiphy, dev);

        ret = mt7925_mcu_set_channel_domain(hw->priv);
        if (ret < 0)
                goto err;

        ret = mt7925_set_tx_sar_pwr(hw, NULL);
        if (ret < 0)
                goto err;

err:
        mt792x_mutex_release(dev);
        dev->regd_change = false;
        dev->regd_in_progress = false;
        wake_up(&dev->wait);

        return ret;
}
EXPORT_SYMBOL_GPL(mt7925_mcu_regd_update);

void mt7925_regd_notifier(struct wiphy *wiphy, struct regulatory_request *req)
{
        struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
        struct mt792x_dev *dev = mt792x_hw_dev(hw);
        struct mt76_connac_pm *pm = &dev->pm;
        struct mt76_dev *mdev = &dev->mt76;

        if (req->initiator == NL80211_REGDOM_SET_BY_USER &&
            !dev->regd_user)
                dev->regd_user = true;

        /* allow world regdom at the first boot only */
        if (!memcmp(req->alpha2, "00", 2) &&
            mdev->alpha2[0] && mdev->alpha2[1])
                return;

        /* do not need to update the same country twice */
        if (!memcmp(req->alpha2, mdev->alpha2, 2) &&
            dev->country_ie_env == req->country_ie_env)
                return;

        memcpy(mdev->alpha2, req->alpha2, 2);
        mdev->region = req->dfs_region;
        dev->country_ie_env = req->country_ie_env;

        dev->regd_change = true;

        if (pm->suspended)
                /* postpone the mcu update to resume */
                return;

        mt7925_mcu_regd_update(dev, req->alpha2,
                               req->country_ie_env);
        return;
}

static bool
mt7925_regd_is_valid_alpha2(const char *alpha2)
{
        if (!alpha2)
                return false;

        if (alpha2[0] == '0' && alpha2[1] == '0')
                return true;

        if (isalpha(alpha2[0]) && isalpha(alpha2[1]))
                return true;

        return false;
}

int mt7925_regd_change(struct mt792x_phy *phy, char *alpha2)
{
        struct wiphy *wiphy = phy->mt76->hw->wiphy;
        struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
        struct mt792x_dev *dev = mt792x_hw_dev(hw);
        struct mt76_dev *mdev = &dev->mt76;

        if (dev->hw_full_reset)
                return 0;

        if (!mt7925_regd_is_valid_alpha2(alpha2) ||
            !mt7925_regd_clc_supported(dev) ||
            dev->regd_user)
                return -EINVAL;

        if (mdev->alpha2[0] != '0' && mdev->alpha2[1] != '0')
                return 0;

        /* do not need to update the same country twice */
        if (!memcmp(alpha2, mdev->alpha2, 2))
                return 0;

        if (phy->chip_cap & MT792x_CHIP_CAP_11D_EN) {
                return regulatory_hint(wiphy, alpha2);
        } else {
                return mt7925_mcu_set_clc(dev, alpha2, ENVIRON_INDOOR);
        }
}
EXPORT_SYMBOL_GPL(mt7925_regd_change);

int mt7925_regd_init(struct mt792x_phy *phy)
{
        struct wiphy *wiphy = phy->mt76->hw->wiphy;
        struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
        struct mt792x_dev *dev = mt792x_hw_dev(hw);
        struct mt76_dev *mdev = &dev->mt76;

        if (phy->chip_cap & MT792x_CHIP_CAP_11D_EN) {
                wiphy->regulatory_flags |= REGULATORY_COUNTRY_IE_IGNORE |
                                           REGULATORY_DISABLE_BEACON_HINTS;
        } else {
                memzero_explicit(&mdev->alpha2, sizeof(mdev->alpha2));
        }

        return 0;
}