162306a36Sopenharmony_ci// SPDX-License-Identifier: ISC
262306a36Sopenharmony_ci/* Copyright (C) 2020 MediaTek Inc. */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include <linux/fs.h>
562306a36Sopenharmony_ci#include <linux/firmware.h>
662306a36Sopenharmony_ci#include "mt7921.h"
762306a36Sopenharmony_ci#include "mcu.h"
862306a36Sopenharmony_ci#include "../mt76_connac2_mac.h"
962306a36Sopenharmony_ci#include "../mt792x_trace.h"
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#define MT_STA_BFER			BIT(0)
1262306a36Sopenharmony_ci#define MT_STA_BFEE			BIT(1)
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_cistatic bool mt7921_disable_clc;
1562306a36Sopenharmony_cimodule_param_named(disable_clc, mt7921_disable_clc, bool, 0644);
1662306a36Sopenharmony_ciMODULE_PARM_DESC(disable_clc, "disable CLC support");
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ciint mt7921_mcu_parse_response(struct mt76_dev *mdev, int cmd,
1962306a36Sopenharmony_ci			      struct sk_buff *skb, int seq)
2062306a36Sopenharmony_ci{
2162306a36Sopenharmony_ci	int mcu_cmd = FIELD_GET(__MCU_CMD_FIELD_ID, cmd);
2262306a36Sopenharmony_ci	struct mt76_connac2_mcu_rxd *rxd;
2362306a36Sopenharmony_ci	int ret = 0;
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci	if (!skb) {
2662306a36Sopenharmony_ci		dev_err(mdev->dev, "Message %08x (seq %d) timeout\n",
2762306a36Sopenharmony_ci			cmd, seq);
2862306a36Sopenharmony_ci		mt792x_reset(mdev);
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci		return -ETIMEDOUT;
3162306a36Sopenharmony_ci	}
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci	rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
3462306a36Sopenharmony_ci	if (seq != rxd->seq)
3562306a36Sopenharmony_ci		return -EAGAIN;
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	if (cmd == MCU_CMD(PATCH_SEM_CONTROL) ||
3862306a36Sopenharmony_ci	    cmd == MCU_CMD(PATCH_FINISH_REQ)) {
3962306a36Sopenharmony_ci		skb_pull(skb, sizeof(*rxd) - 4);
4062306a36Sopenharmony_ci		ret = *skb->data;
4162306a36Sopenharmony_ci	} else if (cmd == MCU_EXT_CMD(THERMAL_CTRL)) {
4262306a36Sopenharmony_ci		skb_pull(skb, sizeof(*rxd) + 4);
4362306a36Sopenharmony_ci		ret = le32_to_cpu(*(__le32 *)skb->data);
4462306a36Sopenharmony_ci	} else if (cmd == MCU_UNI_CMD(DEV_INFO_UPDATE) ||
4562306a36Sopenharmony_ci		   cmd == MCU_UNI_CMD(BSS_INFO_UPDATE) ||
4662306a36Sopenharmony_ci		   cmd == MCU_UNI_CMD(STA_REC_UPDATE) ||
4762306a36Sopenharmony_ci		   cmd == MCU_UNI_CMD(HIF_CTRL) ||
4862306a36Sopenharmony_ci		   cmd == MCU_UNI_CMD(OFFLOAD) ||
4962306a36Sopenharmony_ci		   cmd == MCU_UNI_CMD(SUSPEND)) {
5062306a36Sopenharmony_ci		struct mt76_connac_mcu_uni_event *event;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci		skb_pull(skb, sizeof(*rxd));
5362306a36Sopenharmony_ci		event = (struct mt76_connac_mcu_uni_event *)skb->data;
5462306a36Sopenharmony_ci		ret = le32_to_cpu(event->status);
5562306a36Sopenharmony_ci		/* skip invalid event */
5662306a36Sopenharmony_ci		if (mcu_cmd != event->cid)
5762306a36Sopenharmony_ci			ret = -EAGAIN;
5862306a36Sopenharmony_ci	} else if (cmd == MCU_CE_QUERY(REG_READ)) {
5962306a36Sopenharmony_ci		struct mt76_connac_mcu_reg_event *event;
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_ci		skb_pull(skb, sizeof(*rxd));
6262306a36Sopenharmony_ci		event = (struct mt76_connac_mcu_reg_event *)skb->data;
6362306a36Sopenharmony_ci		ret = (int)le32_to_cpu(event->val);
6462306a36Sopenharmony_ci	} else {
6562306a36Sopenharmony_ci		skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
6662306a36Sopenharmony_ci	}
6762306a36Sopenharmony_ci
6862306a36Sopenharmony_ci	return ret;
6962306a36Sopenharmony_ci}
7062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mt7921_mcu_parse_response);
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_cistatic int mt7921_mcu_read_eeprom(struct mt792x_dev *dev, u32 offset, u8 *val)
7362306a36Sopenharmony_ci{
7462306a36Sopenharmony_ci	struct mt7921_mcu_eeprom_info *res, req = {
7562306a36Sopenharmony_ci		.addr = cpu_to_le32(round_down(offset,
7662306a36Sopenharmony_ci				    MT7921_EEPROM_BLOCK_SIZE)),
7762306a36Sopenharmony_ci	};
7862306a36Sopenharmony_ci	struct sk_buff *skb;
7962306a36Sopenharmony_ci	int ret;
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_EXT_QUERY(EFUSE_ACCESS),
8262306a36Sopenharmony_ci					&req, sizeof(req), true, &skb);
8362306a36Sopenharmony_ci	if (ret)
8462306a36Sopenharmony_ci		return ret;
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	res = (struct mt7921_mcu_eeprom_info *)skb->data;
8762306a36Sopenharmony_ci	*val = res->data[offset % MT7921_EEPROM_BLOCK_SIZE];
8862306a36Sopenharmony_ci	dev_kfree_skb(skb);
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	return 0;
9162306a36Sopenharmony_ci}
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_ci#ifdef CONFIG_PM
9462306a36Sopenharmony_ci
9562306a36Sopenharmony_cistatic int
9662306a36Sopenharmony_cimt7921_mcu_set_ipv6_ns_filter(struct mt76_dev *dev,
9762306a36Sopenharmony_ci			      struct ieee80211_vif *vif, bool suspend)
9862306a36Sopenharmony_ci{
9962306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
10062306a36Sopenharmony_ci	struct {
10162306a36Sopenharmony_ci		struct {
10262306a36Sopenharmony_ci			u8 bss_idx;
10362306a36Sopenharmony_ci			u8 pad[3];
10462306a36Sopenharmony_ci		} __packed hdr;
10562306a36Sopenharmony_ci		struct mt76_connac_arpns_tlv arpns;
10662306a36Sopenharmony_ci	} req = {
10762306a36Sopenharmony_ci		.hdr = {
10862306a36Sopenharmony_ci			.bss_idx = mvif->mt76.idx,
10962306a36Sopenharmony_ci		},
11062306a36Sopenharmony_ci		.arpns = {
11162306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_OFFLOAD_OFFLOAD_ND),
11262306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct mt76_connac_arpns_tlv)),
11362306a36Sopenharmony_ci			.mode = suspend,
11462306a36Sopenharmony_ci		},
11562306a36Sopenharmony_ci	};
11662306a36Sopenharmony_ci
11762306a36Sopenharmony_ci	return mt76_mcu_send_msg(dev, MCU_UNI_CMD_OFFLOAD, &req, sizeof(req),
11862306a36Sopenharmony_ci				 true);
11962306a36Sopenharmony_ci}
12062306a36Sopenharmony_ci
12162306a36Sopenharmony_civoid mt7921_mcu_set_suspend_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
12262306a36Sopenharmony_ci{
12362306a36Sopenharmony_ci	if (IS_ENABLED(CONFIG_IPV6)) {
12462306a36Sopenharmony_ci		struct mt76_phy *phy = priv;
12562306a36Sopenharmony_ci
12662306a36Sopenharmony_ci		mt7921_mcu_set_ipv6_ns_filter(phy->dev, vif,
12762306a36Sopenharmony_ci					      !test_bit(MT76_STATE_RUNNING,
12862306a36Sopenharmony_ci					      &phy->state));
12962306a36Sopenharmony_ci	}
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	mt76_connac_mcu_set_suspend_iter(priv, mac, vif);
13262306a36Sopenharmony_ci}
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci#endif /* CONFIG_PM */
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_cistatic void
13762306a36Sopenharmony_cimt7921_mcu_uni_roc_event(struct mt792x_dev *dev, struct sk_buff *skb)
13862306a36Sopenharmony_ci{
13962306a36Sopenharmony_ci	struct mt7921_roc_grant_tlv *grant;
14062306a36Sopenharmony_ci	struct mt76_connac2_mcu_rxd *rxd;
14162306a36Sopenharmony_ci	int duration;
14262306a36Sopenharmony_ci
14362306a36Sopenharmony_ci	rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
14462306a36Sopenharmony_ci	grant = (struct mt7921_roc_grant_tlv *)(rxd->tlv + 4);
14562306a36Sopenharmony_ci
14662306a36Sopenharmony_ci	/* should never happen */
14762306a36Sopenharmony_ci	WARN_ON_ONCE((le16_to_cpu(grant->tag) != UNI_EVENT_ROC_GRANT));
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci	if (grant->reqtype == MT7921_ROC_REQ_ROC)
15062306a36Sopenharmony_ci		ieee80211_ready_on_channel(dev->mt76.phy.hw);
15162306a36Sopenharmony_ci
15262306a36Sopenharmony_ci	dev->phy.roc_grant = true;
15362306a36Sopenharmony_ci	wake_up(&dev->phy.roc_wait);
15462306a36Sopenharmony_ci	duration = le32_to_cpu(grant->max_interval);
15562306a36Sopenharmony_ci	mod_timer(&dev->phy.roc_timer,
15662306a36Sopenharmony_ci		  jiffies + msecs_to_jiffies(duration));
15762306a36Sopenharmony_ci}
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_cistatic void
16062306a36Sopenharmony_cimt7921_mcu_scan_event(struct mt792x_dev *dev, struct sk_buff *skb)
16162306a36Sopenharmony_ci{
16262306a36Sopenharmony_ci	struct mt76_phy *mphy = &dev->mt76.phy;
16362306a36Sopenharmony_ci	struct mt792x_phy *phy = (struct mt792x_phy *)mphy->priv;
16462306a36Sopenharmony_ci
16562306a36Sopenharmony_ci	spin_lock_bh(&dev->mt76.lock);
16662306a36Sopenharmony_ci	__skb_queue_tail(&phy->scan_event_list, skb);
16762306a36Sopenharmony_ci	spin_unlock_bh(&dev->mt76.lock);
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_ci	ieee80211_queue_delayed_work(mphy->hw, &phy->scan_work,
17062306a36Sopenharmony_ci				     MT792x_HW_SCAN_TIMEOUT);
17162306a36Sopenharmony_ci}
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_cistatic void
17462306a36Sopenharmony_cimt7921_mcu_connection_loss_iter(void *priv, u8 *mac,
17562306a36Sopenharmony_ci				struct ieee80211_vif *vif)
17662306a36Sopenharmony_ci{
17762306a36Sopenharmony_ci	struct mt76_vif *mvif = (struct mt76_vif *)vif->drv_priv;
17862306a36Sopenharmony_ci	struct mt76_connac_beacon_loss_event *event = priv;
17962306a36Sopenharmony_ci
18062306a36Sopenharmony_ci	if (mvif->idx != event->bss_idx)
18162306a36Sopenharmony_ci		return;
18262306a36Sopenharmony_ci
18362306a36Sopenharmony_ci	if (!(vif->driver_flags & IEEE80211_VIF_BEACON_FILTER) ||
18462306a36Sopenharmony_ci	    vif->type != NL80211_IFTYPE_STATION)
18562306a36Sopenharmony_ci		return;
18662306a36Sopenharmony_ci
18762306a36Sopenharmony_ci	ieee80211_connection_loss(vif);
18862306a36Sopenharmony_ci}
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_cistatic void
19162306a36Sopenharmony_cimt7921_mcu_connection_loss_event(struct mt792x_dev *dev, struct sk_buff *skb)
19262306a36Sopenharmony_ci{
19362306a36Sopenharmony_ci	struct mt76_connac_beacon_loss_event *event;
19462306a36Sopenharmony_ci	struct mt76_phy *mphy = &dev->mt76.phy;
19562306a36Sopenharmony_ci
19662306a36Sopenharmony_ci	skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
19762306a36Sopenharmony_ci	event = (struct mt76_connac_beacon_loss_event *)skb->data;
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	ieee80211_iterate_active_interfaces_atomic(mphy->hw,
20062306a36Sopenharmony_ci					IEEE80211_IFACE_ITER_RESUME_ALL,
20162306a36Sopenharmony_ci					mt7921_mcu_connection_loss_iter, event);
20262306a36Sopenharmony_ci}
20362306a36Sopenharmony_ci
20462306a36Sopenharmony_cistatic void
20562306a36Sopenharmony_cimt7921_mcu_debug_msg_event(struct mt792x_dev *dev, struct sk_buff *skb)
20662306a36Sopenharmony_ci{
20762306a36Sopenharmony_ci	struct mt7921_debug_msg {
20862306a36Sopenharmony_ci		__le16 id;
20962306a36Sopenharmony_ci		u8 type;
21062306a36Sopenharmony_ci		u8 flag;
21162306a36Sopenharmony_ci		__le32 value;
21262306a36Sopenharmony_ci		__le16 len;
21362306a36Sopenharmony_ci		u8 content[512];
21462306a36Sopenharmony_ci	} __packed * msg;
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci	skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
21762306a36Sopenharmony_ci	msg = (struct mt7921_debug_msg *)skb->data;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	if (msg->type == 3) { /* fw log */
22062306a36Sopenharmony_ci		u16 len = min_t(u16, le16_to_cpu(msg->len), 512);
22162306a36Sopenharmony_ci		int i;
22262306a36Sopenharmony_ci
22362306a36Sopenharmony_ci		for (i = 0 ; i < len; i++) {
22462306a36Sopenharmony_ci			if (!msg->content[i])
22562306a36Sopenharmony_ci				msg->content[i] = ' ';
22662306a36Sopenharmony_ci		}
22762306a36Sopenharmony_ci		wiphy_info(mt76_hw(dev)->wiphy, "%.*s", len, msg->content);
22862306a36Sopenharmony_ci	}
22962306a36Sopenharmony_ci}
23062306a36Sopenharmony_ci
23162306a36Sopenharmony_cistatic void
23262306a36Sopenharmony_cimt7921_mcu_low_power_event(struct mt792x_dev *dev, struct sk_buff *skb)
23362306a36Sopenharmony_ci{
23462306a36Sopenharmony_ci	struct mt7921_mcu_lp_event {
23562306a36Sopenharmony_ci		u8 state;
23662306a36Sopenharmony_ci		u8 reserved[3];
23762306a36Sopenharmony_ci	} __packed * event;
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ci	skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
24062306a36Sopenharmony_ci	event = (struct mt7921_mcu_lp_event *)skb->data;
24162306a36Sopenharmony_ci
24262306a36Sopenharmony_ci	trace_lp_event(dev, event->state);
24362306a36Sopenharmony_ci}
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_cistatic void
24662306a36Sopenharmony_cimt7921_mcu_tx_done_event(struct mt792x_dev *dev, struct sk_buff *skb)
24762306a36Sopenharmony_ci{
24862306a36Sopenharmony_ci	struct mt7921_mcu_tx_done_event *event;
24962306a36Sopenharmony_ci
25062306a36Sopenharmony_ci	skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
25162306a36Sopenharmony_ci	event = (struct mt7921_mcu_tx_done_event *)skb->data;
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_ci	mt7921_mac_add_txs(dev, event->txs);
25462306a36Sopenharmony_ci}
25562306a36Sopenharmony_ci
25662306a36Sopenharmony_cistatic void
25762306a36Sopenharmony_cimt7921_mcu_rx_unsolicited_event(struct mt792x_dev *dev, struct sk_buff *skb)
25862306a36Sopenharmony_ci{
25962306a36Sopenharmony_ci	struct mt76_connac2_mcu_rxd *rxd;
26062306a36Sopenharmony_ci
26162306a36Sopenharmony_ci	rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
26262306a36Sopenharmony_ci	switch (rxd->eid) {
26362306a36Sopenharmony_ci	case MCU_EVENT_BSS_BEACON_LOSS:
26462306a36Sopenharmony_ci		mt7921_mcu_connection_loss_event(dev, skb);
26562306a36Sopenharmony_ci		break;
26662306a36Sopenharmony_ci	case MCU_EVENT_SCHED_SCAN_DONE:
26762306a36Sopenharmony_ci	case MCU_EVENT_SCAN_DONE:
26862306a36Sopenharmony_ci		mt7921_mcu_scan_event(dev, skb);
26962306a36Sopenharmony_ci		return;
27062306a36Sopenharmony_ci	case MCU_EVENT_DBG_MSG:
27162306a36Sopenharmony_ci		mt7921_mcu_debug_msg_event(dev, skb);
27262306a36Sopenharmony_ci		break;
27362306a36Sopenharmony_ci	case MCU_EVENT_COREDUMP:
27462306a36Sopenharmony_ci		dev->fw_assert = true;
27562306a36Sopenharmony_ci		mt76_connac_mcu_coredump_event(&dev->mt76, skb,
27662306a36Sopenharmony_ci					       &dev->coredump);
27762306a36Sopenharmony_ci		return;
27862306a36Sopenharmony_ci	case MCU_EVENT_LP_INFO:
27962306a36Sopenharmony_ci		mt7921_mcu_low_power_event(dev, skb);
28062306a36Sopenharmony_ci		break;
28162306a36Sopenharmony_ci	case MCU_EVENT_TX_DONE:
28262306a36Sopenharmony_ci		mt7921_mcu_tx_done_event(dev, skb);
28362306a36Sopenharmony_ci		break;
28462306a36Sopenharmony_ci	default:
28562306a36Sopenharmony_ci		break;
28662306a36Sopenharmony_ci	}
28762306a36Sopenharmony_ci	dev_kfree_skb(skb);
28862306a36Sopenharmony_ci}
28962306a36Sopenharmony_ci
29062306a36Sopenharmony_cistatic void
29162306a36Sopenharmony_cimt7921_mcu_uni_rx_unsolicited_event(struct mt792x_dev *dev,
29262306a36Sopenharmony_ci				    struct sk_buff *skb)
29362306a36Sopenharmony_ci{
29462306a36Sopenharmony_ci	struct mt76_connac2_mcu_rxd *rxd;
29562306a36Sopenharmony_ci
29662306a36Sopenharmony_ci	rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
29762306a36Sopenharmony_ci
29862306a36Sopenharmony_ci	switch (rxd->eid) {
29962306a36Sopenharmony_ci	case MCU_UNI_EVENT_ROC:
30062306a36Sopenharmony_ci		mt7921_mcu_uni_roc_event(dev, skb);
30162306a36Sopenharmony_ci		break;
30262306a36Sopenharmony_ci	default:
30362306a36Sopenharmony_ci		break;
30462306a36Sopenharmony_ci	}
30562306a36Sopenharmony_ci	dev_kfree_skb(skb);
30662306a36Sopenharmony_ci}
30762306a36Sopenharmony_ci
30862306a36Sopenharmony_civoid mt7921_mcu_rx_event(struct mt792x_dev *dev, struct sk_buff *skb)
30962306a36Sopenharmony_ci{
31062306a36Sopenharmony_ci	struct mt76_connac2_mcu_rxd *rxd;
31162306a36Sopenharmony_ci
31262306a36Sopenharmony_ci	if (skb_linearize(skb))
31362306a36Sopenharmony_ci		return;
31462306a36Sopenharmony_ci
31562306a36Sopenharmony_ci	rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
31662306a36Sopenharmony_ci
31762306a36Sopenharmony_ci	if (rxd->option & MCU_UNI_CMD_UNSOLICITED_EVENT) {
31862306a36Sopenharmony_ci		mt7921_mcu_uni_rx_unsolicited_event(dev, skb);
31962306a36Sopenharmony_ci		return;
32062306a36Sopenharmony_ci	}
32162306a36Sopenharmony_ci
32262306a36Sopenharmony_ci	if (rxd->eid == 0x6) {
32362306a36Sopenharmony_ci		mt76_mcu_rx_event(&dev->mt76, skb);
32462306a36Sopenharmony_ci		return;
32562306a36Sopenharmony_ci	}
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_ci	if (rxd->ext_eid == MCU_EXT_EVENT_RATE_REPORT ||
32862306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_BSS_BEACON_LOSS ||
32962306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_SCHED_SCAN_DONE ||
33062306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_SCAN_DONE ||
33162306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_TX_DONE ||
33262306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_DBG_MSG ||
33362306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_COREDUMP ||
33462306a36Sopenharmony_ci	    rxd->eid == MCU_EVENT_LP_INFO ||
33562306a36Sopenharmony_ci	    !rxd->seq)
33662306a36Sopenharmony_ci		mt7921_mcu_rx_unsolicited_event(dev, skb);
33762306a36Sopenharmony_ci	else
33862306a36Sopenharmony_ci		mt76_mcu_rx_event(&dev->mt76, skb);
33962306a36Sopenharmony_ci}
34062306a36Sopenharmony_ci
34162306a36Sopenharmony_ci/** starec & wtbl **/
34262306a36Sopenharmony_ciint mt7921_mcu_uni_tx_ba(struct mt792x_dev *dev,
34362306a36Sopenharmony_ci			 struct ieee80211_ampdu_params *params,
34462306a36Sopenharmony_ci			 bool enable)
34562306a36Sopenharmony_ci{
34662306a36Sopenharmony_ci	struct mt792x_sta *msta = (struct mt792x_sta *)params->sta->drv_priv;
34762306a36Sopenharmony_ci
34862306a36Sopenharmony_ci	if (enable && !params->amsdu)
34962306a36Sopenharmony_ci		msta->wcid.amsdu = false;
35062306a36Sopenharmony_ci
35162306a36Sopenharmony_ci	return mt76_connac_mcu_sta_ba(&dev->mt76, &msta->vif->mt76, params,
35262306a36Sopenharmony_ci				      MCU_UNI_CMD(STA_REC_UPDATE),
35362306a36Sopenharmony_ci				      enable, true);
35462306a36Sopenharmony_ci}
35562306a36Sopenharmony_ci
35662306a36Sopenharmony_ciint mt7921_mcu_uni_rx_ba(struct mt792x_dev *dev,
35762306a36Sopenharmony_ci			 struct ieee80211_ampdu_params *params,
35862306a36Sopenharmony_ci			 bool enable)
35962306a36Sopenharmony_ci{
36062306a36Sopenharmony_ci	struct mt792x_sta *msta = (struct mt792x_sta *)params->sta->drv_priv;
36162306a36Sopenharmony_ci
36262306a36Sopenharmony_ci	return mt76_connac_mcu_sta_ba(&dev->mt76, &msta->vif->mt76, params,
36362306a36Sopenharmony_ci				      MCU_UNI_CMD(STA_REC_UPDATE),
36462306a36Sopenharmony_ci				      enable, false);
36562306a36Sopenharmony_ci}
36662306a36Sopenharmony_ci
36762306a36Sopenharmony_cistatic int mt7921_load_clc(struct mt792x_dev *dev, const char *fw_name)
36862306a36Sopenharmony_ci{
36962306a36Sopenharmony_ci	const struct mt76_connac2_fw_trailer *hdr;
37062306a36Sopenharmony_ci	const struct mt76_connac2_fw_region *region;
37162306a36Sopenharmony_ci	const struct mt7921_clc *clc;
37262306a36Sopenharmony_ci	struct mt76_dev *mdev = &dev->mt76;
37362306a36Sopenharmony_ci	struct mt792x_phy *phy = &dev->phy;
37462306a36Sopenharmony_ci	const struct firmware *fw;
37562306a36Sopenharmony_ci	int ret, i, len, offset = 0;
37662306a36Sopenharmony_ci	u8 *clc_base = NULL, hw_encap = 0;
37762306a36Sopenharmony_ci
37862306a36Sopenharmony_ci	if (mt7921_disable_clc ||
37962306a36Sopenharmony_ci	    mt76_is_usb(&dev->mt76))
38062306a36Sopenharmony_ci		return 0;
38162306a36Sopenharmony_ci
38262306a36Sopenharmony_ci	if (mt76_is_mmio(&dev->mt76)) {
38362306a36Sopenharmony_ci		ret = mt7921_mcu_read_eeprom(dev, MT_EE_HW_TYPE, &hw_encap);
38462306a36Sopenharmony_ci		if (ret)
38562306a36Sopenharmony_ci			return ret;
38662306a36Sopenharmony_ci		hw_encap = u8_get_bits(hw_encap, MT_EE_HW_TYPE_ENCAP);
38762306a36Sopenharmony_ci	}
38862306a36Sopenharmony_ci
38962306a36Sopenharmony_ci	ret = request_firmware(&fw, fw_name, mdev->dev);
39062306a36Sopenharmony_ci	if (ret)
39162306a36Sopenharmony_ci		return ret;
39262306a36Sopenharmony_ci
39362306a36Sopenharmony_ci	if (!fw || !fw->data || fw->size < sizeof(*hdr)) {
39462306a36Sopenharmony_ci		dev_err(mdev->dev, "Invalid firmware\n");
39562306a36Sopenharmony_ci		ret = -EINVAL;
39662306a36Sopenharmony_ci		goto out;
39762306a36Sopenharmony_ci	}
39862306a36Sopenharmony_ci
39962306a36Sopenharmony_ci	hdr = (const void *)(fw->data + fw->size - sizeof(*hdr));
40062306a36Sopenharmony_ci	for (i = 0; i < hdr->n_region; i++) {
40162306a36Sopenharmony_ci		region = (const void *)((const u8 *)hdr -
40262306a36Sopenharmony_ci					(hdr->n_region - i) * sizeof(*region));
40362306a36Sopenharmony_ci		len = le32_to_cpu(region->len);
40462306a36Sopenharmony_ci
40562306a36Sopenharmony_ci		/* check if we have valid buffer size */
40662306a36Sopenharmony_ci		if (offset + len > fw->size) {
40762306a36Sopenharmony_ci			dev_err(mdev->dev, "Invalid firmware region\n");
40862306a36Sopenharmony_ci			ret = -EINVAL;
40962306a36Sopenharmony_ci			goto out;
41062306a36Sopenharmony_ci		}
41162306a36Sopenharmony_ci
41262306a36Sopenharmony_ci		if ((region->feature_set & FW_FEATURE_NON_DL) &&
41362306a36Sopenharmony_ci		    region->type == FW_TYPE_CLC) {
41462306a36Sopenharmony_ci			clc_base = (u8 *)(fw->data + offset);
41562306a36Sopenharmony_ci			break;
41662306a36Sopenharmony_ci		}
41762306a36Sopenharmony_ci		offset += len;
41862306a36Sopenharmony_ci	}
41962306a36Sopenharmony_ci
42062306a36Sopenharmony_ci	if (!clc_base)
42162306a36Sopenharmony_ci		goto out;
42262306a36Sopenharmony_ci
42362306a36Sopenharmony_ci	for (offset = 0; offset < len; offset += le32_to_cpu(clc->len)) {
42462306a36Sopenharmony_ci		clc = (const struct mt7921_clc *)(clc_base + offset);
42562306a36Sopenharmony_ci
42662306a36Sopenharmony_ci		/* do not init buf again if chip reset triggered */
42762306a36Sopenharmony_ci		if (phy->clc[clc->idx])
42862306a36Sopenharmony_ci			continue;
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_ci		/* header content sanity */
43162306a36Sopenharmony_ci		if (clc->idx == MT7921_CLC_POWER &&
43262306a36Sopenharmony_ci		    u8_get_bits(clc->type, MT_EE_HW_TYPE_ENCAP) != hw_encap)
43362306a36Sopenharmony_ci			continue;
43462306a36Sopenharmony_ci
43562306a36Sopenharmony_ci		phy->clc[clc->idx] = devm_kmemdup(mdev->dev, clc,
43662306a36Sopenharmony_ci						  le32_to_cpu(clc->len),
43762306a36Sopenharmony_ci						  GFP_KERNEL);
43862306a36Sopenharmony_ci
43962306a36Sopenharmony_ci		if (!phy->clc[clc->idx]) {
44062306a36Sopenharmony_ci			ret = -ENOMEM;
44162306a36Sopenharmony_ci			goto out;
44262306a36Sopenharmony_ci		}
44362306a36Sopenharmony_ci	}
44462306a36Sopenharmony_ci	ret = mt7921_mcu_set_clc(dev, "00", ENVIRON_INDOOR);
44562306a36Sopenharmony_ciout:
44662306a36Sopenharmony_ci	release_firmware(fw);
44762306a36Sopenharmony_ci
44862306a36Sopenharmony_ci	return ret;
44962306a36Sopenharmony_ci}
45062306a36Sopenharmony_ci
45162306a36Sopenharmony_ciint mt7921_mcu_fw_log_2_host(struct mt792x_dev *dev, u8 ctrl)
45262306a36Sopenharmony_ci{
45362306a36Sopenharmony_ci	struct {
45462306a36Sopenharmony_ci		u8 ctrl_val;
45562306a36Sopenharmony_ci		u8 pad[3];
45662306a36Sopenharmony_ci	} data = {
45762306a36Sopenharmony_ci		.ctrl_val = ctrl
45862306a36Sopenharmony_ci	};
45962306a36Sopenharmony_ci
46062306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(FWLOG_2_HOST),
46162306a36Sopenharmony_ci				 &data, sizeof(data), false);
46262306a36Sopenharmony_ci}
46362306a36Sopenharmony_ci
46462306a36Sopenharmony_ciint mt7921_run_firmware(struct mt792x_dev *dev)
46562306a36Sopenharmony_ci{
46662306a36Sopenharmony_ci	int err;
46762306a36Sopenharmony_ci
46862306a36Sopenharmony_ci	err = mt792x_load_firmware(dev);
46962306a36Sopenharmony_ci	if (err)
47062306a36Sopenharmony_ci		return err;
47162306a36Sopenharmony_ci
47262306a36Sopenharmony_ci	err = mt76_connac_mcu_get_nic_capability(&dev->mphy);
47362306a36Sopenharmony_ci	if (err)
47462306a36Sopenharmony_ci		return err;
47562306a36Sopenharmony_ci
47662306a36Sopenharmony_ci	set_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
47762306a36Sopenharmony_ci	err = mt7921_load_clc(dev, mt792x_ram_name(dev));
47862306a36Sopenharmony_ci	if (err)
47962306a36Sopenharmony_ci		return err;
48062306a36Sopenharmony_ci
48162306a36Sopenharmony_ci	return mt7921_mcu_fw_log_2_host(dev, 1);
48262306a36Sopenharmony_ci}
48362306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mt7921_run_firmware);
48462306a36Sopenharmony_ci
48562306a36Sopenharmony_ciint mt7921_mcu_set_tx(struct mt792x_dev *dev, struct ieee80211_vif *vif)
48662306a36Sopenharmony_ci{
48762306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
48862306a36Sopenharmony_ci	struct edca {
48962306a36Sopenharmony_ci		__le16 cw_min;
49062306a36Sopenharmony_ci		__le16 cw_max;
49162306a36Sopenharmony_ci		__le16 txop;
49262306a36Sopenharmony_ci		__le16 aifs;
49362306a36Sopenharmony_ci		u8 guardtime;
49462306a36Sopenharmony_ci		u8 acm;
49562306a36Sopenharmony_ci	} __packed;
49662306a36Sopenharmony_ci	struct mt7921_mcu_tx {
49762306a36Sopenharmony_ci		struct edca edca[IEEE80211_NUM_ACS];
49862306a36Sopenharmony_ci		u8 bss_idx;
49962306a36Sopenharmony_ci		u8 qos;
50062306a36Sopenharmony_ci		u8 wmm_idx;
50162306a36Sopenharmony_ci		u8 pad;
50262306a36Sopenharmony_ci	} __packed req = {
50362306a36Sopenharmony_ci		.bss_idx = mvif->mt76.idx,
50462306a36Sopenharmony_ci		.qos = vif->bss_conf.qos,
50562306a36Sopenharmony_ci		.wmm_idx = mvif->mt76.wmm_idx,
50662306a36Sopenharmony_ci	};
50762306a36Sopenharmony_ci	struct mu_edca {
50862306a36Sopenharmony_ci		u8 cw_min;
50962306a36Sopenharmony_ci		u8 cw_max;
51062306a36Sopenharmony_ci		u8 aifsn;
51162306a36Sopenharmony_ci		u8 acm;
51262306a36Sopenharmony_ci		u8 timer;
51362306a36Sopenharmony_ci		u8 padding[3];
51462306a36Sopenharmony_ci	};
51562306a36Sopenharmony_ci	struct mt7921_mcu_mu_tx {
51662306a36Sopenharmony_ci		u8 ver;
51762306a36Sopenharmony_ci		u8 pad0;
51862306a36Sopenharmony_ci		__le16 len;
51962306a36Sopenharmony_ci		u8 bss_idx;
52062306a36Sopenharmony_ci		u8 qos;
52162306a36Sopenharmony_ci		u8 wmm_idx;
52262306a36Sopenharmony_ci		u8 pad1;
52362306a36Sopenharmony_ci		struct mu_edca edca[IEEE80211_NUM_ACS];
52462306a36Sopenharmony_ci		u8 pad3[32];
52562306a36Sopenharmony_ci	} __packed req_mu = {
52662306a36Sopenharmony_ci		.bss_idx = mvif->mt76.idx,
52762306a36Sopenharmony_ci		.qos = vif->bss_conf.qos,
52862306a36Sopenharmony_ci		.wmm_idx = mvif->mt76.wmm_idx,
52962306a36Sopenharmony_ci	};
53062306a36Sopenharmony_ci	static const int to_aci[] = { 1, 0, 2, 3 };
53162306a36Sopenharmony_ci	int ac, ret;
53262306a36Sopenharmony_ci
53362306a36Sopenharmony_ci	for (ac = 0; ac < IEEE80211_NUM_ACS; ac++) {
53462306a36Sopenharmony_ci		struct ieee80211_tx_queue_params *q = &mvif->queue_params[ac];
53562306a36Sopenharmony_ci		struct edca *e = &req.edca[to_aci[ac]];
53662306a36Sopenharmony_ci
53762306a36Sopenharmony_ci		e->aifs = cpu_to_le16(q->aifs);
53862306a36Sopenharmony_ci		e->txop = cpu_to_le16(q->txop);
53962306a36Sopenharmony_ci
54062306a36Sopenharmony_ci		if (q->cw_min)
54162306a36Sopenharmony_ci			e->cw_min = cpu_to_le16(q->cw_min);
54262306a36Sopenharmony_ci		else
54362306a36Sopenharmony_ci			e->cw_min = cpu_to_le16(5);
54462306a36Sopenharmony_ci
54562306a36Sopenharmony_ci		if (q->cw_max)
54662306a36Sopenharmony_ci			e->cw_max = cpu_to_le16(q->cw_max);
54762306a36Sopenharmony_ci		else
54862306a36Sopenharmony_ci			e->cw_max = cpu_to_le16(10);
54962306a36Sopenharmony_ci	}
55062306a36Sopenharmony_ci
55162306a36Sopenharmony_ci	ret = mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_EDCA_PARMS), &req,
55262306a36Sopenharmony_ci				sizeof(req), false);
55362306a36Sopenharmony_ci	if (ret)
55462306a36Sopenharmony_ci		return ret;
55562306a36Sopenharmony_ci
55662306a36Sopenharmony_ci	if (!vif->bss_conf.he_support)
55762306a36Sopenharmony_ci		return 0;
55862306a36Sopenharmony_ci
55962306a36Sopenharmony_ci	for (ac = 0; ac < IEEE80211_NUM_ACS; ac++) {
56062306a36Sopenharmony_ci		struct ieee80211_he_mu_edca_param_ac_rec *q;
56162306a36Sopenharmony_ci		struct mu_edca *e;
56262306a36Sopenharmony_ci
56362306a36Sopenharmony_ci		if (!mvif->queue_params[ac].mu_edca)
56462306a36Sopenharmony_ci			break;
56562306a36Sopenharmony_ci
56662306a36Sopenharmony_ci		q = &mvif->queue_params[ac].mu_edca_param_rec;
56762306a36Sopenharmony_ci		e = &(req_mu.edca[to_aci[ac]]);
56862306a36Sopenharmony_ci
56962306a36Sopenharmony_ci		e->cw_min = q->ecw_min_max & 0xf;
57062306a36Sopenharmony_ci		e->cw_max = (q->ecw_min_max & 0xf0) >> 4;
57162306a36Sopenharmony_ci		e->aifsn = q->aifsn;
57262306a36Sopenharmony_ci		e->timer = q->mu_edca_timer;
57362306a36Sopenharmony_ci	}
57462306a36Sopenharmony_ci
57562306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_MU_EDCA_PARMS),
57662306a36Sopenharmony_ci				 &req_mu, sizeof(req_mu), false);
57762306a36Sopenharmony_ci}
57862306a36Sopenharmony_ci
57962306a36Sopenharmony_ciint mt7921_mcu_set_roc(struct mt792x_phy *phy, struct mt792x_vif *vif,
58062306a36Sopenharmony_ci		       struct ieee80211_channel *chan, int duration,
58162306a36Sopenharmony_ci		       enum mt7921_roc_req type, u8 token_id)
58262306a36Sopenharmony_ci{
58362306a36Sopenharmony_ci	int center_ch = ieee80211_frequency_to_channel(chan->center_freq);
58462306a36Sopenharmony_ci	struct mt792x_dev *dev = phy->dev;
58562306a36Sopenharmony_ci	struct {
58662306a36Sopenharmony_ci		struct {
58762306a36Sopenharmony_ci			u8 rsv[4];
58862306a36Sopenharmony_ci		} __packed hdr;
58962306a36Sopenharmony_ci		struct roc_acquire_tlv {
59062306a36Sopenharmony_ci			__le16 tag;
59162306a36Sopenharmony_ci			__le16 len;
59262306a36Sopenharmony_ci			u8 bss_idx;
59362306a36Sopenharmony_ci			u8 tokenid;
59462306a36Sopenharmony_ci			u8 control_channel;
59562306a36Sopenharmony_ci			u8 sco;
59662306a36Sopenharmony_ci			u8 band;
59762306a36Sopenharmony_ci			u8 bw;
59862306a36Sopenharmony_ci			u8 center_chan;
59962306a36Sopenharmony_ci			u8 center_chan2;
60062306a36Sopenharmony_ci			u8 bw_from_ap;
60162306a36Sopenharmony_ci			u8 center_chan_from_ap;
60262306a36Sopenharmony_ci			u8 center_chan2_from_ap;
60362306a36Sopenharmony_ci			u8 reqtype;
60462306a36Sopenharmony_ci			__le32 maxinterval;
60562306a36Sopenharmony_ci			u8 dbdcband;
60662306a36Sopenharmony_ci			u8 rsv[3];
60762306a36Sopenharmony_ci		} __packed roc;
60862306a36Sopenharmony_ci	} __packed req = {
60962306a36Sopenharmony_ci		.roc = {
61062306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_ROC_ACQUIRE),
61162306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct roc_acquire_tlv)),
61262306a36Sopenharmony_ci			.tokenid = token_id,
61362306a36Sopenharmony_ci			.reqtype = type,
61462306a36Sopenharmony_ci			.maxinterval = cpu_to_le32(duration),
61562306a36Sopenharmony_ci			.bss_idx = vif->mt76.idx,
61662306a36Sopenharmony_ci			.control_channel = chan->hw_value,
61762306a36Sopenharmony_ci			.bw = CMD_CBW_20MHZ,
61862306a36Sopenharmony_ci			.bw_from_ap = CMD_CBW_20MHZ,
61962306a36Sopenharmony_ci			.center_chan = center_ch,
62062306a36Sopenharmony_ci			.center_chan_from_ap = center_ch,
62162306a36Sopenharmony_ci			.dbdcband = 0xff, /* auto */
62262306a36Sopenharmony_ci		},
62362306a36Sopenharmony_ci	};
62462306a36Sopenharmony_ci
62562306a36Sopenharmony_ci	if (chan->hw_value < center_ch)
62662306a36Sopenharmony_ci		req.roc.sco = 1; /* SCA */
62762306a36Sopenharmony_ci	else if (chan->hw_value > center_ch)
62862306a36Sopenharmony_ci		req.roc.sco = 3; /* SCB */
62962306a36Sopenharmony_ci
63062306a36Sopenharmony_ci	switch (chan->band) {
63162306a36Sopenharmony_ci	case NL80211_BAND_6GHZ:
63262306a36Sopenharmony_ci		req.roc.band = 3;
63362306a36Sopenharmony_ci		break;
63462306a36Sopenharmony_ci	case NL80211_BAND_5GHZ:
63562306a36Sopenharmony_ci		req.roc.band = 2;
63662306a36Sopenharmony_ci		break;
63762306a36Sopenharmony_ci	default:
63862306a36Sopenharmony_ci		req.roc.band = 1;
63962306a36Sopenharmony_ci		break;
64062306a36Sopenharmony_ci	}
64162306a36Sopenharmony_ci
64262306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(ROC),
64362306a36Sopenharmony_ci				 &req, sizeof(req), false);
64462306a36Sopenharmony_ci}
64562306a36Sopenharmony_ci
64662306a36Sopenharmony_ciint mt7921_mcu_abort_roc(struct mt792x_phy *phy, struct mt792x_vif *vif,
64762306a36Sopenharmony_ci			 u8 token_id)
64862306a36Sopenharmony_ci{
64962306a36Sopenharmony_ci	struct mt792x_dev *dev = phy->dev;
65062306a36Sopenharmony_ci	struct {
65162306a36Sopenharmony_ci		struct {
65262306a36Sopenharmony_ci			u8 rsv[4];
65362306a36Sopenharmony_ci		} __packed hdr;
65462306a36Sopenharmony_ci		struct roc_abort_tlv {
65562306a36Sopenharmony_ci			__le16 tag;
65662306a36Sopenharmony_ci			__le16 len;
65762306a36Sopenharmony_ci			u8 bss_idx;
65862306a36Sopenharmony_ci			u8 tokenid;
65962306a36Sopenharmony_ci			u8 dbdcband;
66062306a36Sopenharmony_ci			u8 rsv[5];
66162306a36Sopenharmony_ci		} __packed abort;
66262306a36Sopenharmony_ci	} __packed req = {
66362306a36Sopenharmony_ci		.abort = {
66462306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_ROC_ABORT),
66562306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct roc_abort_tlv)),
66662306a36Sopenharmony_ci			.tokenid = token_id,
66762306a36Sopenharmony_ci			.bss_idx = vif->mt76.idx,
66862306a36Sopenharmony_ci			.dbdcband = 0xff, /* auto*/
66962306a36Sopenharmony_ci		},
67062306a36Sopenharmony_ci	};
67162306a36Sopenharmony_ci
67262306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(ROC),
67362306a36Sopenharmony_ci				 &req, sizeof(req), false);
67462306a36Sopenharmony_ci}
67562306a36Sopenharmony_ci
67662306a36Sopenharmony_ciint mt7921_mcu_set_chan_info(struct mt792x_phy *phy, int cmd)
67762306a36Sopenharmony_ci{
67862306a36Sopenharmony_ci	struct mt792x_dev *dev = phy->dev;
67962306a36Sopenharmony_ci	struct cfg80211_chan_def *chandef = &phy->mt76->chandef;
68062306a36Sopenharmony_ci	int freq1 = chandef->center_freq1;
68162306a36Sopenharmony_ci	struct {
68262306a36Sopenharmony_ci		u8 control_ch;
68362306a36Sopenharmony_ci		u8 center_ch;
68462306a36Sopenharmony_ci		u8 bw;
68562306a36Sopenharmony_ci		u8 tx_streams_num;
68662306a36Sopenharmony_ci		u8 rx_streams;	/* mask or num */
68762306a36Sopenharmony_ci		u8 switch_reason;
68862306a36Sopenharmony_ci		u8 band_idx;
68962306a36Sopenharmony_ci		u8 center_ch2;	/* for 80+80 only */
69062306a36Sopenharmony_ci		__le16 cac_case;
69162306a36Sopenharmony_ci		u8 channel_band;
69262306a36Sopenharmony_ci		u8 rsv0;
69362306a36Sopenharmony_ci		__le32 outband_freq;
69462306a36Sopenharmony_ci		u8 txpower_drop;
69562306a36Sopenharmony_ci		u8 ap_bw;
69662306a36Sopenharmony_ci		u8 ap_center_ch;
69762306a36Sopenharmony_ci		u8 rsv1[57];
69862306a36Sopenharmony_ci	} __packed req = {
69962306a36Sopenharmony_ci		.control_ch = chandef->chan->hw_value,
70062306a36Sopenharmony_ci		.center_ch = ieee80211_frequency_to_channel(freq1),
70162306a36Sopenharmony_ci		.bw = mt76_connac_chan_bw(chandef),
70262306a36Sopenharmony_ci		.tx_streams_num = hweight8(phy->mt76->antenna_mask),
70362306a36Sopenharmony_ci		.rx_streams = phy->mt76->antenna_mask,
70462306a36Sopenharmony_ci		.band_idx = phy != &dev->phy,
70562306a36Sopenharmony_ci	};
70662306a36Sopenharmony_ci
70762306a36Sopenharmony_ci	if (chandef->chan->band == NL80211_BAND_6GHZ)
70862306a36Sopenharmony_ci		req.channel_band = 2;
70962306a36Sopenharmony_ci	else
71062306a36Sopenharmony_ci		req.channel_band = chandef->chan->band;
71162306a36Sopenharmony_ci
71262306a36Sopenharmony_ci	if (cmd == MCU_EXT_CMD(SET_RX_PATH) ||
71362306a36Sopenharmony_ci	    dev->mt76.hw->conf.flags & IEEE80211_CONF_MONITOR)
71462306a36Sopenharmony_ci		req.switch_reason = CH_SWITCH_NORMAL;
71562306a36Sopenharmony_ci	else if (dev->mt76.hw->conf.flags & IEEE80211_CONF_OFFCHANNEL)
71662306a36Sopenharmony_ci		req.switch_reason = CH_SWITCH_SCAN_BYPASS_DPD;
71762306a36Sopenharmony_ci	else if (!cfg80211_reg_can_beacon(dev->mt76.hw->wiphy, chandef,
71862306a36Sopenharmony_ci					  NL80211_IFTYPE_AP))
71962306a36Sopenharmony_ci		req.switch_reason = CH_SWITCH_DFS;
72062306a36Sopenharmony_ci	else
72162306a36Sopenharmony_ci		req.switch_reason = CH_SWITCH_NORMAL;
72262306a36Sopenharmony_ci
72362306a36Sopenharmony_ci	if (cmd == MCU_EXT_CMD(CHANNEL_SWITCH))
72462306a36Sopenharmony_ci		req.rx_streams = hweight8(req.rx_streams);
72562306a36Sopenharmony_ci
72662306a36Sopenharmony_ci	if (chandef->width == NL80211_CHAN_WIDTH_80P80) {
72762306a36Sopenharmony_ci		int freq2 = chandef->center_freq2;
72862306a36Sopenharmony_ci
72962306a36Sopenharmony_ci		req.center_ch2 = ieee80211_frequency_to_channel(freq2);
73062306a36Sopenharmony_ci	}
73162306a36Sopenharmony_ci
73262306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, cmd, &req, sizeof(req), true);
73362306a36Sopenharmony_ci}
73462306a36Sopenharmony_ci
73562306a36Sopenharmony_ciint mt7921_mcu_set_eeprom(struct mt792x_dev *dev)
73662306a36Sopenharmony_ci{
73762306a36Sopenharmony_ci	struct req_hdr {
73862306a36Sopenharmony_ci		u8 buffer_mode;
73962306a36Sopenharmony_ci		u8 format;
74062306a36Sopenharmony_ci		__le16 len;
74162306a36Sopenharmony_ci	} __packed req = {
74262306a36Sopenharmony_ci		.buffer_mode = EE_MODE_EFUSE,
74362306a36Sopenharmony_ci		.format = EE_FORMAT_WHOLE,
74462306a36Sopenharmony_ci	};
74562306a36Sopenharmony_ci
74662306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_EXT_CMD(EFUSE_BUFFER_MODE),
74762306a36Sopenharmony_ci				 &req, sizeof(req), true);
74862306a36Sopenharmony_ci}
74962306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mt7921_mcu_set_eeprom);
75062306a36Sopenharmony_ci
75162306a36Sopenharmony_ciint mt7921_mcu_uni_bss_ps(struct mt792x_dev *dev, struct ieee80211_vif *vif)
75262306a36Sopenharmony_ci{
75362306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
75462306a36Sopenharmony_ci	struct {
75562306a36Sopenharmony_ci		struct {
75662306a36Sopenharmony_ci			u8 bss_idx;
75762306a36Sopenharmony_ci			u8 pad[3];
75862306a36Sopenharmony_ci		} __packed hdr;
75962306a36Sopenharmony_ci		struct ps_tlv {
76062306a36Sopenharmony_ci			__le16 tag;
76162306a36Sopenharmony_ci			__le16 len;
76262306a36Sopenharmony_ci			u8 ps_state; /* 0: device awake
76362306a36Sopenharmony_ci				      * 1: static power save
76462306a36Sopenharmony_ci				      * 2: dynamic power saving
76562306a36Sopenharmony_ci				      * 3: enter TWT power saving
76662306a36Sopenharmony_ci				      * 4: leave TWT power saving
76762306a36Sopenharmony_ci				      */
76862306a36Sopenharmony_ci			u8 pad[3];
76962306a36Sopenharmony_ci		} __packed ps;
77062306a36Sopenharmony_ci	} __packed ps_req = {
77162306a36Sopenharmony_ci		.hdr = {
77262306a36Sopenharmony_ci			.bss_idx = mvif->mt76.idx,
77362306a36Sopenharmony_ci		},
77462306a36Sopenharmony_ci		.ps = {
77562306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_BSS_INFO_PS),
77662306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct ps_tlv)),
77762306a36Sopenharmony_ci			.ps_state = vif->cfg.ps ? 2 : 0,
77862306a36Sopenharmony_ci		},
77962306a36Sopenharmony_ci	};
78062306a36Sopenharmony_ci
78162306a36Sopenharmony_ci	if (vif->type != NL80211_IFTYPE_STATION)
78262306a36Sopenharmony_ci		return -EOPNOTSUPP;
78362306a36Sopenharmony_ci
78462306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
78562306a36Sopenharmony_ci				 &ps_req, sizeof(ps_req), true);
78662306a36Sopenharmony_ci}
78762306a36Sopenharmony_ci
78862306a36Sopenharmony_cistatic int
78962306a36Sopenharmony_cimt7921_mcu_uni_bss_bcnft(struct mt792x_dev *dev, struct ieee80211_vif *vif,
79062306a36Sopenharmony_ci			 bool enable)
79162306a36Sopenharmony_ci{
79262306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
79362306a36Sopenharmony_ci	struct {
79462306a36Sopenharmony_ci		struct {
79562306a36Sopenharmony_ci			u8 bss_idx;
79662306a36Sopenharmony_ci			u8 pad[3];
79762306a36Sopenharmony_ci		} __packed hdr;
79862306a36Sopenharmony_ci		struct bcnft_tlv {
79962306a36Sopenharmony_ci			__le16 tag;
80062306a36Sopenharmony_ci			__le16 len;
80162306a36Sopenharmony_ci			__le16 bcn_interval;
80262306a36Sopenharmony_ci			u8 dtim_period;
80362306a36Sopenharmony_ci			u8 pad;
80462306a36Sopenharmony_ci		} __packed bcnft;
80562306a36Sopenharmony_ci	} __packed bcnft_req = {
80662306a36Sopenharmony_ci		.hdr = {
80762306a36Sopenharmony_ci			.bss_idx = mvif->mt76.idx,
80862306a36Sopenharmony_ci		},
80962306a36Sopenharmony_ci		.bcnft = {
81062306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_BSS_INFO_BCNFT),
81162306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct bcnft_tlv)),
81262306a36Sopenharmony_ci			.bcn_interval = cpu_to_le16(vif->bss_conf.beacon_int),
81362306a36Sopenharmony_ci			.dtim_period = vif->bss_conf.dtim_period,
81462306a36Sopenharmony_ci		},
81562306a36Sopenharmony_ci	};
81662306a36Sopenharmony_ci
81762306a36Sopenharmony_ci	if (vif->type != NL80211_IFTYPE_STATION)
81862306a36Sopenharmony_ci		return 0;
81962306a36Sopenharmony_ci
82062306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
82162306a36Sopenharmony_ci				 &bcnft_req, sizeof(bcnft_req), true);
82262306a36Sopenharmony_ci}
82362306a36Sopenharmony_ci
82462306a36Sopenharmony_ciint
82562306a36Sopenharmony_cimt7921_mcu_set_bss_pm(struct mt792x_dev *dev, struct ieee80211_vif *vif,
82662306a36Sopenharmony_ci		      bool enable)
82762306a36Sopenharmony_ci{
82862306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
82962306a36Sopenharmony_ci	struct {
83062306a36Sopenharmony_ci		u8 bss_idx;
83162306a36Sopenharmony_ci		u8 dtim_period;
83262306a36Sopenharmony_ci		__le16 aid;
83362306a36Sopenharmony_ci		__le16 bcn_interval;
83462306a36Sopenharmony_ci		__le16 atim_window;
83562306a36Sopenharmony_ci		u8 uapsd;
83662306a36Sopenharmony_ci		u8 bmc_delivered_ac;
83762306a36Sopenharmony_ci		u8 bmc_triggered_ac;
83862306a36Sopenharmony_ci		u8 pad;
83962306a36Sopenharmony_ci	} req = {
84062306a36Sopenharmony_ci		.bss_idx = mvif->mt76.idx,
84162306a36Sopenharmony_ci		.aid = cpu_to_le16(vif->cfg.aid),
84262306a36Sopenharmony_ci		.dtim_period = vif->bss_conf.dtim_period,
84362306a36Sopenharmony_ci		.bcn_interval = cpu_to_le16(vif->bss_conf.beacon_int),
84462306a36Sopenharmony_ci	};
84562306a36Sopenharmony_ci	struct {
84662306a36Sopenharmony_ci		u8 bss_idx;
84762306a36Sopenharmony_ci		u8 pad[3];
84862306a36Sopenharmony_ci	} req_hdr = {
84962306a36Sopenharmony_ci		.bss_idx = mvif->mt76.idx,
85062306a36Sopenharmony_ci	};
85162306a36Sopenharmony_ci	int err;
85262306a36Sopenharmony_ci
85362306a36Sopenharmony_ci	err = mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_BSS_ABORT),
85462306a36Sopenharmony_ci				&req_hdr, sizeof(req_hdr), false);
85562306a36Sopenharmony_ci	if (err < 0 || !enable)
85662306a36Sopenharmony_ci		return err;
85762306a36Sopenharmony_ci
85862306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_BSS_CONNECTED),
85962306a36Sopenharmony_ci				 &req, sizeof(req), false);
86062306a36Sopenharmony_ci}
86162306a36Sopenharmony_ci
86262306a36Sopenharmony_ciint mt7921_mcu_sta_update(struct mt792x_dev *dev, struct ieee80211_sta *sta,
86362306a36Sopenharmony_ci			  struct ieee80211_vif *vif, bool enable,
86462306a36Sopenharmony_ci			  enum mt76_sta_info_state state)
86562306a36Sopenharmony_ci{
86662306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
86762306a36Sopenharmony_ci	int rssi = -ewma_rssi_read(&mvif->rssi);
86862306a36Sopenharmony_ci	struct mt76_sta_cmd_info info = {
86962306a36Sopenharmony_ci		.sta = sta,
87062306a36Sopenharmony_ci		.vif = vif,
87162306a36Sopenharmony_ci		.enable = enable,
87262306a36Sopenharmony_ci		.cmd = MCU_UNI_CMD(STA_REC_UPDATE),
87362306a36Sopenharmony_ci		.state = state,
87462306a36Sopenharmony_ci		.offload_fw = true,
87562306a36Sopenharmony_ci		.rcpi = to_rcpi(rssi),
87662306a36Sopenharmony_ci	};
87762306a36Sopenharmony_ci	struct mt792x_sta *msta;
87862306a36Sopenharmony_ci
87962306a36Sopenharmony_ci	msta = sta ? (struct mt792x_sta *)sta->drv_priv : NULL;
88062306a36Sopenharmony_ci	info.wcid = msta ? &msta->wcid : &mvif->sta.wcid;
88162306a36Sopenharmony_ci	info.newly = msta ? state != MT76_STA_INFO_STATE_ASSOC : true;
88262306a36Sopenharmony_ci
88362306a36Sopenharmony_ci	return mt76_connac_mcu_sta_cmd(&dev->mphy, &info);
88462306a36Sopenharmony_ci}
88562306a36Sopenharmony_ci
88662306a36Sopenharmony_ciint mt7921_mcu_set_beacon_filter(struct mt792x_dev *dev,
88762306a36Sopenharmony_ci				 struct ieee80211_vif *vif,
88862306a36Sopenharmony_ci				 bool enable)
88962306a36Sopenharmony_ci{
89062306a36Sopenharmony_ci#define MT7921_FIF_BIT_CLR		BIT(1)
89162306a36Sopenharmony_ci#define MT7921_FIF_BIT_SET		BIT(0)
89262306a36Sopenharmony_ci	int err;
89362306a36Sopenharmony_ci
89462306a36Sopenharmony_ci	if (enable) {
89562306a36Sopenharmony_ci		err = mt7921_mcu_uni_bss_bcnft(dev, vif, true);
89662306a36Sopenharmony_ci		if (err)
89762306a36Sopenharmony_ci			return err;
89862306a36Sopenharmony_ci
89962306a36Sopenharmony_ci		err = mt7921_mcu_set_rxfilter(dev, 0,
90062306a36Sopenharmony_ci					      MT7921_FIF_BIT_SET,
90162306a36Sopenharmony_ci					      MT_WF_RFCR_DROP_OTHER_BEACON);
90262306a36Sopenharmony_ci		if (err)
90362306a36Sopenharmony_ci			return err;
90462306a36Sopenharmony_ci
90562306a36Sopenharmony_ci		return 0;
90662306a36Sopenharmony_ci	}
90762306a36Sopenharmony_ci
90862306a36Sopenharmony_ci	err = mt7921_mcu_set_bss_pm(dev, vif, false);
90962306a36Sopenharmony_ci	if (err)
91062306a36Sopenharmony_ci		return err;
91162306a36Sopenharmony_ci
91262306a36Sopenharmony_ci	err = mt7921_mcu_set_rxfilter(dev, 0,
91362306a36Sopenharmony_ci				      MT7921_FIF_BIT_CLR,
91462306a36Sopenharmony_ci				      MT_WF_RFCR_DROP_OTHER_BEACON);
91562306a36Sopenharmony_ci	if (err)
91662306a36Sopenharmony_ci		return err;
91762306a36Sopenharmony_ci
91862306a36Sopenharmony_ci	return 0;
91962306a36Sopenharmony_ci}
92062306a36Sopenharmony_ci
92162306a36Sopenharmony_ciint mt7921_get_txpwr_info(struct mt792x_dev *dev, struct mt7921_txpwr *txpwr)
92262306a36Sopenharmony_ci{
92362306a36Sopenharmony_ci	struct mt7921_txpwr_event *event;
92462306a36Sopenharmony_ci	struct mt7921_txpwr_req req = {
92562306a36Sopenharmony_ci		.dbdc_idx = 0,
92662306a36Sopenharmony_ci	};
92762306a36Sopenharmony_ci	struct sk_buff *skb;
92862306a36Sopenharmony_ci	int ret;
92962306a36Sopenharmony_ci
93062306a36Sopenharmony_ci	ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_CE_CMD(GET_TXPWR),
93162306a36Sopenharmony_ci					&req, sizeof(req), true, &skb);
93262306a36Sopenharmony_ci	if (ret)
93362306a36Sopenharmony_ci		return ret;
93462306a36Sopenharmony_ci
93562306a36Sopenharmony_ci	event = (struct mt7921_txpwr_event *)skb->data;
93662306a36Sopenharmony_ci	WARN_ON(skb->len != le16_to_cpu(event->len));
93762306a36Sopenharmony_ci	memcpy(txpwr, &event->txpwr, sizeof(event->txpwr));
93862306a36Sopenharmony_ci
93962306a36Sopenharmony_ci	dev_kfree_skb(skb);
94062306a36Sopenharmony_ci
94162306a36Sopenharmony_ci	return 0;
94262306a36Sopenharmony_ci}
94362306a36Sopenharmony_ci
94462306a36Sopenharmony_ciint mt7921_mcu_set_sniffer(struct mt792x_dev *dev, struct ieee80211_vif *vif,
94562306a36Sopenharmony_ci			   bool enable)
94662306a36Sopenharmony_ci{
94762306a36Sopenharmony_ci	struct mt76_vif *mvif = (struct mt76_vif *)vif->drv_priv;
94862306a36Sopenharmony_ci	struct {
94962306a36Sopenharmony_ci		struct {
95062306a36Sopenharmony_ci			u8 band_idx;
95162306a36Sopenharmony_ci			u8 pad[3];
95262306a36Sopenharmony_ci		} __packed hdr;
95362306a36Sopenharmony_ci		struct sniffer_enable_tlv {
95462306a36Sopenharmony_ci			__le16 tag;
95562306a36Sopenharmony_ci			__le16 len;
95662306a36Sopenharmony_ci			u8 enable;
95762306a36Sopenharmony_ci			u8 pad[3];
95862306a36Sopenharmony_ci		} __packed enable;
95962306a36Sopenharmony_ci	} req = {
96062306a36Sopenharmony_ci		.hdr = {
96162306a36Sopenharmony_ci			.band_idx = mvif->band_idx,
96262306a36Sopenharmony_ci		},
96362306a36Sopenharmony_ci		.enable = {
96462306a36Sopenharmony_ci			.tag = cpu_to_le16(0),
96562306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct sniffer_enable_tlv)),
96662306a36Sopenharmony_ci			.enable = enable,
96762306a36Sopenharmony_ci		},
96862306a36Sopenharmony_ci	};
96962306a36Sopenharmony_ci
97062306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(SNIFFER), &req, sizeof(req),
97162306a36Sopenharmony_ci				 true);
97262306a36Sopenharmony_ci}
97362306a36Sopenharmony_ci
97462306a36Sopenharmony_ciint mt7921_mcu_config_sniffer(struct mt792x_vif *vif,
97562306a36Sopenharmony_ci			      struct ieee80211_chanctx_conf *ctx)
97662306a36Sopenharmony_ci{
97762306a36Sopenharmony_ci	struct cfg80211_chan_def *chandef = &ctx->def;
97862306a36Sopenharmony_ci	int freq1 = chandef->center_freq1, freq2 = chandef->center_freq2;
97962306a36Sopenharmony_ci	const u8 ch_band[] = {
98062306a36Sopenharmony_ci		[NL80211_BAND_2GHZ] = 1,
98162306a36Sopenharmony_ci		[NL80211_BAND_5GHZ] = 2,
98262306a36Sopenharmony_ci		[NL80211_BAND_6GHZ] = 3,
98362306a36Sopenharmony_ci	};
98462306a36Sopenharmony_ci	const u8 ch_width[] = {
98562306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_20_NOHT] = 0,
98662306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_20] = 0,
98762306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_40] = 0,
98862306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_80] = 1,
98962306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_160] = 2,
99062306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_80P80] = 3,
99162306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_5] = 4,
99262306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_10] = 5,
99362306a36Sopenharmony_ci		[NL80211_CHAN_WIDTH_320] = 6,
99462306a36Sopenharmony_ci	};
99562306a36Sopenharmony_ci	struct {
99662306a36Sopenharmony_ci		struct {
99762306a36Sopenharmony_ci			u8 band_idx;
99862306a36Sopenharmony_ci			u8 pad[3];
99962306a36Sopenharmony_ci		} __packed hdr;
100062306a36Sopenharmony_ci		struct config_tlv {
100162306a36Sopenharmony_ci			__le16 tag;
100262306a36Sopenharmony_ci			__le16 len;
100362306a36Sopenharmony_ci			u16 aid;
100462306a36Sopenharmony_ci			u8 ch_band;
100562306a36Sopenharmony_ci			u8 bw;
100662306a36Sopenharmony_ci			u8 control_ch;
100762306a36Sopenharmony_ci			u8 sco;
100862306a36Sopenharmony_ci			u8 center_ch;
100962306a36Sopenharmony_ci			u8 center_ch2;
101062306a36Sopenharmony_ci			u8 drop_err;
101162306a36Sopenharmony_ci			u8 pad[3];
101262306a36Sopenharmony_ci		} __packed tlv;
101362306a36Sopenharmony_ci	} __packed req = {
101462306a36Sopenharmony_ci		.hdr = {
101562306a36Sopenharmony_ci			.band_idx = vif->mt76.band_idx,
101662306a36Sopenharmony_ci		},
101762306a36Sopenharmony_ci		.tlv = {
101862306a36Sopenharmony_ci			.tag = cpu_to_le16(1),
101962306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(req.tlv)),
102062306a36Sopenharmony_ci			.control_ch = chandef->chan->hw_value,
102162306a36Sopenharmony_ci			.center_ch = ieee80211_frequency_to_channel(freq1),
102262306a36Sopenharmony_ci			.drop_err = 1,
102362306a36Sopenharmony_ci		},
102462306a36Sopenharmony_ci	};
102562306a36Sopenharmony_ci	if (chandef->chan->band < ARRAY_SIZE(ch_band))
102662306a36Sopenharmony_ci		req.tlv.ch_band = ch_band[chandef->chan->band];
102762306a36Sopenharmony_ci	if (chandef->width < ARRAY_SIZE(ch_width))
102862306a36Sopenharmony_ci		req.tlv.bw = ch_width[chandef->width];
102962306a36Sopenharmony_ci
103062306a36Sopenharmony_ci	if (freq2)
103162306a36Sopenharmony_ci		req.tlv.center_ch2 = ieee80211_frequency_to_channel(freq2);
103262306a36Sopenharmony_ci
103362306a36Sopenharmony_ci	if (req.tlv.control_ch < req.tlv.center_ch)
103462306a36Sopenharmony_ci		req.tlv.sco = 1; /* SCA */
103562306a36Sopenharmony_ci	else if (req.tlv.control_ch > req.tlv.center_ch)
103662306a36Sopenharmony_ci		req.tlv.sco = 3; /* SCB */
103762306a36Sopenharmony_ci
103862306a36Sopenharmony_ci	return mt76_mcu_send_msg(vif->phy->mt76->dev, MCU_UNI_CMD(SNIFFER),
103962306a36Sopenharmony_ci				 &req, sizeof(req), true);
104062306a36Sopenharmony_ci}
104162306a36Sopenharmony_ci
104262306a36Sopenharmony_ciint
104362306a36Sopenharmony_cimt7921_mcu_uni_add_beacon_offload(struct mt792x_dev *dev,
104462306a36Sopenharmony_ci				  struct ieee80211_hw *hw,
104562306a36Sopenharmony_ci				  struct ieee80211_vif *vif,
104662306a36Sopenharmony_ci				  bool enable)
104762306a36Sopenharmony_ci{
104862306a36Sopenharmony_ci	struct mt792x_vif *mvif = (struct mt792x_vif *)vif->drv_priv;
104962306a36Sopenharmony_ci	struct mt76_wcid *wcid = &dev->mt76.global_wcid;
105062306a36Sopenharmony_ci	struct ieee80211_mutable_offsets offs;
105162306a36Sopenharmony_ci	struct {
105262306a36Sopenharmony_ci		struct req_hdr {
105362306a36Sopenharmony_ci			u8 bss_idx;
105462306a36Sopenharmony_ci			u8 pad[3];
105562306a36Sopenharmony_ci		} __packed hdr;
105662306a36Sopenharmony_ci		struct bcn_content_tlv {
105762306a36Sopenharmony_ci			__le16 tag;
105862306a36Sopenharmony_ci			__le16 len;
105962306a36Sopenharmony_ci			__le16 tim_ie_pos;
106062306a36Sopenharmony_ci			__le16 csa_ie_pos;
106162306a36Sopenharmony_ci			__le16 bcc_ie_pos;
106262306a36Sopenharmony_ci			/* 0: disable beacon offload
106362306a36Sopenharmony_ci			 * 1: enable beacon offload
106462306a36Sopenharmony_ci			 * 2: update probe respond offload
106562306a36Sopenharmony_ci			 */
106662306a36Sopenharmony_ci			u8 enable;
106762306a36Sopenharmony_ci			/* 0: legacy format (TXD + payload)
106862306a36Sopenharmony_ci			 * 1: only cap field IE
106962306a36Sopenharmony_ci			 */
107062306a36Sopenharmony_ci			u8 type;
107162306a36Sopenharmony_ci			__le16 pkt_len;
107262306a36Sopenharmony_ci			u8 pkt[512];
107362306a36Sopenharmony_ci		} __packed beacon_tlv;
107462306a36Sopenharmony_ci	} req = {
107562306a36Sopenharmony_ci		.hdr = {
107662306a36Sopenharmony_ci			.bss_idx = mvif->mt76.idx,
107762306a36Sopenharmony_ci		},
107862306a36Sopenharmony_ci		.beacon_tlv = {
107962306a36Sopenharmony_ci			.tag = cpu_to_le16(UNI_BSS_INFO_BCN_CONTENT),
108062306a36Sopenharmony_ci			.len = cpu_to_le16(sizeof(struct bcn_content_tlv)),
108162306a36Sopenharmony_ci			.enable = enable,
108262306a36Sopenharmony_ci		},
108362306a36Sopenharmony_ci	};
108462306a36Sopenharmony_ci	struct sk_buff *skb;
108562306a36Sopenharmony_ci
108662306a36Sopenharmony_ci	/* support enable/update process only
108762306a36Sopenharmony_ci	 * disable flow would be handled in bss stop handler automatically
108862306a36Sopenharmony_ci	 */
108962306a36Sopenharmony_ci	if (!enable)
109062306a36Sopenharmony_ci		return -EOPNOTSUPP;
109162306a36Sopenharmony_ci
109262306a36Sopenharmony_ci	skb = ieee80211_beacon_get_template(mt76_hw(dev), vif, &offs, 0);
109362306a36Sopenharmony_ci	if (!skb)
109462306a36Sopenharmony_ci		return -EINVAL;
109562306a36Sopenharmony_ci
109662306a36Sopenharmony_ci	if (skb->len > 512 - MT_TXD_SIZE) {
109762306a36Sopenharmony_ci		dev_err(dev->mt76.dev, "beacon size limit exceed\n");
109862306a36Sopenharmony_ci		dev_kfree_skb(skb);
109962306a36Sopenharmony_ci		return -EINVAL;
110062306a36Sopenharmony_ci	}
110162306a36Sopenharmony_ci
110262306a36Sopenharmony_ci	mt76_connac2_mac_write_txwi(&dev->mt76, (__le32 *)(req.beacon_tlv.pkt),
110362306a36Sopenharmony_ci				    skb, wcid, NULL, 0, 0, BSS_CHANGED_BEACON);
110462306a36Sopenharmony_ci	memcpy(req.beacon_tlv.pkt + MT_TXD_SIZE, skb->data, skb->len);
110562306a36Sopenharmony_ci	req.beacon_tlv.pkt_len = cpu_to_le16(MT_TXD_SIZE + skb->len);
110662306a36Sopenharmony_ci	req.beacon_tlv.tim_ie_pos = cpu_to_le16(MT_TXD_SIZE + offs.tim_offset);
110762306a36Sopenharmony_ci
110862306a36Sopenharmony_ci	if (offs.cntdwn_counter_offs[0]) {
110962306a36Sopenharmony_ci		u16 csa_offs;
111062306a36Sopenharmony_ci
111162306a36Sopenharmony_ci		csa_offs = MT_TXD_SIZE + offs.cntdwn_counter_offs[0] - 4;
111262306a36Sopenharmony_ci		req.beacon_tlv.csa_ie_pos = cpu_to_le16(csa_offs);
111362306a36Sopenharmony_ci	}
111462306a36Sopenharmony_ci	dev_kfree_skb(skb);
111562306a36Sopenharmony_ci
111662306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
111762306a36Sopenharmony_ci				 &req, sizeof(req), true);
111862306a36Sopenharmony_ci}
111962306a36Sopenharmony_ci
112062306a36Sopenharmony_cistatic
112162306a36Sopenharmony_ciint __mt7921_mcu_set_clc(struct mt792x_dev *dev, u8 *alpha2,
112262306a36Sopenharmony_ci			 enum environment_cap env_cap,
112362306a36Sopenharmony_ci			 struct mt7921_clc *clc,
112462306a36Sopenharmony_ci			 u8 idx)
112562306a36Sopenharmony_ci{
112662306a36Sopenharmony_ci	struct sk_buff *skb;
112762306a36Sopenharmony_ci	struct {
112862306a36Sopenharmony_ci		u8 ver;
112962306a36Sopenharmony_ci		u8 pad0;
113062306a36Sopenharmony_ci		__le16 len;
113162306a36Sopenharmony_ci		u8 idx;
113262306a36Sopenharmony_ci		u8 env;
113362306a36Sopenharmony_ci		u8 acpi_conf;
113462306a36Sopenharmony_ci		u8 pad1;
113562306a36Sopenharmony_ci		u8 alpha2[2];
113662306a36Sopenharmony_ci		u8 type[2];
113762306a36Sopenharmony_ci		u8 rsvd[64];
113862306a36Sopenharmony_ci	} __packed req = {
113962306a36Sopenharmony_ci		.ver = 1,
114062306a36Sopenharmony_ci		.idx = idx,
114162306a36Sopenharmony_ci		.env = env_cap,
114262306a36Sopenharmony_ci		.acpi_conf = mt792x_acpi_get_flags(&dev->phy),
114362306a36Sopenharmony_ci	};
114462306a36Sopenharmony_ci	int ret, valid_cnt = 0;
114562306a36Sopenharmony_ci	u16 buf_len = 0;
114662306a36Sopenharmony_ci	u8 *pos;
114762306a36Sopenharmony_ci
114862306a36Sopenharmony_ci	if (!clc)
114962306a36Sopenharmony_ci		return 0;
115062306a36Sopenharmony_ci
115162306a36Sopenharmony_ci	buf_len = le16_to_cpu(clc->len) - sizeof(*clc);
115262306a36Sopenharmony_ci	pos = clc->data;
115362306a36Sopenharmony_ci	while (buf_len > 16) {
115462306a36Sopenharmony_ci		struct mt7921_clc_rule *rule = (struct mt7921_clc_rule *)pos;
115562306a36Sopenharmony_ci		u16 len = le16_to_cpu(rule->len);
115662306a36Sopenharmony_ci		u16 offset = len + sizeof(*rule);
115762306a36Sopenharmony_ci
115862306a36Sopenharmony_ci		pos += offset;
115962306a36Sopenharmony_ci		buf_len -= offset;
116062306a36Sopenharmony_ci		if (rule->alpha2[0] != alpha2[0] ||
116162306a36Sopenharmony_ci		    rule->alpha2[1] != alpha2[1])
116262306a36Sopenharmony_ci			continue;
116362306a36Sopenharmony_ci
116462306a36Sopenharmony_ci		memcpy(req.alpha2, rule->alpha2, 2);
116562306a36Sopenharmony_ci		memcpy(req.type, rule->type, 2);
116662306a36Sopenharmony_ci
116762306a36Sopenharmony_ci		req.len = cpu_to_le16(sizeof(req) + len);
116862306a36Sopenharmony_ci		skb = __mt76_mcu_msg_alloc(&dev->mt76, &req,
116962306a36Sopenharmony_ci					   le16_to_cpu(req.len),
117062306a36Sopenharmony_ci					   sizeof(req), GFP_KERNEL);
117162306a36Sopenharmony_ci		if (!skb)
117262306a36Sopenharmony_ci			return -ENOMEM;
117362306a36Sopenharmony_ci		skb_put_data(skb, rule->data, len);
117462306a36Sopenharmony_ci
117562306a36Sopenharmony_ci		ret = mt76_mcu_skb_send_msg(&dev->mt76, skb,
117662306a36Sopenharmony_ci					    MCU_CE_CMD(SET_CLC), false);
117762306a36Sopenharmony_ci		if (ret < 0)
117862306a36Sopenharmony_ci			return ret;
117962306a36Sopenharmony_ci		valid_cnt++;
118062306a36Sopenharmony_ci	}
118162306a36Sopenharmony_ci
118262306a36Sopenharmony_ci	if (!valid_cnt)
118362306a36Sopenharmony_ci		return -ENOENT;
118462306a36Sopenharmony_ci
118562306a36Sopenharmony_ci	return 0;
118662306a36Sopenharmony_ci}
118762306a36Sopenharmony_ci
118862306a36Sopenharmony_ciint mt7921_mcu_set_clc(struct mt792x_dev *dev, u8 *alpha2,
118962306a36Sopenharmony_ci		       enum environment_cap env_cap)
119062306a36Sopenharmony_ci{
119162306a36Sopenharmony_ci	struct mt792x_phy *phy = (struct mt792x_phy *)&dev->phy;
119262306a36Sopenharmony_ci	int i, ret;
119362306a36Sopenharmony_ci
119462306a36Sopenharmony_ci	/* submit all clc config */
119562306a36Sopenharmony_ci	for (i = 0; i < ARRAY_SIZE(phy->clc); i++) {
119662306a36Sopenharmony_ci		ret = __mt7921_mcu_set_clc(dev, alpha2, env_cap,
119762306a36Sopenharmony_ci					   phy->clc[i], i);
119862306a36Sopenharmony_ci
119962306a36Sopenharmony_ci		/* If no country found, set "00" as default */
120062306a36Sopenharmony_ci		if (ret == -ENOENT)
120162306a36Sopenharmony_ci			ret = __mt7921_mcu_set_clc(dev, "00",
120262306a36Sopenharmony_ci						   ENVIRON_INDOOR,
120362306a36Sopenharmony_ci						   phy->clc[i], i);
120462306a36Sopenharmony_ci		if (ret < 0)
120562306a36Sopenharmony_ci			return ret;
120662306a36Sopenharmony_ci	}
120762306a36Sopenharmony_ci	return 0;
120862306a36Sopenharmony_ci}
120962306a36Sopenharmony_ci
121062306a36Sopenharmony_ciint mt7921_mcu_get_temperature(struct mt792x_phy *phy)
121162306a36Sopenharmony_ci{
121262306a36Sopenharmony_ci	struct mt792x_dev *dev = phy->dev;
121362306a36Sopenharmony_ci	struct {
121462306a36Sopenharmony_ci		u8 ctrl_id;
121562306a36Sopenharmony_ci		u8 action;
121662306a36Sopenharmony_ci		u8 band_idx;
121762306a36Sopenharmony_ci		u8 rsv[5];
121862306a36Sopenharmony_ci	} req = {
121962306a36Sopenharmony_ci		.ctrl_id = THERMAL_SENSOR_TEMP_QUERY,
122062306a36Sopenharmony_ci		.band_idx = phy->mt76->band_idx,
122162306a36Sopenharmony_ci	};
122262306a36Sopenharmony_ci
122362306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_EXT_CMD(THERMAL_CTRL), &req,
122462306a36Sopenharmony_ci				 sizeof(req), true);
122562306a36Sopenharmony_ci}
122662306a36Sopenharmony_ci
122762306a36Sopenharmony_ciint mt7921_mcu_set_rxfilter(struct mt792x_dev *dev, u32 fif,
122862306a36Sopenharmony_ci			    u8 bit_op, u32 bit_map)
122962306a36Sopenharmony_ci{
123062306a36Sopenharmony_ci	struct {
123162306a36Sopenharmony_ci		u8 rsv[4];
123262306a36Sopenharmony_ci		u8 mode;
123362306a36Sopenharmony_ci		u8 rsv2[3];
123462306a36Sopenharmony_ci		__le32 fif;
123562306a36Sopenharmony_ci		__le32 bit_map; /* bit_* for bitmap update */
123662306a36Sopenharmony_ci		u8 bit_op;
123762306a36Sopenharmony_ci		u8 pad[51];
123862306a36Sopenharmony_ci	} __packed data = {
123962306a36Sopenharmony_ci		.mode = fif ? 1 : 2,
124062306a36Sopenharmony_ci		.fif = cpu_to_le32(fif),
124162306a36Sopenharmony_ci		.bit_map = cpu_to_le32(bit_map),
124262306a36Sopenharmony_ci		.bit_op = bit_op,
124362306a36Sopenharmony_ci	};
124462306a36Sopenharmony_ci
124562306a36Sopenharmony_ci	return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_RX_FILTER),
124662306a36Sopenharmony_ci				 &data, sizeof(data), false);
124762306a36Sopenharmony_ci}
1248