Path: blob/main/sys/contrib/dev/mediatek/mt76/mt7925/mcu.c
48524 views
// SPDX-License-Identifier: ISC1/* Copyright (C) 2023 MediaTek Inc. */23#include <linux/fs.h>4#include <linux/firmware.h>5#include "mt7925.h"6#include "mcu.h"7#include "mac.h"89#define MT_STA_BFER BIT(0)10#define MT_STA_BFEE BIT(1)1112static bool mt7925_disable_clc;13module_param_named(disable_clc, mt7925_disable_clc, bool, 0644);14MODULE_PARM_DESC(disable_clc, "disable CLC support");1516int mt7925_mcu_parse_response(struct mt76_dev *mdev, int cmd,17struct sk_buff *skb, int seq)18{19int mcu_cmd = FIELD_GET(__MCU_CMD_FIELD_ID, cmd);20struct mt7925_mcu_rxd *rxd;21int ret = 0;2223if (!skb) {24dev_err(mdev->dev, "Message %08x (seq %d) timeout\n", cmd, seq);25mt792x_reset(mdev);2627return -ETIMEDOUT;28}2930rxd = (struct mt7925_mcu_rxd *)skb->data;31if (seq != rxd->seq)32return -EAGAIN;3334if (cmd == MCU_CMD(PATCH_SEM_CONTROL) ||35cmd == MCU_CMD(PATCH_FINISH_REQ)) {36skb_pull(skb, sizeof(*rxd) - 4);37ret = *skb->data;38} else if (cmd == MCU_UNI_CMD(DEV_INFO_UPDATE) ||39cmd == MCU_UNI_CMD(BSS_INFO_UPDATE) ||40cmd == MCU_UNI_CMD(STA_REC_UPDATE) ||41cmd == MCU_UNI_CMD(OFFLOAD) ||42cmd == MCU_UNI_CMD(SUSPEND)) {43struct mt7925_mcu_uni_event *event;4445skb_pull(skb, sizeof(*rxd));46event = (struct mt7925_mcu_uni_event *)skb->data;47ret = le32_to_cpu(event->status);48/* skip invalid event */49if (mcu_cmd != event->cid)50ret = -EAGAIN;51} else {52skb_pull(skb, sizeof(*rxd));53}5455return ret;56}57EXPORT_SYMBOL_GPL(mt7925_mcu_parse_response);5859int mt7925_mcu_regval(struct mt792x_dev *dev, u32 regidx, u32 *val, bool set)60{61#define MT_RF_REG_HDR GENMASK(31, 24)62#define MT_RF_REG_ANT GENMASK(23, 16)63#define RF_REG_PREFIX 0x9964struct {65u8 __rsv[4];66union {67struct uni_cmd_access_reg_basic {68__le16 tag;69__le16 len;70__le32 idx;71__le32 data;72} __packed reg;73struct uni_cmd_access_rf_reg_basic {74__le16 tag;75__le16 len;76__le16 ant;77u8 __rsv[2];78__le32 idx;79__le32 data;80} __packed rf_reg;81};82} __packed * res, req;83struct sk_buff *skb;84int ret;8586if (u32_get_bits(regidx, MT_RF_REG_HDR) == RF_REG_PREFIX) {87req.rf_reg.tag = cpu_to_le16(UNI_CMD_ACCESS_RF_REG_BASIC);88req.rf_reg.len = cpu_to_le16(sizeof(req.rf_reg));89req.rf_reg.ant = cpu_to_le16(u32_get_bits(regidx, MT_RF_REG_ANT));90req.rf_reg.idx = cpu_to_le32(regidx);91req.rf_reg.data = set ? cpu_to_le32(*val) : 0;92} else {93req.reg.tag = cpu_to_le16(UNI_CMD_ACCESS_REG_BASIC);94req.reg.len = cpu_to_le16(sizeof(req.reg));95req.reg.idx = cpu_to_le32(regidx);96req.reg.data = set ? cpu_to_le32(*val) : 0;97}9899if (set)100return mt76_mcu_send_msg(&dev->mt76, MCU_WM_UNI_CMD(REG_ACCESS),101&req, sizeof(req), true);102103ret = mt76_mcu_send_and_get_msg(&dev->mt76,104MCU_WM_UNI_CMD_QUERY(REG_ACCESS),105&req, sizeof(req), true, &skb);106if (ret)107return ret;108109res = (void *)skb->data;110if (u32_get_bits(regidx, MT_RF_REG_HDR) == RF_REG_PREFIX)111*val = le32_to_cpu(res->rf_reg.data);112else113*val = le32_to_cpu(res->reg.data);114115dev_kfree_skb(skb);116117return 0;118}119EXPORT_SYMBOL_GPL(mt7925_mcu_regval);120121int mt7925_mcu_update_arp_filter(struct mt76_dev *dev,122struct ieee80211_bss_conf *link_conf)123{124struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);125struct ieee80211_vif *mvif = link_conf->vif;126struct sk_buff *skb;127int i, len = min_t(int, mvif->cfg.arp_addr_cnt,128IEEE80211_BSS_ARP_ADDR_LIST_LEN);129struct {130struct {131u8 bss_idx;132u8 pad[3];133} __packed hdr;134struct mt7925_arpns_tlv arp;135} req = {136.hdr = {137.bss_idx = mconf->mt76.idx,138},139.arp = {140.tag = cpu_to_le16(UNI_OFFLOAD_OFFLOAD_ARP),141.len = cpu_to_le16(sizeof(req) - 4 + len * 2 * sizeof(__be32)),142.ips_num = len,143.enable = true,144},145};146147skb = mt76_mcu_msg_alloc(dev, NULL, sizeof(req) + len * 2 * sizeof(__be32));148if (!skb)149return -ENOMEM;150151skb_put_data(skb, &req, sizeof(req));152for (i = 0; i < len; i++) {153skb_put_data(skb, &mvif->cfg.arp_addr_list[i], sizeof(__be32));154skb_put_zero(skb, sizeof(__be32));155}156157return mt76_mcu_skb_send_msg(dev, skb, MCU_UNI_CMD(OFFLOAD), true);158}159160#ifdef CONFIG_PM161static int162mt7925_connac_mcu_set_wow_ctrl(struct mt76_phy *phy, struct ieee80211_vif *vif,163bool suspend, struct cfg80211_wowlan *wowlan)164{165struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;166struct ieee80211_scan_ies ies = {};167struct mt76_dev *dev = phy->dev;168struct {169struct {170u8 bss_idx;171u8 pad[3];172} __packed hdr;173struct mt76_connac_wow_ctrl_tlv wow_ctrl_tlv;174struct mt76_connac_wow_gpio_param_tlv gpio_tlv;175} req = {176.hdr = {177.bss_idx = mvif->idx,178},179.wow_ctrl_tlv = {180.tag = cpu_to_le16(UNI_SUSPEND_WOW_CTRL),181.len = cpu_to_le16(sizeof(struct mt76_connac_wow_ctrl_tlv)),182.cmd = suspend ? 1 : 2,183},184.gpio_tlv = {185.tag = cpu_to_le16(UNI_SUSPEND_WOW_GPIO_PARAM),186.len = cpu_to_le16(sizeof(struct mt76_connac_wow_gpio_param_tlv)),187.gpio_pin = 0xff, /* follow fw about GPIO pin */188},189};190191if (wowlan->magic_pkt)192req.wow_ctrl_tlv.trigger |= UNI_WOW_DETECT_TYPE_MAGIC;193if (wowlan->disconnect)194req.wow_ctrl_tlv.trigger |= (UNI_WOW_DETECT_TYPE_DISCONNECT |195UNI_WOW_DETECT_TYPE_BCN_LOST);196if (wowlan->nd_config) {197mt7925_mcu_sched_scan_req(phy, vif, wowlan->nd_config, &ies);198req.wow_ctrl_tlv.trigger |= UNI_WOW_DETECT_TYPE_SCH_SCAN_HIT;199mt7925_mcu_sched_scan_enable(phy, vif, suspend);200}201if (wowlan->n_patterns)202req.wow_ctrl_tlv.trigger |= UNI_WOW_DETECT_TYPE_BITMAP;203204if (mt76_is_mmio(dev))205req.wow_ctrl_tlv.wakeup_hif = WOW_PCIE;206else if (mt76_is_usb(dev))207req.wow_ctrl_tlv.wakeup_hif = WOW_USB;208else if (mt76_is_sdio(dev))209req.wow_ctrl_tlv.wakeup_hif = WOW_GPIO;210211return mt76_mcu_send_msg(dev, MCU_UNI_CMD(SUSPEND), &req,212sizeof(req), true);213}214215static int216mt7925_mcu_set_wow_pattern(struct mt76_dev *dev,217struct ieee80211_vif *vif,218u8 index, bool enable,219struct cfg80211_pkt_pattern *pattern)220{221struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;222struct mt7925_wow_pattern_tlv *tlv;223struct sk_buff *skb;224struct {225u8 bss_idx;226u8 pad[3];227} __packed hdr = {228.bss_idx = mvif->idx,229};230231skb = mt76_mcu_msg_alloc(dev, NULL, sizeof(hdr) + sizeof(*tlv));232if (!skb)233return -ENOMEM;234235skb_put_data(skb, &hdr, sizeof(hdr));236tlv = (struct mt7925_wow_pattern_tlv *)skb_put(skb, sizeof(*tlv));237tlv->tag = cpu_to_le16(UNI_SUSPEND_WOW_PATTERN);238tlv->len = cpu_to_le16(sizeof(*tlv));239tlv->bss_idx = 0xF;240tlv->data_len = pattern->pattern_len;241tlv->enable = enable;242tlv->index = index;243tlv->offset = 0;244245memcpy(tlv->pattern, pattern->pattern, pattern->pattern_len);246memcpy(tlv->mask, pattern->mask, DIV_ROUND_UP(pattern->pattern_len, 8));247248return mt76_mcu_skb_send_msg(dev, skb, MCU_UNI_CMD(SUSPEND), true);249}250251void mt7925_mcu_set_suspend_iter(void *priv, u8 *mac,252struct ieee80211_vif *vif)253{254struct mt76_phy *phy = priv;255bool suspend = !test_bit(MT76_STATE_RUNNING, &phy->state);256struct ieee80211_hw *hw = phy->hw;257struct cfg80211_wowlan *wowlan = hw->wiphy->wowlan_config;258int i;259260mt76_connac_mcu_set_gtk_rekey(phy->dev, vif, suspend);261262mt76_connac_mcu_set_suspend_mode(phy->dev, vif, suspend, 1, true);263264for (i = 0; i < wowlan->n_patterns; i++)265mt7925_mcu_set_wow_pattern(phy->dev, vif, i, suspend,266&wowlan->patterns[i]);267mt7925_connac_mcu_set_wow_ctrl(phy, vif, suspend, wowlan);268}269270#endif /* CONFIG_PM */271272static void273mt7925_mcu_connection_loss_iter(void *priv, u8 *mac,274struct ieee80211_vif *vif)275{276struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;277struct mt7925_uni_beacon_loss_event *event = priv;278279if (mvif->idx != event->hdr.bss_idx)280return;281282if (!(vif->driver_flags & IEEE80211_VIF_BEACON_FILTER) ||283vif->type != NL80211_IFTYPE_STATION)284return;285286ieee80211_connection_loss(vif);287}288289static void290mt7925_mcu_connection_loss_event(struct mt792x_dev *dev, struct sk_buff *skb)291{292struct mt7925_uni_beacon_loss_event *event;293struct mt76_phy *mphy = &dev->mt76.phy;294295skb_pull(skb, sizeof(struct mt7925_mcu_rxd));296event = (struct mt7925_uni_beacon_loss_event *)skb->data;297298ieee80211_iterate_active_interfaces_atomic(mphy->hw,299IEEE80211_IFACE_ITER_RESUME_ALL,300mt7925_mcu_connection_loss_iter, event);301}302303static void304mt7925_mcu_roc_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)305{306struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;307struct mt7925_roc_grant_tlv *grant = priv;308309if (ieee80211_vif_is_mld(vif) && vif->type == NL80211_IFTYPE_STATION)310return;311312if (mvif->idx != grant->bss_idx)313return;314315mvif->band_idx = grant->dbdcband;316}317318static void mt7925_mcu_roc_handle_grant(struct mt792x_dev *dev,319struct tlv *tlv)320{321struct ieee80211_hw *hw = dev->mt76.hw;322struct mt7925_roc_grant_tlv *grant;323int duration;324325grant = (struct mt7925_roc_grant_tlv *)tlv;326327/* should never happen */328WARN_ON_ONCE((le16_to_cpu(grant->tag) != UNI_EVENT_ROC_GRANT));329330if (grant->reqtype == MT7925_ROC_REQ_ROC)331ieee80211_ready_on_channel(hw);332else if (grant->reqtype == MT7925_ROC_REQ_JOIN)333ieee80211_iterate_active_interfaces_atomic(hw,334IEEE80211_IFACE_ITER_RESUME_ALL,335mt7925_mcu_roc_iter, grant);336dev->phy.roc_grant = true;337wake_up(&dev->phy.roc_wait);338duration = le32_to_cpu(grant->max_interval);339mod_timer(&dev->phy.roc_timer,340jiffies + msecs_to_jiffies(duration));341}342343static void344mt7925_mcu_handle_hif_ctrl_basic(struct mt792x_dev *dev, struct tlv *tlv)345{346struct mt7925_mcu_hif_ctrl_basic_tlv *basic;347348basic = (struct mt7925_mcu_hif_ctrl_basic_tlv *)tlv;349350if (basic->hifsuspend) {351dev->hif_idle = true;352if (!(basic->hif_tx_traffic_status == HIF_TRAFFIC_IDLE &&353basic->hif_rx_traffic_status == HIF_TRAFFIC_IDLE))354dev_info(dev->mt76.dev, "Hif traffic not idle.\n");355} else {356dev->hif_resumed = true;357}358wake_up(&dev->wait);359}360361static void362mt7925_mcu_uni_hif_ctrl_event(struct mt792x_dev *dev, struct sk_buff *skb)363{364struct tlv *tlv;365u32 tlv_len;366367skb_pull(skb, sizeof(struct mt7925_mcu_rxd) + 4);368tlv = (struct tlv *)skb->data;369tlv_len = skb->len;370371while (tlv_len > 0 && le16_to_cpu(tlv->len) <= tlv_len) {372switch (le16_to_cpu(tlv->tag)) {373case UNI_EVENT_HIF_CTRL_BASIC:374mt7925_mcu_handle_hif_ctrl_basic(dev, tlv);375break;376default:377break;378}379tlv_len -= le16_to_cpu(tlv->len);380tlv = (struct tlv *)((char *)(tlv) + le16_to_cpu(tlv->len));381}382}383384static void385mt7925_mcu_uni_roc_event(struct mt792x_dev *dev, struct sk_buff *skb)386{387struct tlv *tlv;388int i = 0;389390skb_pull(skb, sizeof(struct mt7925_mcu_rxd) + 4);391392while (i < skb->len) {393tlv = (struct tlv *)(skb->data + i);394395switch (le16_to_cpu(tlv->tag)) {396case UNI_EVENT_ROC_GRANT:397mt7925_mcu_roc_handle_grant(dev, tlv);398break;399case UNI_EVENT_ROC_GRANT_SUB_LINK:400break;401}402403i += le16_to_cpu(tlv->len);404}405}406407static void408mt7925_mcu_scan_event(struct mt792x_dev *dev, struct sk_buff *skb)409{410struct mt76_phy *mphy = &dev->mt76.phy;411struct mt792x_phy *phy = mphy->priv;412413spin_lock_bh(&dev->mt76.lock);414__skb_queue_tail(&phy->scan_event_list, skb);415spin_unlock_bh(&dev->mt76.lock);416417ieee80211_queue_delayed_work(mphy->hw, &phy->scan_work,418MT792x_HW_SCAN_TIMEOUT);419}420421static void422mt7925_mcu_tx_done_event(struct mt792x_dev *dev, struct sk_buff *skb)423{424#define UNI_EVENT_TX_DONE_MSG 0425#define UNI_EVENT_TX_DONE_RAW 1426struct mt7925_mcu_txs_event {427u8 ver;428u8 rsv[3];429u8 data[];430} __packed * txs;431struct tlv *tlv;432u32 tlv_len;433434skb_pull(skb, sizeof(struct mt7925_mcu_rxd) + 4);435tlv = (struct tlv *)skb->data;436tlv_len = skb->len;437438while (tlv_len > 0 && le16_to_cpu(tlv->len) <= tlv_len) {439switch (le16_to_cpu(tlv->tag)) {440case UNI_EVENT_TX_DONE_RAW:441txs = (struct mt7925_mcu_txs_event *)tlv->data;442mt7925_mac_add_txs(dev, txs->data);443break;444default:445break;446}447tlv_len -= le16_to_cpu(tlv->len);448tlv = (struct tlv *)((char *)(tlv) + le16_to_cpu(tlv->len));449}450}451452static void453mt7925_mcu_uni_debug_msg_event(struct mt792x_dev *dev, struct sk_buff *skb)454{455struct mt7925_uni_debug_msg {456__le16 tag;457__le16 len;458u8 fmt;459u8 rsv[3];460u8 id;461u8 type:3;462u8 nr_args:5;463union {464struct idxlog {465__le16 rsv;466__le32 ts;467__le32 idx;468u8 data[];469} __packed idx;470struct txtlog {471u8 len;472u8 rsv;473__le32 ts;474u8 data[];475} __packed txt;476};477} __packed * hdr;478479skb_pull(skb, sizeof(struct mt7925_mcu_rxd) + 4);480hdr = (struct mt7925_uni_debug_msg *)skb->data;481482if (hdr->id == 0x28) {483skb_pull(skb, offsetof(struct mt7925_uni_debug_msg, id));484wiphy_info(mt76_hw(dev)->wiphy, "%.*s", skb->len, skb->data);485return;486} else if (hdr->id != 0xa8) {487return;488}489490if (hdr->type == 0) { /* idx log */491int i, ret, len = PAGE_SIZE - 1, nr_val;492struct page *page = dev_alloc_pages(get_order(len));493__le32 *val;494char *buf, *cur;495496if (!page)497return;498499buf = page_address(page);500cur = buf;501502nr_val = (le16_to_cpu(hdr->len) - sizeof(*hdr)) / 4;503val = (__le32 *)hdr->idx.data;504for (i = 0; i < nr_val && len > 0; i++) {505ret = snprintf(cur, len, "0x%x,", le32_to_cpu(val[i]));506if (ret <= 0)507break;508509cur += ret;510len -= ret;511}512if (cur > buf)513wiphy_info(mt76_hw(dev)->wiphy, "idx: 0x%X,%d,%s",514le32_to_cpu(hdr->idx.idx), nr_val, buf);515put_page(page);516} else if (hdr->type == 2) { /* str log */517wiphy_info(mt76_hw(dev)->wiphy, "%.*s", hdr->txt.len, hdr->txt.data);518}519}520521static void522mt7925_mcu_uni_rx_unsolicited_event(struct mt792x_dev *dev,523struct sk_buff *skb)524{525struct mt7925_mcu_rxd *rxd;526527rxd = (struct mt7925_mcu_rxd *)skb->data;528529switch (rxd->eid) {530case MCU_UNI_EVENT_HIF_CTRL:531mt7925_mcu_uni_hif_ctrl_event(dev, skb);532break;533case MCU_UNI_EVENT_FW_LOG_2_HOST:534mt7925_mcu_uni_debug_msg_event(dev, skb);535break;536case MCU_UNI_EVENT_ROC:537mt7925_mcu_uni_roc_event(dev, skb);538break;539case MCU_UNI_EVENT_SCAN_DONE:540mt7925_mcu_scan_event(dev, skb);541return;542case MCU_UNI_EVENT_TX_DONE:543mt7925_mcu_tx_done_event(dev, skb);544break;545case MCU_UNI_EVENT_BSS_BEACON_LOSS:546mt7925_mcu_connection_loss_event(dev, skb);547break;548case MCU_UNI_EVENT_COREDUMP:549dev->fw_assert = true;550mt76_connac_mcu_coredump_event(&dev->mt76, skb, &dev->coredump);551return;552default:553break;554}555dev_kfree_skb(skb);556}557558void mt7925_mcu_rx_event(struct mt792x_dev *dev, struct sk_buff *skb)559{560struct mt7925_mcu_rxd *rxd = (struct mt7925_mcu_rxd *)skb->data;561562if (skb_linearize(skb))563return;564565if (rxd->option & MCU_UNI_CMD_UNSOLICITED_EVENT) {566mt7925_mcu_uni_rx_unsolicited_event(dev, skb);567return;568}569570mt76_mcu_rx_event(&dev->mt76, skb);571}572573static int574mt7925_mcu_sta_ba(struct mt76_dev *dev, struct mt76_vif_link *mvif,575struct ieee80211_ampdu_params *params,576bool enable, bool tx)577{578struct mt76_wcid *wcid = (struct mt76_wcid *)params->sta->drv_priv;579struct sta_rec_ba_uni *ba;580struct sk_buff *skb;581struct tlv *tlv;582int len;583584len = sizeof(struct sta_req_hdr) + sizeof(*ba);585skb = __mt76_connac_mcu_alloc_sta_req(dev, mvif, wcid,586len);587if (IS_ERR(skb))588return PTR_ERR(skb);589590tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_BA, sizeof(*ba));591592ba = (struct sta_rec_ba_uni *)tlv;593ba->ba_type = tx ? MT_BA_TYPE_ORIGINATOR : MT_BA_TYPE_RECIPIENT;594ba->winsize = cpu_to_le16(params->buf_size);595ba->ssn = cpu_to_le16(params->ssn);596ba->ba_en = enable << params->tid;597ba->amsdu = params->amsdu;598ba->tid = params->tid;599600return mt76_mcu_skb_send_msg(dev, skb,601MCU_UNI_CMD(STA_REC_UPDATE), true);602}603604/** starec & wtbl **/605int mt7925_mcu_uni_tx_ba(struct mt792x_dev *dev,606struct ieee80211_ampdu_params *params,607bool enable)608{609struct mt792x_sta *msta = (struct mt792x_sta *)params->sta->drv_priv;610struct mt792x_vif *mvif = msta->vif;611612if (enable && !params->amsdu)613msta->deflink.wcid.amsdu = false;614615return mt7925_mcu_sta_ba(&dev->mt76, &mvif->bss_conf.mt76, params,616enable, true);617}618619int mt7925_mcu_uni_rx_ba(struct mt792x_dev *dev,620struct ieee80211_ampdu_params *params,621bool enable)622{623struct mt792x_sta *msta = (struct mt792x_sta *)params->sta->drv_priv;624struct mt792x_vif *mvif = msta->vif;625626return mt7925_mcu_sta_ba(&dev->mt76, &mvif->bss_conf.mt76, params,627enable, false);628}629630static int mt7925_mcu_read_eeprom(struct mt792x_dev *dev, u32 offset, u8 *val)631{632struct {633u8 rsv[4];634635__le16 tag;636__le16 len;637638__le32 addr;639__le32 valid;640u8 data[MT7925_EEPROM_BLOCK_SIZE];641} __packed req = {642.tag = cpu_to_le16(1),643.len = cpu_to_le16(sizeof(req) - 4),644.addr = cpu_to_le32(round_down(offset,645MT7925_EEPROM_BLOCK_SIZE)),646};647struct evt {648u8 rsv[4];649650__le16 tag;651__le16 len;652653__le32 ver;654__le32 addr;655__le32 valid;656__le32 size;657__le32 magic_num;658__le32 type;659__le32 rsv1[4];660u8 data[32];661} __packed *res;662struct sk_buff *skb;663int ret;664665ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_WM_UNI_CMD_QUERY(EFUSE_CTRL),666&req, sizeof(req), true, &skb);667if (ret)668return ret;669670res = (struct evt *)skb->data;671*val = res->data[offset % MT7925_EEPROM_BLOCK_SIZE];672673dev_kfree_skb(skb);674675return 0;676}677678static int mt7925_load_clc(struct mt792x_dev *dev, const char *fw_name)679{680const struct mt76_connac2_fw_trailer *hdr;681const struct mt76_connac2_fw_region *region;682const struct mt7925_clc *clc;683struct mt76_dev *mdev = &dev->mt76;684struct mt792x_phy *phy = &dev->phy;685const struct firmware *fw;686#if defined(__linux__)687u8 *clc_base = NULL, hw_encap = 0;688#elif defined(__FreeBSD__)689const u8 *clc_base = NULL;690u8 hw_encap = 0;691#endif692int ret, i, len, offset = 0;693694dev->phy.clc_chan_conf = 0xff;695if (mt7925_disable_clc ||696mt76_is_usb(&dev->mt76))697return 0;698699if (mt76_is_mmio(&dev->mt76)) {700ret = mt7925_mcu_read_eeprom(dev, MT_EE_HW_TYPE, &hw_encap);701if (ret)702return ret;703hw_encap = u8_get_bits(hw_encap, MT_EE_HW_TYPE_ENCAP);704}705706ret = request_firmware(&fw, fw_name, mdev->dev);707if (ret)708return ret;709710if (!fw || !fw->data || fw->size < sizeof(*hdr)) {711dev_err(mdev->dev, "Invalid firmware\n");712ret = -EINVAL;713goto out;714}715716hdr = (const void *)(fw->data + fw->size - sizeof(*hdr));717for (i = 0; i < hdr->n_region; i++) {718region = (const void *)((const u8 *)hdr -719(hdr->n_region - i) * sizeof(*region));720len = le32_to_cpu(region->len);721722/* check if we have valid buffer size */723if (offset + len > fw->size) {724dev_err(mdev->dev, "Invalid firmware region\n");725ret = -EINVAL;726goto out;727}728729if ((region->feature_set & FW_FEATURE_NON_DL) &&730region->type == FW_TYPE_CLC) {731#if defined(__linux__)732clc_base = (u8 *)(fw->data + offset);733#elif defined(__FreeBSD__)734clc_base = (const u8 *)(fw->data + offset);735#endif736break;737}738offset += len;739}740741if (!clc_base)742goto out;743744for (offset = 0; offset < len; offset += le32_to_cpu(clc->len)) {745clc = (const struct mt7925_clc *)(clc_base + offset);746747if (clc->idx >= ARRAY_SIZE(phy->clc))748break;749750/* do not init buf again if chip reset triggered */751if (phy->clc[clc->idx])752continue;753754/* header content sanity */755if ((clc->idx == MT792x_CLC_BE_CTRL &&756u8_get_bits(clc->t2.type, MT_EE_HW_TYPE_ENCAP) != hw_encap) ||757u8_get_bits(clc->t0.type, MT_EE_HW_TYPE_ENCAP) != hw_encap)758continue;759760phy->clc[clc->idx] = devm_kmemdup(mdev->dev, clc,761le32_to_cpu(clc->len),762GFP_KERNEL);763764if (!phy->clc[clc->idx]) {765ret = -ENOMEM;766goto out;767}768}769770ret = mt7925_mcu_set_clc(dev, "00", ENVIRON_INDOOR);771out:772release_firmware(fw);773774return ret;775}776777int mt7925_mcu_fw_log_2_host(struct mt792x_dev *dev, u8 ctrl)778{779struct {780u8 _rsv[4];781782__le16 tag;783__le16 len;784u8 ctrl;785u8 interval;786u8 _rsv2[2];787} __packed req = {788.tag = cpu_to_le16(UNI_WSYS_CONFIG_FW_LOG_CTRL),789.len = cpu_to_le16(sizeof(req) - 4),790.ctrl = ctrl,791};792int ret;793794ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_CMD(WSYS_CONFIG),795&req, sizeof(req), true, NULL);796return ret;797}798799int mt7925_mcu_get_temperature(struct mt792x_phy *phy)800{801struct {802u8 _rsv[4];803804__le16 tag;805__le16 len;806u8 _rsv2[4];807} __packed req = {808.tag = cpu_to_le16(0x0),809.len = cpu_to_le16(sizeof(req) - 4),810};811struct mt7925_thermal_evt {812u8 rsv[4];813__le32 temperature;814} __packed * evt;815struct mt792x_dev *dev = phy->dev;816int temperature, ret;817struct sk_buff *skb;818819ret = mt76_mcu_send_and_get_msg(&dev->mt76,820MCU_WM_UNI_CMD_QUERY(THERMAL),821&req, sizeof(req), true, &skb);822if (ret)823return ret;824825skb_pull(skb, 4 + sizeof(struct tlv));826evt = (struct mt7925_thermal_evt *)skb->data;827828temperature = le32_to_cpu(evt->temperature);829830dev_kfree_skb(skb);831832return temperature;833}834835static void836mt7925_mcu_parse_phy_cap(struct mt792x_dev *dev, char *data)837{838struct mt76_phy *mphy = &dev->mt76.phy;839struct mt76_dev *mdev = mphy->dev;840struct mt7925_mcu_phy_cap {841u8 ht;842u8 vht;843u8 _5g;844u8 max_bw;845u8 nss;846u8 dbdc;847u8 tx_ldpc;848u8 rx_ldpc;849u8 tx_stbc;850u8 rx_stbc;851u8 hw_path;852u8 he;853u8 eht;854} __packed * cap;855enum {856WF0_24G,857WF0_5G858};859860cap = (struct mt7925_mcu_phy_cap *)data;861862mdev->phy.antenna_mask = BIT(cap->nss) - 1;863mdev->phy.chainmask = mdev->phy.antenna_mask;864mdev->phy.cap.has_2ghz = cap->hw_path & BIT(WF0_24G);865mdev->phy.cap.has_5ghz = cap->hw_path & BIT(WF0_5G);866}867868static void869mt7925_mcu_parse_eml_cap(struct mt792x_dev *dev, char *data)870{871struct mt7925_mcu_eml_cap {872u8 rsv[4];873__le16 eml_cap;874u8 rsv2[6];875} __packed * cap;876877cap = (struct mt7925_mcu_eml_cap *)data;878879dev->phy.eml_cap = le16_to_cpu(cap->eml_cap);880}881882static int883mt7925_mcu_get_nic_capability(struct mt792x_dev *dev)884{885struct mt76_phy *mphy = &dev->mt76.phy;886struct {887u8 _rsv[4];888889__le16 tag;890__le16 len;891} __packed req = {892.tag = cpu_to_le16(UNI_CHIP_CONFIG_NIC_CAPA),893.len = cpu_to_le16(sizeof(req) - 4),894};895struct mt76_connac_cap_hdr {896__le16 n_element;897u8 rsv[2];898} __packed * hdr;899struct sk_buff *skb;900int ret, i;901902ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_CMD(CHIP_CONFIG),903&req, sizeof(req), true, &skb);904if (ret)905return ret;906907hdr = (struct mt76_connac_cap_hdr *)skb->data;908if (skb->len < sizeof(*hdr)) {909ret = -EINVAL;910goto out;911}912913skb_pull(skb, sizeof(*hdr));914915for (i = 0; i < le16_to_cpu(hdr->n_element); i++) {916struct tlv *tlv = (struct tlv *)skb->data;917int len;918919if (skb->len < sizeof(*tlv))920break;921922len = le16_to_cpu(tlv->len);923if (skb->len < len)924break;925926switch (le16_to_cpu(tlv->tag)) {927case MT_NIC_CAP_6G:928mphy->cap.has_6ghz = !!tlv->data[0];929break;930case MT_NIC_CAP_MAC_ADDR:931memcpy(mphy->macaddr, (void *)tlv->data, ETH_ALEN);932break;933case MT_NIC_CAP_PHY:934mt7925_mcu_parse_phy_cap(dev, tlv->data);935break;936case MT_NIC_CAP_CHIP_CAP:937dev->phy.chip_cap = le64_to_cpu(*(__le64 *)tlv->data);938break;939case MT_NIC_CAP_EML_CAP:940mt7925_mcu_parse_eml_cap(dev, tlv->data);941break;942default:943break;944}945skb_pull(skb, len);946}947out:948dev_kfree_skb(skb);949return ret;950}951952int mt7925_mcu_chip_config(struct mt792x_dev *dev, const char *cmd)953{954u16 len = strlen(cmd) + 1;955struct {956u8 _rsv[4];957__le16 tag;958__le16 len;959struct mt76_connac_config config;960} __packed req = {961.tag = cpu_to_le16(UNI_CHIP_CONFIG_CHIP_CFG),962.len = cpu_to_le16(sizeof(req) - 4),963.config = {964.resp_type = 0,965.type = 0,966.data_size = cpu_to_le16(len),967},968};969970memcpy(req.config.data, cmd, len);971972return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(CHIP_CONFIG),973&req, sizeof(req), false);974}975976int mt7925_mcu_set_deep_sleep(struct mt792x_dev *dev, bool enable)977{978char cmd[16];979980snprintf(cmd, sizeof(cmd), "KeepFullPwr %d", !enable);981982return mt7925_mcu_chip_config(dev, cmd);983}984EXPORT_SYMBOL_GPL(mt7925_mcu_set_deep_sleep);985986int mt7925_mcu_set_thermal_protect(struct mt792x_dev *dev)987{988char cmd[64];989int ret = 0;990991snprintf(cmd, sizeof(cmd), "ThermalProtGband %d %d %d %d %d %d %d %d %d %d",9920, 100, 90, 80, 30, 1, 1, 115, 105, 5);993ret = mt7925_mcu_chip_config(dev, cmd);994995snprintf(cmd, sizeof(cmd), "ThermalProtAband %d %d %d %d %d %d %d %d %d %d",9961, 100, 90, 80, 30, 1, 1, 115, 105, 5);997ret |= mt7925_mcu_chip_config(dev, cmd);998999return ret;1000}1001EXPORT_SYMBOL_GPL(mt7925_mcu_set_thermal_protect);10021003int mt7925_run_firmware(struct mt792x_dev *dev)1004{1005int err;10061007err = mt792x_load_firmware(dev);1008if (err)1009return err;10101011err = mt7925_mcu_get_nic_capability(dev);1012if (err)1013return err;10141015set_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);1016err = mt7925_load_clc(dev, mt792x_ram_name(dev));1017if (err)1018return err;10191020return mt7925_mcu_fw_log_2_host(dev, 1);1021}1022EXPORT_SYMBOL_GPL(mt7925_run_firmware);10231024static void1025mt7925_mcu_sta_hdr_trans_tlv(struct sk_buff *skb,1026struct ieee80211_vif *vif,1027struct ieee80211_link_sta *link_sta)1028{1029struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1030struct sta_rec_hdr_trans *hdr_trans;1031struct mt76_wcid *wcid;1032struct tlv *tlv;10331034tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_HDR_TRANS, sizeof(*hdr_trans));1035hdr_trans = (struct sta_rec_hdr_trans *)tlv;1036hdr_trans->dis_rx_hdr_tran = true;10371038if (vif->type == NL80211_IFTYPE_STATION)1039hdr_trans->to_ds = true;1040else1041hdr_trans->from_ds = true;10421043if (link_sta) {1044struct mt792x_sta *msta = (struct mt792x_sta *)link_sta->sta->drv_priv;1045struct mt792x_link_sta *mlink;10461047mlink = mt792x_sta_to_link(msta, link_sta->link_id);1048wcid = &mlink->wcid;1049} else {1050wcid = &mvif->sta.deflink.wcid;1051}10521053if (!wcid)1054return;10551056hdr_trans->dis_rx_hdr_tran = !test_bit(MT_WCID_FLAG_HDR_TRANS, &wcid->flags);1057if (test_bit(MT_WCID_FLAG_4ADDR, &wcid->flags)) {1058hdr_trans->to_ds = true;1059hdr_trans->from_ds = true;1060}1061}10621063int mt7925_mcu_wtbl_update_hdr_trans(struct mt792x_dev *dev,1064struct ieee80211_vif *vif,1065struct ieee80211_sta *sta,1066int link_id)1067{1068struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1069struct ieee80211_link_sta *link_sta = sta ? &sta->deflink : NULL;1070struct mt792x_link_sta *mlink;1071struct mt792x_bss_conf *mconf;1072struct mt792x_sta *msta;1073struct sk_buff *skb;10741075msta = sta ? (struct mt792x_sta *)sta->drv_priv : &mvif->sta;10761077mlink = mt792x_sta_to_link(msta, link_id);1078link_sta = mt792x_sta_to_link_sta(vif, sta, link_id);1079mconf = mt792x_vif_to_link(mvif, link_id);10801081skb = __mt76_connac_mcu_alloc_sta_req(&dev->mt76, &mconf->mt76,1082&mlink->wcid,1083MT7925_STA_UPDATE_MAX_SIZE);1084if (IS_ERR(skb))1085return PTR_ERR(skb);10861087/* starec hdr trans */1088mt7925_mcu_sta_hdr_trans_tlv(skb, vif, link_sta);1089return mt76_mcu_skb_send_msg(&dev->mt76, skb,1090MCU_WMWA_UNI_CMD(STA_REC_UPDATE), true);1091}10921093int mt7925_mcu_set_tx(struct mt792x_dev *dev,1094struct ieee80211_bss_conf *bss_conf)1095{1096#define MCU_EDCA_AC_PARAM 01097#define WMM_AIFS_SET BIT(0)1098#define WMM_CW_MIN_SET BIT(1)1099#define WMM_CW_MAX_SET BIT(2)1100#define WMM_TXOP_SET BIT(3)1101#define WMM_PARAM_SET (WMM_AIFS_SET | WMM_CW_MIN_SET | \1102WMM_CW_MAX_SET | WMM_TXOP_SET)1103struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(bss_conf);1104struct {1105u8 bss_idx;1106u8 __rsv[3];1107} __packed hdr = {1108.bss_idx = mconf->mt76.idx,1109};1110struct sk_buff *skb;1111int len = sizeof(hdr) + IEEE80211_NUM_ACS * sizeof(struct edca);1112int ac;11131114skb = mt76_mcu_msg_alloc(&dev->mt76, NULL, len);1115if (!skb)1116return -ENOMEM;11171118skb_put_data(skb, &hdr, sizeof(hdr));11191120for (ac = 0; ac < IEEE80211_NUM_ACS; ac++) {1121struct ieee80211_tx_queue_params *q = &mconf->queue_params[ac];1122struct edca *e;1123struct tlv *tlv;11241125tlv = mt76_connac_mcu_add_tlv(skb, MCU_EDCA_AC_PARAM, sizeof(*e));11261127e = (struct edca *)tlv;1128e->set = WMM_PARAM_SET;1129e->queue = ac;1130e->aifs = q->aifs;1131e->txop = cpu_to_le16(q->txop);11321133if (q->cw_min)1134e->cw_min = fls(q->cw_min);1135else1136e->cw_min = 5;11371138if (q->cw_max)1139e->cw_max = fls(q->cw_max);1140else1141e->cw_max = 10;1142}11431144return mt76_mcu_skb_send_msg(&dev->mt76, skb,1145MCU_UNI_CMD(EDCA_UPDATE), true);1146}11471148static int1149mt7925_mcu_sta_key_tlv(struct mt76_wcid *wcid,1150struct mt76_connac_sta_key_conf *sta_key_conf,1151struct sk_buff *skb,1152struct ieee80211_key_conf *key,1153enum set_key_cmd cmd,1154struct mt792x_sta *msta)1155{1156struct mt792x_vif *mvif = msta->vif;1157struct mt792x_bss_conf *mconf = mt792x_vif_to_link(mvif, wcid->link_id);1158struct sta_rec_sec_uni *sec;1159struct ieee80211_sta *sta;1160struct ieee80211_vif *vif;1161struct tlv *tlv;11621163sta = msta == &mvif->sta ?1164NULL :1165container_of((void *)msta, struct ieee80211_sta, drv_priv);1166vif = container_of((void *)mvif, struct ieee80211_vif, drv_priv);11671168tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_KEY_V3, sizeof(*sec));1169sec = (struct sta_rec_sec_uni *)tlv;1170sec->bss_idx = mconf->mt76.idx;1171sec->is_authenticator = 0;1172sec->mgmt_prot = 1; /* only used in MLO mode */1173sec->wlan_idx = (u8)wcid->idx;11741175if (sta) {1176struct ieee80211_link_sta *link_sta;11771178sec->tx_key = 1;1179sec->key_type = 1;1180link_sta = mt792x_sta_to_link_sta(vif, sta, wcid->link_id);11811182if (link_sta)1183memcpy(sec->peer_addr, link_sta->addr, ETH_ALEN);1184} else {1185struct ieee80211_bss_conf *link_conf;11861187link_conf = mt792x_vif_to_bss_conf(vif, wcid->link_id);11881189if (link_conf)1190memcpy(sec->peer_addr, link_conf->bssid, ETH_ALEN);1191}11921193if (cmd == SET_KEY) {1194u8 cipher;11951196sec->add = 1;1197cipher = mt7925_mcu_get_cipher(key->cipher);1198if (cipher == CONNAC3_CIPHER_NONE)1199return -EOPNOTSUPP;12001201if (cipher == CONNAC3_CIPHER_BIP_CMAC_128) {1202sec->cipher_id = CONNAC3_CIPHER_BIP_CMAC_128;1203sec->key_id = sta_key_conf->keyidx;1204sec->key_len = 32;1205memcpy(sec->key, sta_key_conf->key, 16);1206memcpy(sec->key + 16, key->key, 16);1207} else {1208sec->cipher_id = cipher;1209sec->key_id = key->keyidx;1210sec->key_len = key->keylen;1211memcpy(sec->key, key->key, key->keylen);12121213if (cipher == CONNAC3_CIPHER_TKIP) {1214/* Rx/Tx MIC keys are swapped */1215memcpy(sec->key + 16, key->key + 24, 8);1216memcpy(sec->key + 24, key->key + 16, 8);1217}12181219/* store key_conf for BIP batch update */1220if (cipher == CONNAC3_CIPHER_AES_CCMP) {1221memcpy(sta_key_conf->key, key->key, key->keylen);1222sta_key_conf->keyidx = key->keyidx;1223}1224}1225} else {1226sec->add = 0;1227}12281229return 0;1230}12311232int mt7925_mcu_add_key(struct mt76_dev *dev, struct ieee80211_vif *vif,1233struct mt76_connac_sta_key_conf *sta_key_conf,1234struct ieee80211_key_conf *key, int mcu_cmd,1235struct mt76_wcid *wcid, enum set_key_cmd cmd,1236struct mt792x_sta *msta)1237{1238struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1239struct mt792x_bss_conf *mconf = mt792x_vif_to_link(mvif, wcid->link_id);1240struct sk_buff *skb;1241int ret;12421243skb = __mt76_connac_mcu_alloc_sta_req(dev, &mconf->mt76, wcid,1244MT7925_STA_UPDATE_MAX_SIZE);1245if (IS_ERR(skb))1246return PTR_ERR(skb);12471248ret = mt7925_mcu_sta_key_tlv(wcid, sta_key_conf, skb, key, cmd, msta);1249if (ret)1250return ret;12511252return mt76_mcu_skb_send_msg(dev, skb, mcu_cmd, true);1253}12541255int mt7925_mcu_set_mlo_roc(struct mt792x_bss_conf *mconf, u16 sel_links,1256int duration, u8 token_id)1257{1258struct mt792x_vif *mvif = mconf->vif;1259struct ieee80211_vif *vif = container_of((void *)mvif,1260struct ieee80211_vif, drv_priv);1261struct ieee80211_bss_conf *link_conf;1262struct ieee80211_channel *chan;1263const u8 ch_band[] = {1264[NL80211_BAND_2GHZ] = 1,1265[NL80211_BAND_5GHZ] = 2,1266[NL80211_BAND_6GHZ] = 3,1267};1268enum mt7925_roc_req type;1269int center_ch, i = 0;1270bool is_AG_band = false;1271struct {1272u8 id;1273u8 bss_idx;1274u16 tag;1275struct mt792x_bss_conf *mconf;1276struct ieee80211_channel *chan;1277} links[2];12781279struct {1280struct {1281u8 rsv[4];1282} __packed hdr;1283struct roc_acquire_tlv roc[2];1284} __packed req = {1285.roc[0].tag = cpu_to_le16(UNI_ROC_NUM),1286.roc[0].len = cpu_to_le16(sizeof(struct roc_acquire_tlv)),1287.roc[1].tag = cpu_to_le16(UNI_ROC_NUM),1288.roc[1].len = cpu_to_le16(sizeof(struct roc_acquire_tlv))1289};12901291if (!mconf || hweight16(vif->valid_links) < 2 ||1292hweight16(sel_links) != 2)1293return -EPERM;12941295for (i = 0; i < ARRAY_SIZE(links); i++) {1296links[i].id = i ? __ffs(~BIT(mconf->link_id) & sel_links) :1297mconf->link_id;1298link_conf = mt792x_vif_to_bss_conf(vif, links[i].id);1299if (WARN_ON_ONCE(!link_conf))1300return -EPERM;13011302links[i].chan = link_conf->chanreq.oper.chan;1303if (WARN_ON_ONCE(!links[i].chan))1304return -EPERM;13051306links[i].mconf = mt792x_vif_to_link(mvif, links[i].id);1307links[i].tag = links[i].id == mconf->link_id ?1308UNI_ROC_ACQUIRE : UNI_ROC_SUB_LINK;13091310is_AG_band |= links[i].chan->band == NL80211_BAND_2GHZ;1311}13121313if (vif->cfg.eml_cap & IEEE80211_EML_CAP_EMLSR_SUPP)1314type = is_AG_band ? MT7925_ROC_REQ_MLSR_AG :1315MT7925_ROC_REQ_MLSR_AA;1316else1317type = MT7925_ROC_REQ_JOIN;13181319for (i = 0; i < ARRAY_SIZE(links) && i < hweight16(vif->active_links); i++) {1320if (WARN_ON_ONCE(!links[i].mconf || !links[i].chan))1321continue;13221323chan = links[i].chan;1324center_ch = ieee80211_frequency_to_channel(chan->center_freq);1325req.roc[i].len = cpu_to_le16(sizeof(struct roc_acquire_tlv));1326req.roc[i].tag = cpu_to_le16(links[i].tag);1327req.roc[i].tokenid = token_id;1328req.roc[i].reqtype = type;1329req.roc[i].maxinterval = cpu_to_le32(duration);1330req.roc[i].bss_idx = links[i].mconf->mt76.idx;1331req.roc[i].control_channel = chan->hw_value;1332req.roc[i].bw = CMD_CBW_20MHZ;1333req.roc[i].bw_from_ap = CMD_CBW_20MHZ;1334req.roc[i].center_chan = center_ch;1335req.roc[i].center_chan_from_ap = center_ch;1336req.roc[i].center_chan2 = 0;1337req.roc[i].center_chan2_from_ap = 0;13381339/* STR : 0xfe indicates BAND_ALL with enabling DBDC1340* EMLSR : 0xff indicates (BAND_AUTO) without DBDC1341*/1342req.roc[i].dbdcband = type == MT7925_ROC_REQ_JOIN ? 0xfe : 0xff;13431344if (chan->hw_value < center_ch)1345req.roc[i].sco = 1; /* SCA */1346else if (chan->hw_value > center_ch)1347req.roc[i].sco = 3; /* SCB */13481349req.roc[i].band = ch_band[chan->band];1350}13511352return mt76_mcu_send_msg(&mvif->phy->dev->mt76, MCU_UNI_CMD(ROC),1353&req, sizeof(req), true);1354}13551356int mt7925_mcu_set_roc(struct mt792x_phy *phy, struct mt792x_bss_conf *mconf,1357struct ieee80211_channel *chan, int duration,1358enum mt7925_roc_req type, u8 token_id)1359{1360int center_ch = ieee80211_frequency_to_channel(chan->center_freq);1361struct mt792x_dev *dev = phy->dev;1362struct {1363struct {1364u8 rsv[4];1365} __packed hdr;1366struct roc_acquire_tlv roc;1367} __packed req = {1368.roc = {1369.tag = cpu_to_le16(UNI_ROC_ACQUIRE),1370.len = cpu_to_le16(sizeof(struct roc_acquire_tlv)),1371.tokenid = token_id,1372.reqtype = type,1373.maxinterval = cpu_to_le32(duration),1374.bss_idx = mconf->mt76.idx,1375.control_channel = chan->hw_value,1376.bw = CMD_CBW_20MHZ,1377.bw_from_ap = CMD_CBW_20MHZ,1378.center_chan = center_ch,1379.center_chan_from_ap = center_ch,1380.dbdcband = 0xff, /* auto */1381},1382};13831384if (chan->hw_value < center_ch)1385req.roc.sco = 1; /* SCA */1386else if (chan->hw_value > center_ch)1387req.roc.sco = 3; /* SCB */13881389switch (chan->band) {1390case NL80211_BAND_6GHZ:1391req.roc.band = 3;1392break;1393case NL80211_BAND_5GHZ:1394req.roc.band = 2;1395break;1396default:1397req.roc.band = 1;1398break;1399}14001401return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(ROC),1402&req, sizeof(req), true);1403}14041405int mt7925_mcu_abort_roc(struct mt792x_phy *phy, struct mt792x_bss_conf *mconf,1406u8 token_id)1407{1408struct mt792x_dev *dev = phy->dev;1409struct {1410struct {1411u8 rsv[4];1412} __packed hdr;1413struct roc_abort_tlv {1414__le16 tag;1415__le16 len;1416u8 bss_idx;1417u8 tokenid;1418u8 dbdcband;1419u8 rsv[5];1420} __packed abort;1421} __packed req = {1422.abort = {1423.tag = cpu_to_le16(UNI_ROC_ABORT),1424.len = cpu_to_le16(sizeof(struct roc_abort_tlv)),1425.tokenid = token_id,1426.bss_idx = mconf->mt76.idx,1427.dbdcband = 0xff, /* auto*/1428},1429};14301431return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(ROC),1432&req, sizeof(req), true);1433}14341435int mt7925_mcu_set_eeprom(struct mt792x_dev *dev)1436{1437struct {1438u8 _rsv[4];14391440__le16 tag;1441__le16 len;1442u8 buffer_mode;1443u8 format;1444__le16 buf_len;1445} __packed req = {1446.tag = cpu_to_le16(UNI_EFUSE_BUFFER_MODE),1447.len = cpu_to_le16(sizeof(req) - 4),1448.buffer_mode = EE_MODE_EFUSE,1449.format = EE_FORMAT_WHOLE1450};14511452return mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_CMD(EFUSE_CTRL),1453&req, sizeof(req), true, NULL);1454}1455EXPORT_SYMBOL_GPL(mt7925_mcu_set_eeprom);14561457int mt7925_mcu_uni_bss_ps(struct mt792x_dev *dev,1458struct ieee80211_bss_conf *link_conf)1459{1460struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);1461struct {1462struct {1463u8 bss_idx;1464u8 pad[3];1465} __packed hdr;1466struct ps_tlv {1467__le16 tag;1468__le16 len;1469u8 ps_state; /* 0: device awake1470* 1: static power save1471* 2: dynamic power saving1472* 3: enter TWT power saving1473* 4: leave TWT power saving1474*/1475u8 pad[3];1476} __packed ps;1477} __packed ps_req = {1478.hdr = {1479.bss_idx = mconf->mt76.idx,1480},1481.ps = {1482.tag = cpu_to_le16(UNI_BSS_INFO_PS),1483.len = cpu_to_le16(sizeof(struct ps_tlv)),1484.ps_state = link_conf->vif->cfg.ps ? 2 : 0,1485},1486};14871488if (link_conf->vif->type != NL80211_IFTYPE_STATION)1489return -EOPNOTSUPP;14901491return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),1492&ps_req, sizeof(ps_req), true);1493}14941495int1496mt7925_mcu_uni_bss_bcnft(struct mt792x_dev *dev,1497struct ieee80211_bss_conf *link_conf, bool enable)1498{1499struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);1500struct {1501struct {1502u8 bss_idx;1503u8 pad[3];1504} __packed hdr;1505struct bcnft_tlv {1506__le16 tag;1507__le16 len;1508__le16 bcn_interval;1509u8 dtim_period;1510u8 bmc_delivered_ac;1511u8 bmc_triggered_ac;1512u8 pad[3];1513} __packed bcnft;1514} __packed bcnft_req = {1515.hdr = {1516.bss_idx = mconf->mt76.idx,1517},1518.bcnft = {1519.tag = cpu_to_le16(UNI_BSS_INFO_BCNFT),1520.len = cpu_to_le16(sizeof(struct bcnft_tlv)),1521.bcn_interval = cpu_to_le16(link_conf->beacon_int),1522.dtim_period = link_conf->dtim_period,1523},1524};15251526if (link_conf->vif->type != NL80211_IFTYPE_STATION)1527return 0;15281529return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),1530&bcnft_req, sizeof(bcnft_req), true);1531}15321533int1534mt7925_mcu_set_bss_pm(struct mt792x_dev *dev,1535struct ieee80211_bss_conf *link_conf,1536bool enable)1537{1538struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);1539struct {1540struct {1541u8 bss_idx;1542u8 pad[3];1543} __packed hdr;1544struct bcnft_tlv {1545__le16 tag;1546__le16 len;1547__le16 bcn_interval;1548u8 dtim_period;1549u8 bmc_delivered_ac;1550u8 bmc_triggered_ac;1551u8 pad[3];1552} __packed enable;1553} req = {1554.hdr = {1555.bss_idx = mconf->mt76.idx,1556},1557.enable = {1558.tag = cpu_to_le16(UNI_BSS_INFO_BCNFT),1559.len = cpu_to_le16(sizeof(struct bcnft_tlv)),1560.dtim_period = link_conf->dtim_period,1561.bcn_interval = cpu_to_le16(link_conf->beacon_int),1562},1563};1564struct {1565struct {1566u8 bss_idx;1567u8 pad[3];1568} __packed hdr;1569struct pm_disable {1570__le16 tag;1571__le16 len;1572} __packed disable;1573} req1 = {1574.hdr = {1575.bss_idx = mconf->mt76.idx,1576},1577.disable = {1578.tag = cpu_to_le16(UNI_BSS_INFO_PM_DISABLE),1579.len = cpu_to_le16(sizeof(struct pm_disable))1580},1581};1582int err;15831584err = mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),1585&req1, sizeof(req1), true);1586if (err < 0 || !enable)1587return err;15881589return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),1590&req, sizeof(req), true);1591}15921593static void1594mt7925_mcu_sta_he_tlv(struct sk_buff *skb, struct ieee80211_link_sta *link_sta)1595{1596if (!link_sta->he_cap.has_he)1597return;15981599mt76_connac_mcu_sta_he_tlv_v2(skb, link_sta->sta);1600}16011602static void1603mt7925_mcu_sta_he_6g_tlv(struct sk_buff *skb,1604struct ieee80211_link_sta *link_sta)1605{1606struct sta_rec_he_6g_capa *he_6g;1607struct tlv *tlv;16081609if (!link_sta->he_6ghz_capa.capa)1610return;16111612tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_HE_6G, sizeof(*he_6g));16131614he_6g = (struct sta_rec_he_6g_capa *)tlv;1615he_6g->capa = link_sta->he_6ghz_capa.capa;1616}16171618static void1619mt7925_mcu_sta_eht_tlv(struct sk_buff *skb, struct ieee80211_link_sta *link_sta)1620{1621struct ieee80211_eht_mcs_nss_supp *mcs_map;1622struct ieee80211_eht_cap_elem_fixed *elem;1623struct sta_rec_eht *eht;1624struct tlv *tlv;16251626if (!link_sta->eht_cap.has_eht)1627return;16281629mcs_map = &link_sta->eht_cap.eht_mcs_nss_supp;1630elem = &link_sta->eht_cap.eht_cap_elem;16311632tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_EHT, sizeof(*eht));16331634eht = (struct sta_rec_eht *)tlv;1635eht->tid_bitmap = 0xff;1636eht->mac_cap = cpu_to_le16(*(u16 *)elem->mac_cap_info);1637eht->phy_cap = cpu_to_le64(*(u64 *)elem->phy_cap_info);1638eht->phy_cap_ext = cpu_to_le64(elem->phy_cap_info[8]);16391640if (link_sta->bandwidth == IEEE80211_STA_RX_BW_20)1641memcpy(eht->mcs_map_bw20, &mcs_map->only_20mhz, sizeof(eht->mcs_map_bw20));1642memcpy(eht->mcs_map_bw80, &mcs_map->bw._80, sizeof(eht->mcs_map_bw80));1643memcpy(eht->mcs_map_bw160, &mcs_map->bw._160, sizeof(eht->mcs_map_bw160));1644}16451646static void1647mt7925_mcu_sta_ht_tlv(struct sk_buff *skb, struct ieee80211_link_sta *link_sta)1648{1649struct sta_rec_ht *ht;1650struct tlv *tlv;16511652if (!link_sta->ht_cap.ht_supported)1653return;16541655tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_HT, sizeof(*ht));16561657ht = (struct sta_rec_ht *)tlv;1658ht->ht_cap = cpu_to_le16(link_sta->ht_cap.cap);1659}16601661static void1662mt7925_mcu_sta_vht_tlv(struct sk_buff *skb, struct ieee80211_link_sta *link_sta)1663{1664struct sta_rec_vht *vht;1665struct tlv *tlv;16661667/* For 6G band, this tlv is necessary to let hw work normally */1668if (!link_sta->he_6ghz_capa.capa && !link_sta->vht_cap.vht_supported)1669return;16701671tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_VHT, sizeof(*vht));16721673vht = (struct sta_rec_vht *)tlv;1674vht->vht_cap = cpu_to_le32(link_sta->vht_cap.cap);1675vht->vht_rx_mcs_map = link_sta->vht_cap.vht_mcs.rx_mcs_map;1676vht->vht_tx_mcs_map = link_sta->vht_cap.vht_mcs.tx_mcs_map;1677}16781679static void1680mt7925_mcu_sta_amsdu_tlv(struct sk_buff *skb,1681struct ieee80211_vif *vif,1682struct ieee80211_link_sta *link_sta)1683{1684struct mt792x_sta *msta = (struct mt792x_sta *)link_sta->sta->drv_priv;1685struct mt792x_link_sta *mlink;1686struct sta_rec_amsdu *amsdu;1687struct tlv *tlv;16881689if (vif->type != NL80211_IFTYPE_STATION &&1690vif->type != NL80211_IFTYPE_AP)1691return;16921693if (!link_sta->agg.max_amsdu_len)1694return;16951696tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_HW_AMSDU, sizeof(*amsdu));1697amsdu = (struct sta_rec_amsdu *)tlv;1698amsdu->max_amsdu_num = 8;1699amsdu->amsdu_en = true;17001701mlink = mt792x_sta_to_link(msta, link_sta->link_id);1702mlink->wcid.amsdu = true;17031704switch (link_sta->agg.max_amsdu_len) {1705case IEEE80211_MAX_MPDU_LEN_VHT_11454:1706amsdu->max_mpdu_size =1707IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454;1708return;1709case IEEE80211_MAX_MPDU_LEN_HT_7935:1710case IEEE80211_MAX_MPDU_LEN_VHT_7991:1711amsdu->max_mpdu_size = IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_7991;1712return;1713default:1714amsdu->max_mpdu_size = IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_3895;1715return;1716}1717}17181719static void1720mt7925_mcu_sta_phy_tlv(struct sk_buff *skb,1721struct ieee80211_vif *vif,1722struct ieee80211_link_sta *link_sta)1723{1724struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1725struct ieee80211_bss_conf *link_conf;1726struct cfg80211_chan_def *chandef;1727struct mt792x_bss_conf *mconf;1728struct sta_rec_phy *phy;1729struct tlv *tlv;1730u8 af = 0, mm = 0;17311732link_conf = mt792x_vif_to_bss_conf(vif, link_sta->link_id);1733mconf = mt792x_vif_to_link(mvif, link_sta->link_id);1734chandef = mconf->mt76.ctx ? &mconf->mt76.ctx->def :1735&link_conf->chanreq.oper;17361737tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_PHY, sizeof(*phy));1738phy = (struct sta_rec_phy *)tlv;1739phy->phy_type = mt76_connac_get_phy_mode_v2(mvif->phy->mt76, vif,1740chandef->chan->band,1741link_sta);1742phy->basic_rate = cpu_to_le16((u16)link_conf->basic_rates);1743if (link_sta->ht_cap.ht_supported) {1744af = link_sta->ht_cap.ampdu_factor;1745mm = link_sta->ht_cap.ampdu_density;1746}17471748if (link_sta->vht_cap.vht_supported) {1749u8 vht_af = FIELD_GET(IEEE80211_VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MASK,1750link_sta->vht_cap.cap);17511752af = max_t(u8, af, vht_af);1753}17541755if (link_sta->he_6ghz_capa.capa) {1756af = le16_get_bits(link_sta->he_6ghz_capa.capa,1757IEEE80211_HE_6GHZ_CAP_MAX_AMPDU_LEN_EXP);1758mm = le16_get_bits(link_sta->he_6ghz_capa.capa,1759IEEE80211_HE_6GHZ_CAP_MIN_MPDU_START);1760}17611762phy->ampdu = FIELD_PREP(IEEE80211_HT_AMPDU_PARM_FACTOR, af) |1763FIELD_PREP(IEEE80211_HT_AMPDU_PARM_DENSITY, mm);1764phy->max_ampdu_len = af;1765}17661767static void1768mt7925_mcu_sta_state_v2_tlv(struct mt76_phy *mphy, struct sk_buff *skb,1769struct ieee80211_link_sta *link_sta,1770struct ieee80211_vif *vif,1771u8 rcpi, u8 sta_state)1772{1773struct sta_rec_state_v2 {1774__le16 tag;1775__le16 len;1776u8 state;1777u8 rsv[3];1778__le32 flags;1779u8 vht_opmode;1780u8 action;1781u8 rsv2[2];1782} __packed * state;1783struct tlv *tlv;17841785tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_STATE, sizeof(*state));1786state = (struct sta_rec_state_v2 *)tlv;1787state->state = sta_state;17881789if (link_sta->vht_cap.vht_supported) {1790state->vht_opmode = link_sta->bandwidth;1791state->vht_opmode |= link_sta->rx_nss <<1792IEEE80211_OPMODE_NOTIF_RX_NSS_SHIFT;1793}1794}17951796static void1797mt7925_mcu_sta_rate_ctrl_tlv(struct sk_buff *skb,1798struct ieee80211_vif *vif,1799struct ieee80211_link_sta *link_sta)1800{1801struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1802struct ieee80211_bss_conf *link_conf;1803struct cfg80211_chan_def *chandef;1804struct sta_rec_ra_info *ra_info;1805struct mt792x_bss_conf *mconf;1806enum nl80211_band band;1807struct tlv *tlv;1808u16 supp_rates;18091810link_conf = mt792x_vif_to_bss_conf(vif, link_sta->link_id);1811mconf = mt792x_vif_to_link(mvif, link_sta->link_id);1812chandef = mconf->mt76.ctx ? &mconf->mt76.ctx->def :1813&link_conf->chanreq.oper;1814band = chandef->chan->band;18151816tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_RA, sizeof(*ra_info));1817ra_info = (struct sta_rec_ra_info *)tlv;18181819supp_rates = link_sta->supp_rates[band];1820if (band == NL80211_BAND_2GHZ)1821supp_rates = FIELD_PREP(RA_LEGACY_OFDM, supp_rates >> 4) |1822FIELD_PREP(RA_LEGACY_CCK, supp_rates & 0xf);1823else1824supp_rates = FIELD_PREP(RA_LEGACY_OFDM, supp_rates);18251826ra_info->legacy = cpu_to_le16(supp_rates);18271828if (link_sta->ht_cap.ht_supported)1829memcpy(ra_info->rx_mcs_bitmask,1830link_sta->ht_cap.mcs.rx_mask,1831HT_MCS_MASK_NUM);1832}18331834static void1835mt7925_mcu_sta_eht_mld_tlv(struct sk_buff *skb,1836struct ieee80211_vif *vif, struct ieee80211_sta *sta)1837{1838struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1839struct wiphy *wiphy = mvif->phy->mt76->hw->wiphy;1840const struct wiphy_iftype_ext_capab *ext_capa;1841struct sta_rec_eht_mld *eht_mld;1842struct tlv *tlv;1843u16 eml_cap;18441845if (!ieee80211_vif_is_mld(vif))1846return;18471848tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_EHT_MLD, sizeof(*eht_mld));1849eht_mld = (struct sta_rec_eht_mld *)tlv;1850eht_mld->mld_type = 0xff;18511852ext_capa = cfg80211_get_iftype_ext_capa(wiphy,1853ieee80211_vif_type_p2p(vif));1854if (!ext_capa)1855return;18561857eml_cap = (vif->cfg.eml_cap & (IEEE80211_EML_CAP_EMLSR_SUPP |1858IEEE80211_EML_CAP_TRANSITION_TIMEOUT)) |1859(ext_capa->eml_capabilities & (IEEE80211_EML_CAP_EMLSR_PADDING_DELAY |1860IEEE80211_EML_CAP_EMLSR_TRANSITION_DELAY));18611862if (eml_cap & IEEE80211_EML_CAP_EMLSR_SUPP) {1863eht_mld->eml_cap[0] = u16_get_bits(eml_cap, GENMASK(7, 0));1864eht_mld->eml_cap[1] = u16_get_bits(eml_cap, GENMASK(15, 8));1865} else {1866eht_mld->str_cap[0] = BIT(1);1867}1868}18691870static void1871mt7925_mcu_sta_mld_tlv(struct sk_buff *skb,1872struct ieee80211_vif *vif, struct ieee80211_sta *sta)1873{1874struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1875struct mt792x_sta *msta = (struct mt792x_sta *)sta->drv_priv;1876unsigned long valid = mvif->valid_links;1877struct mt792x_bss_conf *mconf;1878struct mt792x_link_sta *mlink;1879struct sta_rec_mld *mld;1880struct tlv *tlv;1881int i, cnt = 0;18821883tlv = mt76_connac_mcu_add_tlv(skb, STA_REC_MLD, sizeof(*mld));1884mld = (struct sta_rec_mld *)tlv;1885memcpy(mld->mac_addr, sta->addr, ETH_ALEN);1886mld->primary_id = cpu_to_le16(msta->deflink.wcid.idx);1887mld->wlan_id = cpu_to_le16(msta->deflink.wcid.idx);1888mld->link_num = min_t(u8, hweight16(mvif->valid_links), 2);18891890for_each_set_bit(i, &valid, IEEE80211_MLD_MAX_NUM_LINKS) {1891if (cnt == mld->link_num)1892break;18931894mconf = mt792x_vif_to_link(mvif, i);1895mlink = mt792x_sta_to_link(msta, i);1896mld->link[cnt].wlan_id = cpu_to_le16(mlink->wcid.idx);1897mld->link[cnt++].bss_idx = mconf->mt76.idx;18981899if (mlink != &msta->deflink)1900mld->secondary_id = cpu_to_le16(mlink->wcid.idx);1901}1902}19031904static void1905mt7925_mcu_sta_remove_tlv(struct sk_buff *skb)1906{1907struct sta_rec_remove *rem;1908struct tlv *tlv;19091910tlv = mt76_connac_mcu_add_tlv(skb, 0x25, sizeof(*rem));1911rem = (struct sta_rec_remove *)tlv;1912rem->action = 0;1913}19141915static int1916mt7925_mcu_sta_cmd(struct mt76_phy *phy,1917struct mt76_sta_cmd_info *info)1918{1919struct mt792x_vif *mvif = (struct mt792x_vif *)info->vif->drv_priv;1920struct mt76_dev *dev = phy->dev;1921struct mt792x_bss_conf *mconf;1922struct sk_buff *skb;1923int conn_state;19241925mconf = mt792x_vif_to_link(mvif, info->wcid->link_id);19261927skb = __mt76_connac_mcu_alloc_sta_req(dev, &mconf->mt76, info->wcid,1928MT7925_STA_UPDATE_MAX_SIZE);1929if (IS_ERR(skb))1930return PTR_ERR(skb);19311932conn_state = info->enable ? CONN_STATE_PORT_SECURE :1933CONN_STATE_DISCONNECT;19341935if (info->enable && info->link_sta) {1936mt76_connac_mcu_sta_basic_tlv(dev, skb, info->link_conf,1937info->link_sta,1938conn_state, info->newly);1939mt7925_mcu_sta_phy_tlv(skb, info->vif, info->link_sta);1940mt7925_mcu_sta_ht_tlv(skb, info->link_sta);1941mt7925_mcu_sta_vht_tlv(skb, info->link_sta);1942mt76_connac_mcu_sta_uapsd(skb, info->vif, info->link_sta->sta);1943mt7925_mcu_sta_amsdu_tlv(skb, info->vif, info->link_sta);1944mt7925_mcu_sta_he_tlv(skb, info->link_sta);1945mt7925_mcu_sta_he_6g_tlv(skb, info->link_sta);1946mt7925_mcu_sta_eht_tlv(skb, info->link_sta);1947mt7925_mcu_sta_rate_ctrl_tlv(skb, info->vif,1948info->link_sta);1949mt7925_mcu_sta_state_v2_tlv(phy, skb, info->link_sta,1950info->vif, info->rcpi,1951info->state);19521953if (info->state != MT76_STA_INFO_STATE_NONE) {1954mt7925_mcu_sta_mld_tlv(skb, info->vif, info->link_sta->sta);1955mt7925_mcu_sta_eht_mld_tlv(skb, info->vif, info->link_sta->sta);1956}1957}19581959if (!info->enable) {1960mt7925_mcu_sta_remove_tlv(skb);1961mt76_connac_mcu_add_tlv(skb, STA_REC_MLD_OFF,1962sizeof(struct tlv));1963} else {1964mt7925_mcu_sta_hdr_trans_tlv(skb, info->vif, info->link_sta);1965}19661967return mt76_mcu_skb_send_msg(dev, skb, info->cmd, true);1968}19691970int mt7925_mcu_sta_update(struct mt792x_dev *dev,1971struct ieee80211_link_sta *link_sta,1972struct ieee80211_vif *vif, bool enable,1973enum mt76_sta_info_state state)1974{1975struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;1976int rssi = -ewma_rssi_read(&mvif->bss_conf.rssi);1977struct mt76_sta_cmd_info info = {1978.link_sta = link_sta,1979.vif = vif,1980.link_conf = &vif->bss_conf,1981.enable = enable,1982.cmd = MCU_UNI_CMD(STA_REC_UPDATE),1983.state = state,1984.offload_fw = true,1985.rcpi = to_rcpi(rssi),1986};1987struct mt792x_sta *msta;1988struct mt792x_link_sta *mlink;19891990if (link_sta) {1991msta = (struct mt792x_sta *)link_sta->sta->drv_priv;1992mlink = mt792x_sta_to_link(msta, link_sta->link_id);1993}1994info.wcid = link_sta ? &mlink->wcid : &mvif->sta.deflink.wcid;1995info.newly = state != MT76_STA_INFO_STATE_ASSOC;19961997return mt7925_mcu_sta_cmd(&dev->mphy, &info);1998}19992000int mt7925_mcu_set_beacon_filter(struct mt792x_dev *dev,2001struct ieee80211_vif *vif,2002bool enable)2003{2004#define MT7925_FIF_BIT_CLR BIT(1)2005#define MT7925_FIF_BIT_SET BIT(0)2006int err = 0;20072008if (enable) {2009err = mt7925_mcu_uni_bss_bcnft(dev, &vif->bss_conf, true);2010if (err < 0)2011return err;20122013return mt7925_mcu_set_rxfilter(dev, 0,2014MT7925_FIF_BIT_SET,2015MT_WF_RFCR_DROP_OTHER_BEACON);2016}20172018err = mt7925_mcu_set_bss_pm(dev, &vif->bss_conf, false);2019if (err < 0)2020return err;20212022return mt7925_mcu_set_rxfilter(dev, 0,2023MT7925_FIF_BIT_CLR,2024MT_WF_RFCR_DROP_OTHER_BEACON);2025}20262027int mt7925_get_txpwr_info(struct mt792x_dev *dev, u8 band_idx, struct mt7925_txpwr *txpwr)2028{2029#define TX_POWER_SHOW_INFO 0x72030#define TXPOWER_ALL_RATE_POWER_INFO 0x22031struct mt7925_txpwr_event *event;2032struct mt7925_txpwr_req req = {2033.tag = cpu_to_le16(TX_POWER_SHOW_INFO),2034.len = cpu_to_le16(sizeof(req) - 4),2035.catg = TXPOWER_ALL_RATE_POWER_INFO,2036.band_idx = band_idx,2037};2038struct sk_buff *skb;2039int ret;20402041ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_CMD(TXPOWER),2042&req, sizeof(req), true, &skb);2043if (ret)2044return ret;20452046event = (struct mt7925_txpwr_event *)skb->data;2047memcpy(txpwr, &event->txpwr, sizeof(event->txpwr));20482049dev_kfree_skb(skb);20502051return 0;2052}20532054int mt7925_mcu_set_sniffer(struct mt792x_dev *dev, struct ieee80211_vif *vif,2055bool enable)2056{2057struct {2058struct {2059u8 band_idx;2060u8 pad[3];2061} __packed hdr;2062struct sniffer_enable_tlv {2063__le16 tag;2064__le16 len;2065u8 enable;2066u8 pad[3];2067} __packed enable;2068} __packed req = {2069.hdr = {2070.band_idx = 0,2071},2072.enable = {2073.tag = cpu_to_le16(UNI_SNIFFER_ENABLE),2074.len = cpu_to_le16(sizeof(struct sniffer_enable_tlv)),2075.enable = enable,2076},2077};20782079return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(SNIFFER), &req, sizeof(req),2080true);2081}20822083int mt7925_mcu_config_sniffer(struct mt792x_vif *vif,2084struct ieee80211_chanctx_conf *ctx)2085{2086struct mt76_phy *mphy = vif->phy->mt76;2087struct cfg80211_chan_def *chandef = ctx ? &ctx->def : &mphy->chandef;2088int freq1 = chandef->center_freq1, freq2 = chandef->center_freq2;20892090static const u8 ch_band[] = {2091[NL80211_BAND_2GHZ] = 1,2092[NL80211_BAND_5GHZ] = 2,2093[NL80211_BAND_6GHZ] = 3,2094};2095static const u8 ch_width[] = {2096[NL80211_CHAN_WIDTH_20_NOHT] = 0,2097[NL80211_CHAN_WIDTH_20] = 0,2098[NL80211_CHAN_WIDTH_40] = 0,2099[NL80211_CHAN_WIDTH_80] = 1,2100[NL80211_CHAN_WIDTH_160] = 2,2101[NL80211_CHAN_WIDTH_80P80] = 3,2102[NL80211_CHAN_WIDTH_5] = 4,2103[NL80211_CHAN_WIDTH_10] = 5,2104[NL80211_CHAN_WIDTH_320] = 6,2105};21062107struct {2108struct {2109u8 band_idx;2110u8 pad[3];2111} __packed hdr;2112struct config_tlv {2113__le16 tag;2114__le16 len;2115u16 aid;2116u8 ch_band;2117u8 bw;2118u8 control_ch;2119u8 sco;2120u8 center_ch;2121u8 center_ch2;2122u8 drop_err;2123u8 pad[3];2124} __packed tlv;2125} __packed req = {2126.hdr = {2127.band_idx = 0,2128},2129.tlv = {2130.tag = cpu_to_le16(UNI_SNIFFER_CONFIG),2131.len = cpu_to_le16(sizeof(req.tlv)),2132.control_ch = chandef->chan->hw_value,2133.center_ch = ieee80211_frequency_to_channel(freq1),2134.drop_err = 1,2135},2136};21372138if (chandef->chan->band < ARRAY_SIZE(ch_band))2139req.tlv.ch_band = ch_band[chandef->chan->band];2140if (chandef->width < ARRAY_SIZE(ch_width))2141req.tlv.bw = ch_width[chandef->width];21422143if (freq2)2144req.tlv.center_ch2 = ieee80211_frequency_to_channel(freq2);21452146if (req.tlv.control_ch < req.tlv.center_ch)2147req.tlv.sco = 1; /* SCA */2148else if (req.tlv.control_ch > req.tlv.center_ch)2149req.tlv.sco = 3; /* SCB */21502151return mt76_mcu_send_msg(mphy->dev, MCU_UNI_CMD(SNIFFER),2152&req, sizeof(req), true);2153}21542155int2156mt7925_mcu_uni_add_beacon_offload(struct mt792x_dev *dev,2157struct ieee80211_hw *hw,2158struct ieee80211_vif *vif,2159bool enable)2160{2161struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;2162struct ieee80211_mutable_offsets offs;2163struct {2164struct req_hdr {2165u8 bss_idx;2166u8 pad[3];2167} __packed hdr;2168struct bcn_content_tlv {2169__le16 tag;2170__le16 len;2171__le16 tim_ie_pos;2172__le16 csa_ie_pos;2173__le16 bcc_ie_pos;2174/* 0: disable beacon offload2175* 1: enable beacon offload2176* 2: update probe respond offload2177*/2178u8 enable;2179/* 0: legacy format (TXD + payload)2180* 1: only cap field IE2181*/2182u8 type;2183__le16 pkt_len;2184u8 pkt[512];2185} __packed beacon_tlv;2186} req = {2187.hdr = {2188.bss_idx = mvif->bss_conf.mt76.idx,2189},2190.beacon_tlv = {2191.tag = cpu_to_le16(UNI_BSS_INFO_BCN_CONTENT),2192.len = cpu_to_le16(sizeof(struct bcn_content_tlv)),2193.enable = enable,2194.type = 1,2195},2196};2197struct sk_buff *skb;2198u8 cap_offs;21992200/* support enable/update process only2201* disable flow would be handled in bss stop handler automatically2202*/2203if (!enable)2204return -EOPNOTSUPP;22052206skb = ieee80211_beacon_get_template(mt76_hw(dev), vif, &offs, 0);2207if (!skb)2208return -EINVAL;22092210cap_offs = offsetof(struct ieee80211_mgmt, u.beacon.capab_info);2211if (!skb_pull(skb, cap_offs)) {2212dev_err(dev->mt76.dev, "beacon format err\n");2213dev_kfree_skb(skb);2214return -EINVAL;2215}22162217if (skb->len > 512) {2218dev_err(dev->mt76.dev, "beacon size limit exceed\n");2219dev_kfree_skb(skb);2220return -EINVAL;2221}22222223memcpy(req.beacon_tlv.pkt, skb->data, skb->len);2224req.beacon_tlv.pkt_len = cpu_to_le16(skb->len);2225offs.tim_offset -= cap_offs;2226req.beacon_tlv.tim_ie_pos = cpu_to_le16(offs.tim_offset);22272228if (offs.cntdwn_counter_offs[0]) {2229u16 csa_offs;22302231csa_offs = offs.cntdwn_counter_offs[0] - cap_offs - 4;2232req.beacon_tlv.csa_ie_pos = cpu_to_le16(csa_offs);2233}2234dev_kfree_skb(skb);22352236return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),2237&req, sizeof(req), true);2238}22392240static2241void mt7925_mcu_bss_rlm_tlv(struct sk_buff *skb, struct mt76_phy *phy,2242struct ieee80211_bss_conf *link_conf,2243struct ieee80211_chanctx_conf *ctx)2244{2245struct cfg80211_chan_def *chandef = ctx ? &ctx->def :2246&link_conf->chanreq.oper;2247int freq1 = chandef->center_freq1, freq2 = chandef->center_freq2;2248enum nl80211_band band = chandef->chan->band;2249struct bss_rlm_tlv *req;2250struct tlv *tlv;22512252tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_RLM, sizeof(*req));2253req = (struct bss_rlm_tlv *)tlv;2254req->control_channel = chandef->chan->hw_value;2255req->center_chan = ieee80211_frequency_to_channel(freq1);2256req->center_chan2 = 0;2257req->tx_streams = hweight8(phy->antenna_mask);2258req->ht_op_info = 4; /* set HT 40M allowed */2259req->rx_streams = hweight8(phy->antenna_mask);2260req->center_chan2 = 0;2261req->sco = 0;2262req->band = 1;22632264switch (band) {2265case NL80211_BAND_2GHZ:2266req->band = 1;2267break;2268case NL80211_BAND_5GHZ:2269req->band = 2;2270break;2271case NL80211_BAND_6GHZ:2272req->band = 3;2273break;2274default:2275break;2276}22772278switch (chandef->width) {2279case NL80211_CHAN_WIDTH_40:2280req->bw = CMD_CBW_40MHZ;2281break;2282case NL80211_CHAN_WIDTH_80:2283req->bw = CMD_CBW_80MHZ;2284break;2285case NL80211_CHAN_WIDTH_80P80:2286req->bw = CMD_CBW_8080MHZ;2287req->center_chan2 = ieee80211_frequency_to_channel(freq2);2288break;2289case NL80211_CHAN_WIDTH_160:2290req->bw = CMD_CBW_160MHZ;2291break;2292case NL80211_CHAN_WIDTH_5:2293req->bw = CMD_CBW_5MHZ;2294break;2295case NL80211_CHAN_WIDTH_10:2296req->bw = CMD_CBW_10MHZ;2297break;2298case NL80211_CHAN_WIDTH_20_NOHT:2299case NL80211_CHAN_WIDTH_20:2300default:2301req->bw = CMD_CBW_20MHZ;2302req->ht_op_info = 0;2303break;2304}23052306if (req->control_channel < req->center_chan)2307req->sco = 1; /* SCA */2308else if (req->control_channel > req->center_chan)2309req->sco = 3; /* SCB */2310}23112312static struct sk_buff *2313__mt7925_mcu_alloc_bss_req(struct mt76_dev *dev, struct mt76_vif_link *mvif, int len)2314{2315struct bss_req_hdr hdr = {2316.bss_idx = mvif->idx,2317};2318struct sk_buff *skb;23192320skb = mt76_mcu_msg_alloc(dev, NULL, len);2321if (!skb)2322return ERR_PTR(-ENOMEM);23232324skb_put_data(skb, &hdr, sizeof(hdr));23252326return skb;2327}23282329static2330void mt7925_mcu_bss_eht_tlv(struct sk_buff *skb, struct mt76_phy *phy,2331struct ieee80211_bss_conf *link_conf,2332struct ieee80211_chanctx_conf *ctx)2333{2334struct cfg80211_chan_def *chandef = ctx ? &ctx->def :2335&link_conf->chanreq.oper;23362337struct bss_eht_tlv *req;2338struct tlv *tlv;23392340tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_EHT, sizeof(*req));2341req = (struct bss_eht_tlv *)tlv;2342req->is_eth_dscb_present = chandef->punctured ? 1 : 0;2343req->eht_dis_sub_chan_bitmap = cpu_to_le16(chandef->punctured);2344}23452346int mt7925_mcu_set_eht_pp(struct mt76_phy *phy, struct mt76_vif_link *mvif,2347struct ieee80211_bss_conf *link_conf,2348struct ieee80211_chanctx_conf *ctx)2349{2350struct sk_buff *skb;23512352skb = __mt7925_mcu_alloc_bss_req(phy->dev, mvif,2353MT7925_BSS_UPDATE_MAX_SIZE);2354if (IS_ERR(skb))2355return PTR_ERR(skb);23562357mt7925_mcu_bss_eht_tlv(skb, phy, link_conf, ctx);23582359return mt76_mcu_skb_send_msg(phy->dev, skb,2360MCU_UNI_CMD(BSS_INFO_UPDATE), true);2361}23622363int mt7925_mcu_set_chctx(struct mt76_phy *phy, struct mt76_vif_link *mvif,2364struct ieee80211_bss_conf *link_conf,2365struct ieee80211_chanctx_conf *ctx)2366{2367struct sk_buff *skb;23682369skb = __mt7925_mcu_alloc_bss_req(phy->dev, mvif,2370MT7925_BSS_UPDATE_MAX_SIZE);2371if (IS_ERR(skb))2372return PTR_ERR(skb);23732374mt7925_mcu_bss_rlm_tlv(skb, phy, link_conf, ctx);23752376return mt76_mcu_skb_send_msg(phy->dev, skb,2377MCU_UNI_CMD(BSS_INFO_UPDATE), true);2378}23792380static u82381mt7925_get_phy_mode_ext(struct mt76_phy *phy, struct ieee80211_vif *vif,2382enum nl80211_band band,2383struct ieee80211_link_sta *link_sta)2384{2385struct ieee80211_he_6ghz_capa *he_6ghz_capa;2386const struct ieee80211_sta_eht_cap *eht_cap;2387__le16 capa = 0;2388u8 mode = 0;23892390if (link_sta) {2391he_6ghz_capa = &link_sta->he_6ghz_capa;2392eht_cap = &link_sta->eht_cap;2393} else {2394struct ieee80211_supported_band *sband;23952396sband = phy->hw->wiphy->bands[band];2397capa = ieee80211_get_he_6ghz_capa(sband, vif->type);2398he_6ghz_capa = (struct ieee80211_he_6ghz_capa *)&capa;23992400eht_cap = ieee80211_get_eht_iftype_cap(sband, vif->type);2401}24022403switch (band) {2404case NL80211_BAND_2GHZ:2405if (eht_cap && eht_cap->has_eht)2406mode |= PHY_MODE_BE_24G;2407break;2408case NL80211_BAND_5GHZ:2409if (eht_cap && eht_cap->has_eht)2410mode |= PHY_MODE_BE_5G;2411break;2412case NL80211_BAND_6GHZ:2413if (he_6ghz_capa && he_6ghz_capa->capa)2414mode |= PHY_MODE_AX_6G;24152416if (eht_cap && eht_cap->has_eht)2417mode |= PHY_MODE_BE_6G;2418break;2419default:2420break;2421}24222423return mode;2424}24252426static void2427mt7925_mcu_bss_basic_tlv(struct sk_buff *skb,2428struct ieee80211_bss_conf *link_conf,2429struct ieee80211_link_sta *link_sta,2430struct ieee80211_chanctx_conf *ctx,2431struct mt76_phy *phy, u16 wlan_idx,2432bool enable)2433{2434struct ieee80211_vif *vif = link_conf->vif;2435struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2436struct cfg80211_chan_def *chandef = ctx ? &ctx->def :2437&link_conf->chanreq.oper;2438enum nl80211_band band = chandef->chan->band;2439struct mt76_connac_bss_basic_tlv *basic_req;2440struct mt792x_link_sta *mlink;2441struct tlv *tlv;2442int conn_type;2443u8 idx;24442445tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_BASIC, sizeof(*basic_req));2446basic_req = (struct mt76_connac_bss_basic_tlv *)tlv;24472448idx = mconf->mt76.omac_idx > EXT_BSSID_START ? HW_BSSID_0 :2449mconf->mt76.omac_idx;2450basic_req->hw_bss_idx = idx;24512452basic_req->phymode_ext = mt7925_get_phy_mode_ext(phy, vif, band,2453link_sta);24542455if (band == NL80211_BAND_2GHZ)2456basic_req->nonht_basic_phy = cpu_to_le16(PHY_TYPE_ERP_INDEX);2457else2458basic_req->nonht_basic_phy = cpu_to_le16(PHY_TYPE_OFDM_INDEX);24592460memcpy(basic_req->bssid, link_conf->bssid, ETH_ALEN);2461basic_req->phymode = mt76_connac_get_phy_mode(phy, vif, band, link_sta);2462basic_req->bcn_interval = cpu_to_le16(link_conf->beacon_int);2463basic_req->dtim_period = link_conf->dtim_period;2464basic_req->bmc_tx_wlan_idx = cpu_to_le16(wlan_idx);2465basic_req->link_idx = mconf->mt76.idx;24662467if (link_sta) {2468struct mt792x_sta *msta;24692470msta = (struct mt792x_sta *)link_sta->sta->drv_priv;2471mlink = mt792x_sta_to_link(msta, link_sta->link_id);24722473} else {2474mlink = &mconf->vif->sta.deflink;2475}24762477basic_req->sta_idx = cpu_to_le16(mlink->wcid.idx);2478basic_req->omac_idx = mconf->mt76.omac_idx;2479basic_req->band_idx = mconf->mt76.band_idx;2480basic_req->wmm_idx = mconf->mt76.wmm_idx;2481basic_req->conn_state = !enable;24822483switch (vif->type) {2484case NL80211_IFTYPE_MESH_POINT:2485case NL80211_IFTYPE_AP:2486if (vif->p2p)2487conn_type = CONNECTION_P2P_GO;2488else2489conn_type = CONNECTION_INFRA_AP;2490basic_req->conn_type = cpu_to_le32(conn_type);2491basic_req->active = enable;2492break;2493case NL80211_IFTYPE_STATION:2494if (vif->p2p)2495conn_type = CONNECTION_P2P_GC;2496else2497conn_type = CONNECTION_INFRA_STA;2498basic_req->conn_type = cpu_to_le32(conn_type);2499basic_req->active = true;2500break;2501case NL80211_IFTYPE_ADHOC:2502basic_req->conn_type = cpu_to_le32(CONNECTION_IBSS_ADHOC);2503basic_req->active = true;2504break;2505default:2506WARN_ON(1);2507break;2508}2509}25102511static void2512mt7925_mcu_bss_sec_tlv(struct sk_buff *skb,2513struct ieee80211_bss_conf *link_conf)2514{2515struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2516struct mt76_vif_link *mvif = &mconf->mt76;2517struct bss_sec_tlv {2518__le16 tag;2519__le16 len;2520u8 mode;2521u8 status;2522u8 cipher;2523u8 __rsv;2524} __packed * sec;2525struct tlv *tlv;25262527tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_SEC, sizeof(*sec));2528sec = (struct bss_sec_tlv *)tlv;25292530switch (mvif->cipher) {2531case CONNAC3_CIPHER_GCMP_256:2532case CONNAC3_CIPHER_GCMP:2533sec->mode = MODE_WPA3_SAE;2534sec->status = 8;2535break;2536case CONNAC3_CIPHER_AES_CCMP:2537sec->mode = MODE_WPA2_PSK;2538sec->status = 6;2539break;2540case CONNAC3_CIPHER_TKIP:2541sec->mode = MODE_WPA2_PSK;2542sec->status = 4;2543break;2544case CONNAC3_CIPHER_WEP104:2545case CONNAC3_CIPHER_WEP40:2546sec->mode = MODE_SHARED;2547sec->status = 0;2548break;2549default:2550sec->mode = MODE_OPEN;2551sec->status = 1;2552break;2553}25542555sec->cipher = mvif->cipher;2556}25572558static void2559mt7925_mcu_bss_bmc_tlv(struct sk_buff *skb, struct mt792x_phy *phy,2560struct ieee80211_chanctx_conf *ctx,2561struct ieee80211_bss_conf *link_conf)2562{2563struct cfg80211_chan_def *chandef = ctx ? &ctx->def :2564&link_conf->chanreq.oper;2565struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2566enum nl80211_band band = chandef->chan->band;2567struct mt76_vif_link *mvif = &mconf->mt76;2568struct bss_rate_tlv *bmc;2569struct tlv *tlv;2570u8 idx = mvif->mcast_rates_idx ?2571mvif->mcast_rates_idx : mvif->basic_rates_idx;25722573tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_RATE, sizeof(*bmc));25742575bmc = (struct bss_rate_tlv *)tlv;25762577if (band == NL80211_BAND_2GHZ)2578bmc->basic_rate = cpu_to_le16(HR_DSSS_ERP_BASIC_RATE);2579else2580bmc->basic_rate = cpu_to_le16(OFDM_BASIC_RATE);25812582bmc->short_preamble = (band == NL80211_BAND_2GHZ);2583bmc->bc_fixed_rate = idx;2584bmc->mc_fixed_rate = idx;2585}25862587static void2588mt7925_mcu_bss_mld_tlv(struct sk_buff *skb,2589struct ieee80211_bss_conf *link_conf)2590{2591struct ieee80211_vif *vif = link_conf->vif;2592struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2593struct mt792x_vif *mvif = (struct mt792x_vif *)link_conf->vif->drv_priv;2594struct mt792x_phy *phy = mvif->phy;2595struct bss_mld_tlv *mld;2596struct tlv *tlv;2597bool is_mld;25982599is_mld = ieee80211_vif_is_mld(link_conf->vif) ||2600(hweight16(mvif->valid_links) > 1);26012602tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_MLD, sizeof(*mld));2603mld = (struct bss_mld_tlv *)tlv;26042605mld->link_id = is_mld ? link_conf->link_id : 0xff;2606/* apply the index of the primary link */2607mld->group_mld_id = is_mld ? mvif->bss_conf.mt76.idx : 0xff;2608mld->own_mld_id = mconf->mt76.idx + 32;2609mld->remap_idx = 0xff;26102611if (phy->chip_cap & MT792x_CHIP_CAP_MLO_EML_EN) {2612mld->eml_enable = !!(link_conf->vif->cfg.eml_cap &2613IEEE80211_EML_CAP_EMLSR_SUPP);2614} else {2615mld->eml_enable = 0;2616}26172618memcpy(mld->mac_addr, vif->addr, ETH_ALEN);2619}26202621static void2622mt7925_mcu_bss_qos_tlv(struct sk_buff *skb, struct ieee80211_bss_conf *link_conf)2623{2624struct mt76_connac_bss_qos_tlv *qos;2625struct tlv *tlv;26262627tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_QBSS, sizeof(*qos));2628qos = (struct mt76_connac_bss_qos_tlv *)tlv;2629qos->qos = link_conf->qos;2630}26312632static void2633mt7925_mcu_bss_he_tlv(struct sk_buff *skb, struct ieee80211_bss_conf *link_conf,2634struct mt792x_phy *phy)2635{2636#define DEFAULT_HE_PE_DURATION 42637#define DEFAULT_HE_DURATION_RTS_THRES 10232638const struct ieee80211_sta_he_cap *cap;2639struct bss_info_uni_he *he;2640struct tlv *tlv;26412642cap = mt76_connac_get_he_phy_cap(phy->mt76, link_conf->vif);26432644tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_HE_BASIC, sizeof(*he));26452646he = (struct bss_info_uni_he *)tlv;2647he->he_pe_duration = link_conf->htc_trig_based_pkt_ext;2648if (!he->he_pe_duration)2649he->he_pe_duration = DEFAULT_HE_PE_DURATION;26502651he->he_rts_thres = cpu_to_le16(link_conf->frame_time_rts_th);2652if (!he->he_rts_thres)2653he->he_rts_thres = cpu_to_le16(DEFAULT_HE_DURATION_RTS_THRES);26542655he->max_nss_mcs[CMD_HE_MCS_BW80] = cap->he_mcs_nss_supp.tx_mcs_80;2656he->max_nss_mcs[CMD_HE_MCS_BW160] = cap->he_mcs_nss_supp.tx_mcs_160;2657he->max_nss_mcs[CMD_HE_MCS_BW8080] = cap->he_mcs_nss_supp.tx_mcs_80p80;2658}26592660static void2661mt7925_mcu_bss_color_tlv(struct sk_buff *skb, struct ieee80211_bss_conf *link_conf,2662bool enable)2663{2664struct bss_info_uni_bss_color *color;2665struct tlv *tlv;26662667tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_BSS_COLOR, sizeof(*color));2668color = (struct bss_info_uni_bss_color *)tlv;26692670color->enable = enable ?2671link_conf->he_bss_color.enabled : 0;2672color->bss_color = enable ?2673link_conf->he_bss_color.color : 0;2674}26752676static void2677mt7925_mcu_bss_ifs_tlv(struct sk_buff *skb,2678struct ieee80211_bss_conf *link_conf)2679{2680struct mt792x_vif *mvif = (struct mt792x_vif *)link_conf->vif->drv_priv;2681struct mt792x_phy *phy = mvif->phy;2682struct bss_ifs_time_tlv *ifs_time;2683struct tlv *tlv;26842685tlv = mt76_connac_mcu_add_tlv(skb, UNI_BSS_INFO_IFS_TIME, sizeof(*ifs_time));2686ifs_time = (struct bss_ifs_time_tlv *)tlv;2687ifs_time->slot_valid = true;2688ifs_time->slot_time = cpu_to_le16(phy->slottime);2689}26902691int mt7925_mcu_set_timing(struct mt792x_phy *phy,2692struct ieee80211_bss_conf *link_conf)2693{2694struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2695struct mt792x_dev *dev = phy->dev;2696struct sk_buff *skb;26972698skb = __mt7925_mcu_alloc_bss_req(&dev->mt76, &mconf->mt76,2699MT7925_BSS_UPDATE_MAX_SIZE);2700if (IS_ERR(skb))2701return PTR_ERR(skb);27022703mt7925_mcu_bss_ifs_tlv(skb, link_conf);27042705return mt76_mcu_skb_send_msg(&dev->mt76, skb,2706MCU_UNI_CMD(BSS_INFO_UPDATE), true);2707}27082709void mt7925_mcu_del_dev(struct mt76_dev *mdev,2710struct ieee80211_vif *vif)2711{2712struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;2713struct {2714struct {2715u8 omac_idx;2716u8 band_idx;2717__le16 pad;2718} __packed hdr;2719struct req_tlv {2720__le16 tag;2721__le16 len;2722u8 active;2723u8 link_idx; /* hw link idx */2724u8 omac_addr[ETH_ALEN];2725} __packed tlv;2726} dev_req = {2727.tlv = {2728.tag = cpu_to_le16(DEV_INFO_ACTIVE),2729.len = cpu_to_le16(sizeof(struct req_tlv)),2730.active = true,2731},2732};2733struct {2734struct {2735u8 bss_idx;2736u8 pad[3];2737} __packed hdr;2738struct mt76_connac_bss_basic_tlv basic;2739} basic_req = {2740.basic = {2741.tag = cpu_to_le16(UNI_BSS_INFO_BASIC),2742.len = cpu_to_le16(sizeof(struct mt76_connac_bss_basic_tlv)),2743.active = true,2744.conn_state = 1,2745},2746};27472748dev_req.hdr.omac_idx = mvif->omac_idx;2749dev_req.hdr.band_idx = mvif->band_idx;27502751basic_req.hdr.bss_idx = mvif->idx;2752basic_req.basic.omac_idx = mvif->omac_idx;2753basic_req.basic.band_idx = mvif->band_idx;2754basic_req.basic.link_idx = mvif->link_idx;27552756mt76_mcu_send_msg(mdev, MCU_UNI_CMD(BSS_INFO_UPDATE),2757&basic_req, sizeof(basic_req), true);27582759/* recovery omac address for the legacy interface */2760memcpy(dev_req.tlv.omac_addr, vif->addr, ETH_ALEN);2761mt76_mcu_send_msg(mdev, MCU_UNI_CMD(DEV_INFO_UPDATE),2762&dev_req, sizeof(dev_req), true);2763}27642765int mt7925_mcu_add_bss_info(struct mt792x_phy *phy,2766struct ieee80211_chanctx_conf *ctx,2767struct ieee80211_bss_conf *link_conf,2768struct ieee80211_link_sta *link_sta,2769int enable)2770{2771struct mt792x_vif *mvif = (struct mt792x_vif *)link_conf->vif->drv_priv;2772struct mt792x_bss_conf *mconf = mt792x_link_conf_to_mconf(link_conf);2773struct mt792x_dev *dev = phy->dev;2774struct mt792x_link_sta *mlink_bc;2775struct sk_buff *skb;27762777skb = __mt7925_mcu_alloc_bss_req(&dev->mt76, &mconf->mt76,2778MT7925_BSS_UPDATE_MAX_SIZE);2779if (IS_ERR(skb))2780return PTR_ERR(skb);27812782mlink_bc = mt792x_sta_to_link(&mvif->sta, mconf->link_id);27832784/* bss_basic must be first */2785mt7925_mcu_bss_basic_tlv(skb, link_conf, link_sta, ctx, phy->mt76,2786mlink_bc->wcid.idx, enable);2787mt7925_mcu_bss_sec_tlv(skb, link_conf);2788mt7925_mcu_bss_bmc_tlv(skb, phy, ctx, link_conf);2789mt7925_mcu_bss_qos_tlv(skb, link_conf);2790mt7925_mcu_bss_mld_tlv(skb, link_conf);2791mt7925_mcu_bss_ifs_tlv(skb, link_conf);27922793if (link_conf->he_support) {2794mt7925_mcu_bss_he_tlv(skb, link_conf, phy);2795mt7925_mcu_bss_color_tlv(skb, link_conf, enable);2796}27972798if (enable)2799mt7925_mcu_bss_rlm_tlv(skb, phy->mt76, link_conf, ctx);28002801return mt76_mcu_skb_send_msg(&dev->mt76, skb,2802MCU_UNI_CMD(BSS_INFO_UPDATE), true);2803}28042805int mt7925_mcu_set_dbdc(struct mt76_phy *phy, bool enable)2806{2807struct mt76_dev *mdev = phy->dev;28082809struct mbmc_conf_tlv *conf;2810struct mbmc_set_req *hdr;2811struct sk_buff *skb;2812struct tlv *tlv;2813int max_len, err;28142815max_len = sizeof(*hdr) + sizeof(*conf);2816skb = mt76_mcu_msg_alloc(mdev, NULL, max_len);2817if (!skb)2818return -ENOMEM;28192820hdr = (struct mbmc_set_req *)skb_put(skb, sizeof(*hdr));28212822tlv = mt76_connac_mcu_add_tlv(skb, UNI_MBMC_SETTING, sizeof(*conf));2823conf = (struct mbmc_conf_tlv *)tlv;28242825conf->mbmc_en = enable;2826conf->band = 0; /* unused */28272828err = mt76_mcu_skb_send_msg(mdev, skb, MCU_UNI_CMD(SET_DBDC_PARMS),2829true);28302831return err;2832}28332834static void2835mt7925_mcu_build_scan_ie_tlv(struct mt76_dev *mdev,2836struct sk_buff *skb,2837struct ieee80211_scan_ies *scan_ies)2838{2839u32 max_len = sizeof(struct scan_ie_tlv) + MT76_CONNAC_SCAN_IE_LEN;2840struct scan_ie_tlv *ie;2841enum nl80211_band i;2842struct tlv *tlv;2843const u8 *ies;2844u16 ies_len;28452846for (i = 0; i <= NL80211_BAND_6GHZ; i++) {2847if (i == NL80211_BAND_60GHZ)2848continue;28492850ies = scan_ies->ies[i];2851ies_len = scan_ies->len[i];28522853if (!ies || !ies_len)2854continue;28552856if (ies_len > max_len)2857return;28582859tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_IE,2860sizeof(*ie) + ies_len);2861ie = (struct scan_ie_tlv *)tlv;28622863memcpy(ie->ies, ies, ies_len);2864ie->ies_len = cpu_to_le16(ies_len);28652866switch (i) {2867case NL80211_BAND_2GHZ:2868ie->band = 1;2869break;2870case NL80211_BAND_6GHZ:2871ie->band = 3;2872break;2873default:2874ie->band = 2;2875break;2876}28772878max_len -= (sizeof(*ie) + ies_len);2879}2880}28812882int mt7925_mcu_hw_scan(struct mt76_phy *phy, struct ieee80211_vif *vif,2883struct ieee80211_scan_request *scan_req)2884{2885struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;2886struct cfg80211_scan_request *sreq = &scan_req->req;2887int n_ssids = 0, err, i;2888struct ieee80211_channel **scan_list = sreq->channels;2889struct mt76_dev *mdev = phy->dev;2890struct mt76_connac_mcu_scan_channel *chan;2891struct sk_buff *skb;2892struct scan_hdr_tlv *hdr;2893struct scan_req_tlv *req;2894struct scan_ssid_tlv *ssid;2895struct scan_bssid_tlv *bssid;2896struct scan_chan_info_tlv *chan_info;2897struct scan_ie_tlv *ie;2898struct scan_misc_tlv *misc;2899struct tlv *tlv;2900int max_len;29012902if (test_bit(MT76_HW_SCANNING, &phy->state))2903return -EBUSY;29042905max_len = sizeof(*hdr) + sizeof(*req) + sizeof(*ssid) +2906sizeof(*bssid) * MT7925_RNR_SCAN_MAX_BSSIDS +2907sizeof(*chan_info) + sizeof(*misc) + sizeof(*ie) +2908MT76_CONNAC_SCAN_IE_LEN;29092910skb = mt76_mcu_msg_alloc(mdev, NULL, max_len);2911if (!skb)2912return -ENOMEM;29132914set_bit(MT76_HW_SCANNING, &phy->state);2915mvif->scan_seq_num = (mvif->scan_seq_num + 1) & 0x7f;29162917hdr = (struct scan_hdr_tlv *)skb_put(skb, sizeof(*hdr));2918hdr->seq_num = mvif->scan_seq_num | mvif->band_idx << 7;2919hdr->bss_idx = mvif->idx;29202921tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_REQ, sizeof(*req));2922req = (struct scan_req_tlv *)tlv;2923req->scan_type = sreq->n_ssids ? 1 : 0;2924req->probe_req_num = sreq->n_ssids ? 2 : 0;29252926tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_SSID, sizeof(*ssid));2927ssid = (struct scan_ssid_tlv *)tlv;2928for (i = 0; i < sreq->n_ssids; i++) {2929if (!sreq->ssids[i].ssid_len)2930continue;2931if (i >= MT7925_RNR_SCAN_MAX_BSSIDS)2932break;29332934ssid->ssids[n_ssids].ssid_len = cpu_to_le32(sreq->ssids[i].ssid_len);2935memcpy(ssid->ssids[n_ssids].ssid, sreq->ssids[i].ssid,2936sreq->ssids[i].ssid_len);2937n_ssids++;2938}2939ssid->ssid_type = n_ssids ? BIT(2) : BIT(0);2940ssid->ssids_num = n_ssids;29412942if (sreq->n_6ghz_params) {2943u8 j;29442945mt76_connac_mcu_build_rnr_scan_param(mdev, sreq);29462947for (j = 0; j < mdev->rnr.bssid_num; j++) {2948if (j >= MT7925_RNR_SCAN_MAX_BSSIDS)2949break;29502951tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_BSSID,2952sizeof(*bssid));2953bssid = (struct scan_bssid_tlv *)tlv;29542955ether_addr_copy(bssid->bssid, mdev->rnr.bssid[j]);2956bssid->match_ch = mdev->rnr.channel[j];2957bssid->match_ssid_ind = MT7925_RNR_SCAN_MAX_BSSIDS;2958bssid->match_short_ssid_ind = MT7925_RNR_SCAN_MAX_BSSIDS;2959}2960req->scan_func |= SCAN_FUNC_RNR_SCAN;2961} else {2962tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_BSSID, sizeof(*bssid));2963bssid = (struct scan_bssid_tlv *)tlv;29642965ether_addr_copy(bssid->bssid, sreq->bssid);2966}29672968tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_CHANNEL, sizeof(*chan_info));2969chan_info = (struct scan_chan_info_tlv *)tlv;2970chan_info->channels_num = min_t(u8, sreq->n_channels,2971ARRAY_SIZE(chan_info->channels));2972for (i = 0; i < chan_info->channels_num; i++) {2973chan = &chan_info->channels[i];29742975switch (scan_list[i]->band) {2976case NL80211_BAND_2GHZ:2977chan->band = 1;2978break;2979case NL80211_BAND_6GHZ:2980chan->band = 3;2981break;2982default:2983chan->band = 2;2984break;2985}2986chan->channel_num = scan_list[i]->hw_value;2987}2988chan_info->channel_type = sreq->n_channels ? 4 : 0;29892990req->scan_func |= SCAN_FUNC_SPLIT_SCAN;29912992tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_MISC, sizeof(*misc));2993misc = (struct scan_misc_tlv *)tlv;2994if (sreq->flags & NL80211_SCAN_FLAG_RANDOM_ADDR) {2995get_random_mask_addr(misc->random_mac, sreq->mac_addr,2996sreq->mac_addr_mask);2997req->scan_func |= SCAN_FUNC_RANDOM_MAC;2998}29993000/* Append scan probe IEs as the last tlv */3001mt7925_mcu_build_scan_ie_tlv(mdev, skb, &scan_req->ies);30023003err = mt76_mcu_skb_send_msg(mdev, skb, MCU_UNI_CMD(SCAN_REQ),3004true);3005if (err < 0)3006clear_bit(MT76_HW_SCANNING, &phy->state);30073008return err;3009}3010EXPORT_SYMBOL_GPL(mt7925_mcu_hw_scan);30113012int mt7925_mcu_sched_scan_req(struct mt76_phy *phy,3013struct ieee80211_vif *vif,3014struct cfg80211_sched_scan_request *sreq,3015struct ieee80211_scan_ies *ies)3016{3017struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;3018struct ieee80211_channel **scan_list = sreq->channels;3019struct mt76_connac_mcu_scan_channel *chan;3020struct mt76_dev *mdev = phy->dev;3021struct cfg80211_match_set *cfg_match;3022struct cfg80211_ssid *cfg_ssid;30233024struct scan_hdr_tlv *hdr;3025struct scan_sched_req *req;3026struct scan_ssid_tlv *ssid;3027struct scan_chan_info_tlv *chan_info;3028struct scan_ie_tlv *ie;3029struct scan_sched_ssid_match_sets *match;3030struct sk_buff *skb;3031struct tlv *tlv;3032int i, max_len;30333034max_len = sizeof(*hdr) + sizeof(*req) + sizeof(*ssid) +3035sizeof(*chan_info) + sizeof(*ie) +3036sizeof(*match);30373038skb = mt76_mcu_msg_alloc(mdev, NULL, max_len);3039if (!skb)3040return -ENOMEM;30413042mvif->scan_seq_num = (mvif->scan_seq_num + 1) & 0x7f;30433044hdr = (struct scan_hdr_tlv *)skb_put(skb, sizeof(*hdr));3045hdr->seq_num = mvif->scan_seq_num | mvif->band_idx << 7;3046hdr->bss_idx = mvif->idx;30473048tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_SCHED_REQ, sizeof(*req));3049req = (struct scan_sched_req *)tlv;3050req->version = 1;30513052if (sreq->flags & NL80211_SCAN_FLAG_RANDOM_ADDR)3053req->scan_func |= SCAN_FUNC_RANDOM_MAC;30543055req->intervals_num = sreq->n_scan_plans;3056for (i = 0; i < req->intervals_num; i++)3057req->intervals[i] = cpu_to_le16(sreq->scan_plans[i].interval);30583059tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_SSID, sizeof(*ssid));3060ssid = (struct scan_ssid_tlv *)tlv;30613062ssid->ssids_num = sreq->n_ssids;3063ssid->ssid_type = BIT(2);3064for (i = 0; i < ssid->ssids_num; i++) {3065cfg_ssid = &sreq->ssids[i];3066memcpy(ssid->ssids[i].ssid, cfg_ssid->ssid, cfg_ssid->ssid_len);3067ssid->ssids[i].ssid_len = cpu_to_le32(cfg_ssid->ssid_len);3068}30693070tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_SSID_MATCH_SETS, sizeof(*match));3071match = (struct scan_sched_ssid_match_sets *)tlv;3072match->match_num = sreq->n_match_sets;3073for (i = 0; i < match->match_num; i++) {3074cfg_match = &sreq->match_sets[i];3075memcpy(match->match[i].ssid, cfg_match->ssid.ssid,3076cfg_match->ssid.ssid_len);3077match->match[i].rssi_th = cpu_to_le32(cfg_match->rssi_thold);3078match->match[i].ssid_len = cfg_match->ssid.ssid_len;3079}30803081tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_CHANNEL, sizeof(*chan_info));3082chan_info = (struct scan_chan_info_tlv *)tlv;3083chan_info->channels_num = min_t(u8, sreq->n_channels,3084ARRAY_SIZE(chan_info->channels));3085for (i = 0; i < chan_info->channels_num; i++) {3086chan = &chan_info->channels[i];30873088switch (scan_list[i]->band) {3089case NL80211_BAND_2GHZ:3090chan->band = 1;3091break;3092case NL80211_BAND_6GHZ:3093chan->band = 3;3094break;3095default:3096chan->band = 2;3097break;3098}3099chan->channel_num = scan_list[i]->hw_value;3100}3101chan_info->channel_type = sreq->n_channels ? 4 : 0;31023103/* Append scan probe IEs as the last tlv */3104mt7925_mcu_build_scan_ie_tlv(mdev, skb, ies);31053106return mt76_mcu_skb_send_msg(mdev, skb, MCU_UNI_CMD(SCAN_REQ),3107true);3108}3109EXPORT_SYMBOL_GPL(mt7925_mcu_sched_scan_req);31103111int3112mt7925_mcu_sched_scan_enable(struct mt76_phy *phy,3113struct ieee80211_vif *vif,3114bool enable)3115{3116struct mt76_dev *mdev = phy->dev;3117struct scan_sched_enable *req;3118struct scan_hdr_tlv *hdr;3119struct sk_buff *skb;3120struct tlv *tlv;3121int max_len;31223123max_len = sizeof(*hdr) + sizeof(*req);31243125skb = mt76_mcu_msg_alloc(mdev, NULL, max_len);3126if (!skb)3127return -ENOMEM;31283129hdr = (struct scan_hdr_tlv *)skb_put(skb, sizeof(*hdr));3130hdr->seq_num = 0;3131hdr->bss_idx = 0;31323133tlv = mt76_connac_mcu_add_tlv(skb, UNI_SCAN_SCHED_ENABLE, sizeof(*req));3134req = (struct scan_sched_enable *)tlv;3135req->active = !enable;31363137if (enable)3138set_bit(MT76_HW_SCHED_SCANNING, &phy->state);3139else3140clear_bit(MT76_HW_SCHED_SCANNING, &phy->state);31413142return mt76_mcu_skb_send_msg(mdev, skb, MCU_UNI_CMD(SCAN_REQ),3143true);3144}31453146int mt7925_mcu_cancel_hw_scan(struct mt76_phy *phy,3147struct ieee80211_vif *vif)3148{3149struct mt76_vif_link *mvif = (struct mt76_vif_link *)vif->drv_priv;3150struct {3151struct scan_hdr {3152u8 seq_num;3153u8 bss_idx;3154u8 pad[2];3155} __packed hdr;3156struct scan_cancel_tlv {3157__le16 tag;3158__le16 len;3159u8 is_ext_channel;3160u8 rsv[3];3161} __packed cancel;3162} req = {3163.hdr = {3164.seq_num = mvif->scan_seq_num,3165.bss_idx = mvif->idx,3166},3167.cancel = {3168.tag = cpu_to_le16(UNI_SCAN_CANCEL),3169.len = cpu_to_le16(sizeof(struct scan_cancel_tlv)),3170},3171};31723173if (test_and_clear_bit(MT76_HW_SCANNING, &phy->state)) {3174struct cfg80211_scan_info info = {3175.aborted = true,3176};31773178ieee80211_scan_completed(phy->hw, &info);3179}31803181return mt76_mcu_send_msg(phy->dev, MCU_UNI_CMD(SCAN_REQ),3182&req, sizeof(req), true);3183}3184EXPORT_SYMBOL_GPL(mt7925_mcu_cancel_hw_scan);31853186int mt7925_mcu_set_channel_domain(struct mt76_phy *phy)3187{3188int len, i, n_max_channels, n_2ch = 0, n_5ch = 0, n_6ch = 0;3189struct {3190struct {3191u8 alpha2[4]; /* regulatory_request.alpha2 */3192u8 bw_2g; /* BW_20_40M 03193* BW_20M 13194* BW_20_40_80M 23195* BW_20_40_80_160M 33196* BW_20_40_80_8080M 43197*/3198u8 bw_5g;3199u8 bw_6g;3200u8 pad;3201} __packed hdr;3202struct n_chan {3203__le16 tag;3204__le16 len;3205u8 n_2ch;3206u8 n_5ch;3207u8 n_6ch;3208u8 pad;3209} __packed n_ch;3210} req = {3211.hdr = {3212.bw_2g = 0,3213.bw_5g = 3, /* BW_20_40_80_160M */3214.bw_6g = 3,3215},3216.n_ch = {3217.tag = cpu_to_le16(2),3218},3219};3220struct mt76_connac_mcu_chan {3221__le16 hw_value;3222__le16 pad;3223__le32 flags;3224} __packed channel;3225struct mt76_dev *dev = phy->dev;3226struct ieee80211_channel *chan;3227struct sk_buff *skb;32283229n_max_channels = phy->sband_2g.sband.n_channels +3230phy->sband_5g.sband.n_channels +3231phy->sband_6g.sband.n_channels;3232len = sizeof(req) + n_max_channels * sizeof(channel);32333234skb = mt76_mcu_msg_alloc(dev, NULL, len);3235if (!skb)3236return -ENOMEM;32373238skb_reserve(skb, sizeof(req));32393240for (i = 0; i < phy->sband_2g.sband.n_channels; i++) {3241chan = &phy->sband_2g.sband.channels[i];3242if (chan->flags & IEEE80211_CHAN_DISABLED)3243continue;32443245channel.hw_value = cpu_to_le16(chan->hw_value);3246channel.flags = cpu_to_le32(chan->flags);3247channel.pad = 0;32483249skb_put_data(skb, &channel, sizeof(channel));3250n_2ch++;3251}3252for (i = 0; i < phy->sband_5g.sband.n_channels; i++) {3253chan = &phy->sband_5g.sband.channels[i];3254if (chan->flags & IEEE80211_CHAN_DISABLED)3255continue;32563257channel.hw_value = cpu_to_le16(chan->hw_value);3258channel.flags = cpu_to_le32(chan->flags);3259channel.pad = 0;32603261skb_put_data(skb, &channel, sizeof(channel));3262n_5ch++;3263}3264for (i = 0; i < phy->sband_6g.sband.n_channels; i++) {3265chan = &phy->sband_6g.sband.channels[i];3266if (chan->flags & IEEE80211_CHAN_DISABLED)3267continue;32683269channel.hw_value = cpu_to_le16(chan->hw_value);3270channel.flags = cpu_to_le32(chan->flags);3271channel.pad = 0;32723273skb_put_data(skb, &channel, sizeof(channel));3274n_6ch++;3275}32763277BUILD_BUG_ON(sizeof(dev->alpha2) > sizeof(req.hdr.alpha2));3278memcpy(req.hdr.alpha2, dev->alpha2, sizeof(dev->alpha2));3279req.n_ch.n_2ch = n_2ch;3280req.n_ch.n_5ch = n_5ch;3281req.n_ch.n_6ch = n_6ch;3282len = sizeof(struct n_chan) + (n_2ch + n_5ch + n_6ch) * sizeof(channel);3283req.n_ch.len = cpu_to_le16(len);3284memcpy(__skb_push(skb, sizeof(req)), &req, sizeof(req));32853286return mt76_mcu_skb_send_msg(dev, skb, MCU_UNI_CMD(SET_DOMAIN_INFO),3287true);3288}3289EXPORT_SYMBOL_GPL(mt7925_mcu_set_channel_domain);32903291static int3292__mt7925_mcu_set_clc(struct mt792x_dev *dev, u8 *alpha2,3293enum environment_cap env_cap,3294struct mt7925_clc *clc, u8 idx)3295{3296struct mt7925_clc_segment *seg;3297struct sk_buff *skb;3298struct {3299u8 rsv[4];3300__le16 tag;3301__le16 len;33023303u8 ver;3304u8 pad0;3305__le16 size;3306u8 idx;3307u8 env;3308u8 acpi_conf;3309u8 pad1;3310u8 alpha2[2];3311u8 type[2];3312u8 rsvd[64];3313} __packed req = {3314.tag = cpu_to_le16(0x3),3315.len = cpu_to_le16(sizeof(req) - 4),33163317.idx = idx,3318.env = env_cap,3319};3320int ret, valid_cnt = 0;3321u8 *pos, *last_pos;33223323if (!clc)3324return 0;33253326req.ver = clc->ver;3327pos = clc->data + sizeof(*seg) * clc->t0.nr_seg;3328last_pos = clc->data + le32_to_cpu(*(__le32 *)(clc->data + 4));3329while (pos < last_pos) {3330struct mt7925_clc_rule *rule = (struct mt7925_clc_rule *)pos;33313332pos += sizeof(*rule);3333if (rule->alpha2[0] != alpha2[0] ||3334rule->alpha2[1] != alpha2[1])3335continue;33363337seg = (struct mt7925_clc_segment *)clc->data3338+ rule->seg_idx - 1;33393340memcpy(req.alpha2, rule->alpha2, 2);3341memcpy(req.type, rule->type, 2);33423343req.size = cpu_to_le16(seg->len);3344dev->phy.clc_chan_conf = clc->ver == 1 ? 0xff : rule->flag;3345skb = __mt76_mcu_msg_alloc(&dev->mt76, &req,3346le16_to_cpu(req.size) + sizeof(req),3347sizeof(req), GFP_KERNEL);3348if (!skb)3349return -ENOMEM;3350skb_put_data(skb, clc->data + seg->offset, seg->len);33513352ret = mt76_mcu_skb_send_msg(&dev->mt76, skb,3353MCU_UNI_CMD(SET_POWER_LIMIT),3354true);3355if (ret < 0)3356return ret;3357valid_cnt++;3358}33593360if (!valid_cnt) {3361dev->phy.clc_chan_conf = 0xff;3362return -ENOENT;3363}33643365return 0;3366}33673368int mt7925_mcu_set_clc(struct mt792x_dev *dev, u8 *alpha2,3369enum environment_cap env_cap)3370{3371struct mt792x_phy *phy = (struct mt792x_phy *)&dev->phy;3372int i, ret;33733374/* submit all clc config */3375for (i = 0; i < ARRAY_SIZE(phy->clc); i++) {3376if (i == MT792x_CLC_BE_CTRL)3377continue;33783379ret = __mt7925_mcu_set_clc(dev, alpha2, env_cap,3380phy->clc[i], i);33813382/* If no country found, set "00" as default */3383if (ret == -ENOENT)3384ret = __mt7925_mcu_set_clc(dev, "00",3385ENVIRON_INDOOR,3386phy->clc[i], i);3387if (ret < 0)3388return ret;3389}3390return 0;3391}33923393int mt7925_mcu_fill_message(struct mt76_dev *mdev, struct sk_buff *skb,3394int cmd, int *wait_seq)3395{3396int txd_len, mcu_cmd = FIELD_GET(__MCU_CMD_FIELD_ID, cmd);3397struct mt76_connac2_mcu_uni_txd *uni_txd;3398struct mt76_connac2_mcu_txd *mcu_txd;3399__le32 *txd;3400u32 val;3401u8 seq;34023403/* TODO: make dynamic based on msg type */3404mdev->mcu.timeout = 20 * HZ;34053406seq = ++mdev->mcu.msg_seq & 0xf;3407if (!seq)3408seq = ++mdev->mcu.msg_seq & 0xf;34093410if (cmd == MCU_CMD(FW_SCATTER))3411goto exit;34123413txd_len = cmd & __MCU_CMD_FIELD_UNI ? sizeof(*uni_txd) : sizeof(*mcu_txd);3414txd = (__le32 *)skb_push(skb, txd_len);34153416val = FIELD_PREP(MT_TXD0_TX_BYTES, skb->len) |3417FIELD_PREP(MT_TXD0_PKT_FMT, MT_TX_TYPE_CMD) |3418FIELD_PREP(MT_TXD0_Q_IDX, MT_TX_MCU_PORT_RX_Q0);3419txd[0] = cpu_to_le32(val);34203421val = FIELD_PREP(MT_TXD1_HDR_FORMAT, MT_HDR_FORMAT_CMD);3422txd[1] = cpu_to_le32(val);34233424if (cmd & __MCU_CMD_FIELD_UNI) {3425uni_txd = (struct mt76_connac2_mcu_uni_txd *)txd;3426uni_txd->len = cpu_to_le16(skb->len - sizeof(uni_txd->txd));3427uni_txd->cid = cpu_to_le16(mcu_cmd);3428uni_txd->s2d_index = MCU_S2D_H2N;3429uni_txd->pkt_type = MCU_PKT_ID;3430uni_txd->seq = seq;34313432if (cmd & __MCU_CMD_FIELD_QUERY)3433uni_txd->option = MCU_CMD_UNI_QUERY_ACK;3434else3435uni_txd->option = MCU_CMD_UNI_EXT_ACK;34363437if (cmd == MCU_UNI_CMD(HIF_CTRL) ||3438cmd == MCU_UNI_CMD(CHIP_CONFIG))3439uni_txd->option &= ~MCU_CMD_ACK;34403441if (mcu_cmd == MCU_UNI_CMD_TESTMODE_CTRL ||3442mcu_cmd == MCU_UNI_CMD_TESTMODE_RX_STAT) {3443if (cmd & __MCU_CMD_FIELD_QUERY)3444uni_txd->option = 0x2;3445else3446uni_txd->option = 0x6;3447}34483449goto exit;3450}34513452mcu_txd = (struct mt76_connac2_mcu_txd *)txd;3453mcu_txd->len = cpu_to_le16(skb->len - sizeof(mcu_txd->txd));3454mcu_txd->pq_id = cpu_to_le16(MCU_PQ_ID(MT_TX_PORT_IDX_MCU,3455MT_TX_MCU_PORT_RX_Q0));3456mcu_txd->pkt_type = MCU_PKT_ID;3457mcu_txd->seq = seq;3458mcu_txd->cid = mcu_cmd;3459mcu_txd->ext_cid = FIELD_GET(__MCU_CMD_FIELD_EXT_ID, cmd);34603461if (mcu_txd->ext_cid || (cmd & __MCU_CMD_FIELD_CE)) {3462if (cmd & __MCU_CMD_FIELD_QUERY)3463mcu_txd->set_query = MCU_Q_QUERY;3464else3465mcu_txd->set_query = MCU_Q_SET;3466mcu_txd->ext_cid_ack = !!mcu_txd->ext_cid;3467} else {3468mcu_txd->set_query = MCU_Q_NA;3469}34703471if (cmd & __MCU_CMD_FIELD_WA)3472mcu_txd->s2d_index = MCU_S2D_H2C;3473else3474mcu_txd->s2d_index = MCU_S2D_H2N;34753476exit:3477if (wait_seq)3478*wait_seq = seq;34793480return 0;3481}3482EXPORT_SYMBOL_GPL(mt7925_mcu_fill_message);34833484int mt7925_mcu_set_rts_thresh(struct mt792x_phy *phy, u32 val)3485{3486struct {3487u8 band_idx;3488u8 _rsv[3];34893490__le16 tag;3491__le16 len;3492__le32 len_thresh;3493__le32 pkt_thresh;3494} __packed req = {3495.band_idx = phy->mt76->band_idx,3496.tag = cpu_to_le16(UNI_BAND_CONFIG_RTS_THRESHOLD),3497.len = cpu_to_le16(sizeof(req) - 4),3498.len_thresh = cpu_to_le32(val),3499.pkt_thresh = cpu_to_le32(0x2),3500};35013502return mt76_mcu_send_msg(&phy->dev->mt76, MCU_UNI_CMD(BAND_CONFIG),3503&req, sizeof(req), true);3504}35053506int mt7925_mcu_set_radio_en(struct mt792x_phy *phy, bool enable)3507{3508struct {3509u8 band_idx;3510u8 _rsv[3];35113512__le16 tag;3513__le16 len;3514u8 enable;3515u8 _rsv2[3];3516} __packed req = {3517.band_idx = phy->mt76->band_idx,3518.tag = cpu_to_le16(UNI_BAND_CONFIG_RADIO_ENABLE),3519.len = cpu_to_le16(sizeof(req) - 4),3520.enable = enable,3521};35223523return mt76_mcu_send_msg(&phy->dev->mt76, MCU_UNI_CMD(BAND_CONFIG),3524&req, sizeof(req), true);3525}35263527static void3528mt7925_mcu_build_sku(struct mt76_dev *dev, s8 *sku,3529struct mt76_power_limits *limits,3530enum nl80211_band band)3531{3532int i, offset = sizeof(limits->cck);35333534memset(sku, 127, MT_CONNAC3_SKU_POWER_LIMIT);35353536if (band == NL80211_BAND_2GHZ) {3537/* cck */3538memcpy(sku, limits->cck, sizeof(limits->cck));3539}35403541/* ofdm */3542memcpy(&sku[offset], limits->ofdm, sizeof(limits->ofdm));3543offset += (sizeof(limits->ofdm) * 5);35443545/* ht */3546for (i = 0; i < 2; i++) {3547memcpy(&sku[offset], limits->mcs[i], 8);3548offset += 8;3549}3550sku[offset++] = limits->mcs[0][0];35513552/* vht */3553for (i = 0; i < ARRAY_SIZE(limits->mcs); i++) {3554memcpy(&sku[offset], limits->mcs[i],3555ARRAY_SIZE(limits->mcs[i]));3556offset += 12;3557}35583559/* he */3560for (i = 0; i < ARRAY_SIZE(limits->ru); i++) {3561memcpy(&sku[offset], limits->ru[i], ARRAY_SIZE(limits->ru[i]));3562offset += ARRAY_SIZE(limits->ru[i]);3563}35643565/* eht */3566for (i = 0; i < ARRAY_SIZE(limits->eht); i++) {3567memcpy(&sku[offset], limits->eht[i], ARRAY_SIZE(limits->eht[i]));3568offset += ARRAY_SIZE(limits->eht[i]);3569}3570}35713572static int3573mt7925_mcu_rate_txpower_band(struct mt76_phy *phy,3574enum nl80211_band band)3575{3576int tx_power, n_chan, last_ch, err = 0, idx = 0;3577int i, sku_len, batch_size, batch_len = 3;3578struct mt76_dev *dev = phy->dev;3579static const u8 chan_list_2ghz[] = {35801, 2, 3, 4, 5, 6, 7,35818, 9, 10, 11, 12, 13, 143582};3583static const u8 chan_list_5ghz[] = {358436, 38, 40, 42, 44, 46, 48,358550, 52, 54, 56, 58, 60, 62,358664, 100, 102, 104, 106, 108, 110,3587112, 114, 116, 118, 120, 122, 124,3588126, 128, 132, 134, 136, 138, 140,3589142, 144, 149, 151, 153, 155, 157,3590159, 161, 165, 1673591};3592static const u8 chan_list_6ghz[] = {35931, 3, 5, 7, 9, 11, 13,359415, 17, 19, 21, 23, 25, 27,359529, 33, 35, 37, 39, 41, 43,359645, 47, 49, 51, 53, 55, 57,359759, 61, 65, 67, 69, 71, 73,359875, 77, 79, 81, 83, 85, 87,359989, 91, 93, 97, 99, 101, 103,3600105, 107, 109, 111, 113, 115, 117,3601119, 121, 123, 125, 129, 131, 133,3602135, 137, 139, 141, 143, 145, 147,3603149, 151, 153, 155, 157, 161, 163,3604165, 167, 169, 171, 173, 175, 177,3605179, 181, 183, 185, 187, 189, 193,3606195, 197, 199, 201, 203, 205, 207,3607209, 211, 213, 215, 217, 219, 221,3608225, 227, 229, 2333609};3610struct mt76_power_limits *limits;3611struct mt7925_sku_tlv *sku_tlbv;3612const u8 *ch_list;36133614sku_len = sizeof(*sku_tlbv);3615tx_power = 2 * phy->hw->conf.power_level;3616if (!tx_power)3617tx_power = 127;36183619if (band == NL80211_BAND_2GHZ) {3620n_chan = ARRAY_SIZE(chan_list_2ghz);3621ch_list = chan_list_2ghz;3622last_ch = chan_list_2ghz[ARRAY_SIZE(chan_list_2ghz) - 1];3623} else if (band == NL80211_BAND_6GHZ) {3624n_chan = ARRAY_SIZE(chan_list_6ghz);3625ch_list = chan_list_6ghz;3626last_ch = chan_list_6ghz[ARRAY_SIZE(chan_list_6ghz) - 1];3627} else {3628n_chan = ARRAY_SIZE(chan_list_5ghz);3629ch_list = chan_list_5ghz;3630last_ch = chan_list_5ghz[ARRAY_SIZE(chan_list_5ghz) - 1];3631}3632batch_size = DIV_ROUND_UP(n_chan, batch_len);36333634limits = devm_kmalloc(dev->dev, sizeof(*limits), GFP_KERNEL);3635if (!limits)3636return -ENOMEM;36373638sku_tlbv = devm_kmalloc(dev->dev, sku_len, GFP_KERNEL);3639if (!sku_tlbv) {3640devm_kfree(dev->dev, limits);3641return -ENOMEM;3642}36433644for (i = 0; i < batch_size; i++) {3645struct mt7925_tx_power_limit_tlv *tx_power_tlv;3646int j, msg_len, num_ch;3647struct sk_buff *skb;36483649num_ch = i == batch_size - 1 ? n_chan % batch_len : batch_len;3650msg_len = sizeof(*tx_power_tlv) + num_ch * sku_len;3651skb = mt76_mcu_msg_alloc(dev, NULL, msg_len);3652if (!skb) {3653err = -ENOMEM;3654goto out;3655}36563657tx_power_tlv = (struct mt7925_tx_power_limit_tlv *)3658skb_put(skb, sizeof(*tx_power_tlv));36593660BUILD_BUG_ON(sizeof(dev->alpha2) > sizeof(tx_power_tlv->alpha2));3661memcpy(tx_power_tlv->alpha2, dev->alpha2, sizeof(dev->alpha2));3662tx_power_tlv->n_chan = num_ch;3663tx_power_tlv->tag = cpu_to_le16(0x1);3664tx_power_tlv->len = cpu_to_le16(sizeof(*tx_power_tlv));36653666switch (band) {3667case NL80211_BAND_2GHZ:3668tx_power_tlv->band = 1;3669break;3670case NL80211_BAND_6GHZ:3671tx_power_tlv->band = 3;3672break;3673default:3674tx_power_tlv->band = 2;3675break;3676}36773678for (j = 0; j < num_ch; j++, idx++) {3679struct ieee80211_channel chan = {3680.hw_value = ch_list[idx],3681.band = band,3682};3683s8 reg_power, sar_power;36843685reg_power = mt76_connac_get_ch_power(phy, &chan,3686tx_power);3687sar_power = mt76_get_sar_power(phy, &chan, reg_power);36883689mt76_get_rate_power_limits(phy, &chan, limits,3690sar_power);36913692tx_power_tlv->last_msg = ch_list[idx] == last_ch;3693sku_tlbv->channel = ch_list[idx];36943695mt7925_mcu_build_sku(dev, sku_tlbv->pwr_limit,3696limits, band);3697skb_put_data(skb, sku_tlbv, sku_len);3698}3699err = mt76_mcu_skb_send_msg(dev, skb,3700MCU_UNI_CMD(SET_POWER_LIMIT),3701true);3702if (err < 0)3703goto out;3704}37053706out:3707devm_kfree(dev->dev, sku_tlbv);3708devm_kfree(dev->dev, limits);3709return err;3710}37113712int mt7925_mcu_set_rate_txpower(struct mt76_phy *phy)3713{3714int err;37153716if (phy->cap.has_2ghz) {3717err = mt7925_mcu_rate_txpower_band(phy,3718NL80211_BAND_2GHZ);3719if (err < 0)3720return err;3721}37223723if (phy->cap.has_5ghz) {3724err = mt7925_mcu_rate_txpower_band(phy,3725NL80211_BAND_5GHZ);3726if (err < 0)3727return err;3728}37293730if (phy->cap.has_6ghz) {3731err = mt7925_mcu_rate_txpower_band(phy,3732NL80211_BAND_6GHZ);3733if (err < 0)3734return err;3735}37363737return 0;3738}37393740int mt7925_mcu_wf_rf_pin_ctrl(struct mt792x_phy *phy)3741{3742#define UNI_CMD_RADIO_STATUS_GET 03743struct mt792x_dev *dev = phy->dev;3744struct sk_buff *skb;3745int ret;3746struct {3747__le16 tag;3748__le16 len;3749u8 rsv[4];3750} __packed req = {3751.tag = UNI_CMD_RADIO_STATUS_GET,3752.len = cpu_to_le16(sizeof(req)),3753};3754struct mt7925_radio_status_event {3755__le16 tag;3756__le16 len;37573758u8 data;3759u8 rsv[3];3760} __packed *status;37613762ret = mt76_mcu_send_and_get_msg(&dev->mt76,3763MCU_UNI_CMD(RADIO_STATUS),3764&req, sizeof(req), true, &skb);3765if (ret)3766return ret;37673768skb_pull(skb, sizeof(struct tlv));3769status = (struct mt7925_radio_status_event *)skb->data;3770ret = status->data;37713772dev_kfree_skb(skb);37733774return ret;3775}37763777int mt7925_mcu_set_rxfilter(struct mt792x_dev *dev, u32 fif,3778u8 bit_op, u32 bit_map)3779{3780struct mt792x_phy *phy = &dev->phy;3781struct {3782u8 band_idx;3783u8 rsv1[3];37843785__le16 tag;3786__le16 len;3787u8 mode;3788u8 rsv2[3];3789__le32 fif;3790__le32 bit_map; /* bit_* for bitmap update */3791u8 bit_op;3792u8 pad[51];3793} __packed req = {3794.band_idx = phy->mt76->band_idx,3795.tag = cpu_to_le16(UNI_BAND_CONFIG_SET_MAC80211_RX_FILTER),3796.len = cpu_to_le16(sizeof(req) - 4),37973798.mode = fif ? 0 : 1,3799.fif = cpu_to_le32(fif),3800.bit_map = cpu_to_le32(bit_map),3801.bit_op = bit_op,3802};38033804return mt76_mcu_send_msg(&phy->dev->mt76, MCU_UNI_CMD(BAND_CONFIG),3805&req, sizeof(req), true);3806}380738083809