1 // SPDX-License-Identifier: ISC
2 /* Copyright (C) 2020 MediaTek Inc. */
6 #include "mt7921_trace.h"
10 #define MT_STA_BFER BIT(0)
11 #define MT_STA_BFEE BIT(1)
14 mt7921_mcu_parse_eeprom(struct mt76_dev *dev, struct sk_buff *skb)
16 struct mt7921_mcu_eeprom_info *res;
22 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
24 res = (struct mt7921_mcu_eeprom_info *)skb->data;
25 buf = dev->eeprom.data + le32_to_cpu(res->addr);
26 memcpy(buf, res->data, 16);
31 int mt7921_mcu_parse_response(struct mt76_dev *mdev, int cmd,
32 struct sk_buff *skb, int seq)
34 int mcu_cmd = FIELD_GET(__MCU_CMD_FIELD_ID, cmd);
35 struct mt76_connac2_mcu_rxd *rxd;
39 dev_err(mdev->dev, "Message %08x (seq %d) timeout\n",
46 rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
50 if (cmd == MCU_CMD(PATCH_SEM_CONTROL) ||
51 cmd == MCU_CMD(PATCH_FINISH_REQ)) {
52 skb_pull(skb, sizeof(*rxd) - 4);
54 } else if (cmd == MCU_EXT_CMD(THERMAL_CTRL)) {
55 skb_pull(skb, sizeof(*rxd) + 4);
56 ret = le32_to_cpu(*(__le32 *)skb->data);
57 } else if (cmd == MCU_EXT_CMD(EFUSE_ACCESS)) {
58 ret = mt7921_mcu_parse_eeprom(mdev, skb);
59 } else if (cmd == MCU_UNI_CMD(DEV_INFO_UPDATE) ||
60 cmd == MCU_UNI_CMD(BSS_INFO_UPDATE) ||
61 cmd == MCU_UNI_CMD(STA_REC_UPDATE) ||
62 cmd == MCU_UNI_CMD(HIF_CTRL) ||
63 cmd == MCU_UNI_CMD(OFFLOAD) ||
64 cmd == MCU_UNI_CMD(SUSPEND)) {
65 struct mt7921_mcu_uni_event *event;
67 skb_pull(skb, sizeof(*rxd));
68 event = (struct mt7921_mcu_uni_event *)skb->data;
69 ret = le32_to_cpu(event->status);
70 /* skip invalid event */
71 if (mcu_cmd != event->cid)
73 } else if (cmd == MCU_CE_QUERY(REG_READ)) {
74 struct mt7921_mcu_reg_event *event;
76 skb_pull(skb, sizeof(*rxd));
77 event = (struct mt7921_mcu_reg_event *)skb->data;
78 ret = (int)le32_to_cpu(event->val);
80 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
85 EXPORT_SYMBOL_GPL(mt7921_mcu_parse_response);
90 mt7921_mcu_set_ipv6_ns_filter(struct mt76_dev *dev,
91 struct ieee80211_vif *vif, bool suspend)
93 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
99 struct mt76_connac_arpns_tlv arpns;
102 .bss_idx = mvif->mt76.idx,
105 .tag = cpu_to_le16(UNI_OFFLOAD_OFFLOAD_ND),
106 .len = cpu_to_le16(sizeof(struct mt76_connac_arpns_tlv)),
111 return mt76_mcu_send_msg(dev, MCU_UNI_CMD_OFFLOAD, &req, sizeof(req),
115 void mt7921_mcu_set_suspend_iter(void *priv, u8 *mac, struct ieee80211_vif *vif)
117 if (IS_ENABLED(CONFIG_IPV6)) {
118 struct mt76_phy *phy = priv;
120 mt7921_mcu_set_ipv6_ns_filter(phy->dev, vif,
121 !test_bit(MT76_STATE_RUNNING,
125 mt76_connac_mcu_set_suspend_iter(priv, mac, vif);
128 #endif /* CONFIG_PM */
131 mt7921_mcu_scan_event(struct mt7921_dev *dev, struct sk_buff *skb)
133 struct mt76_phy *mphy = &dev->mt76.phy;
134 struct mt7921_phy *phy = (struct mt7921_phy *)mphy->priv;
136 spin_lock_bh(&dev->mt76.lock);
137 __skb_queue_tail(&phy->scan_event_list, skb);
138 spin_unlock_bh(&dev->mt76.lock);
140 ieee80211_queue_delayed_work(mphy->hw, &phy->scan_work,
141 MT7921_HW_SCAN_TIMEOUT);
145 mt7921_mcu_connection_loss_iter(void *priv, u8 *mac,
146 struct ieee80211_vif *vif)
148 struct mt76_vif *mvif = (struct mt76_vif *)vif->drv_priv;
149 struct mt76_connac_beacon_loss_event *event = priv;
151 if (mvif->idx != event->bss_idx)
154 if (!(vif->driver_flags & IEEE80211_VIF_BEACON_FILTER) ||
155 vif->type != NL80211_IFTYPE_STATION)
158 ieee80211_connection_loss(vif);
162 mt7921_mcu_connection_loss_event(struct mt7921_dev *dev, struct sk_buff *skb)
164 struct mt76_connac_beacon_loss_event *event;
165 struct mt76_phy *mphy = &dev->mt76.phy;
167 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
168 event = (struct mt76_connac_beacon_loss_event *)skb->data;
170 ieee80211_iterate_active_interfaces_atomic(mphy->hw,
171 IEEE80211_IFACE_ITER_RESUME_ALL,
172 mt7921_mcu_connection_loss_iter, event);
176 mt7921_mcu_bss_event(struct mt7921_dev *dev, struct sk_buff *skb)
178 struct mt76_phy *mphy = &dev->mt76.phy;
179 struct mt76_connac_mcu_bss_event *event;
181 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
182 event = (struct mt76_connac_mcu_bss_event *)skb->data;
183 if (event->is_absent)
184 ieee80211_stop_queues(mphy->hw);
186 ieee80211_wake_queues(mphy->hw);
190 mt7921_mcu_debug_msg_event(struct mt7921_dev *dev, struct sk_buff *skb)
192 struct mt7921_debug_msg {
201 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
202 msg = (struct mt7921_debug_msg *)skb->data;
204 if (msg->type == 3) { /* fw log */
205 u16 len = min_t(u16, le16_to_cpu(msg->len), 512);
208 for (i = 0 ; i < len; i++) {
209 if (!msg->content[i])
210 msg->content[i] = ' ';
212 wiphy_info(mt76_hw(dev)->wiphy, "%.*s", len, msg->content);
217 mt7921_mcu_low_power_event(struct mt7921_dev *dev, struct sk_buff *skb)
219 struct mt7921_mcu_lp_event {
224 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
225 event = (struct mt7921_mcu_lp_event *)skb->data;
227 trace_lp_event(dev, event->state);
231 mt7921_mcu_tx_done_event(struct mt7921_dev *dev, struct sk_buff *skb)
233 struct mt7921_mcu_tx_done_event *event;
235 skb_pull(skb, sizeof(struct mt76_connac2_mcu_rxd));
236 event = (struct mt7921_mcu_tx_done_event *)skb->data;
238 mt7921_mac_add_txs(dev, event->txs);
242 mt7921_mcu_rx_unsolicited_event(struct mt7921_dev *dev, struct sk_buff *skb)
244 struct mt76_connac2_mcu_rxd *rxd;
246 rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
248 case MCU_EVENT_BSS_BEACON_LOSS:
249 mt7921_mcu_connection_loss_event(dev, skb);
251 case MCU_EVENT_SCHED_SCAN_DONE:
252 case MCU_EVENT_SCAN_DONE:
253 mt7921_mcu_scan_event(dev, skb);
255 case MCU_EVENT_BSS_ABSENCE:
256 mt7921_mcu_bss_event(dev, skb);
258 case MCU_EVENT_DBG_MSG:
259 mt7921_mcu_debug_msg_event(dev, skb);
261 case MCU_EVENT_COREDUMP:
262 dev->fw_assert = true;
263 mt76_connac_mcu_coredump_event(&dev->mt76, skb,
266 case MCU_EVENT_LP_INFO:
267 mt7921_mcu_low_power_event(dev, skb);
269 case MCU_EVENT_TX_DONE:
270 mt7921_mcu_tx_done_event(dev, skb);
278 void mt7921_mcu_rx_event(struct mt7921_dev *dev, struct sk_buff *skb)
280 struct mt76_connac2_mcu_rxd *rxd;
282 if (skb_linearize(skb))
285 rxd = (struct mt76_connac2_mcu_rxd *)skb->data;
287 if (rxd->eid == 0x6) {
288 mt76_mcu_rx_event(&dev->mt76, skb);
292 if (rxd->ext_eid == MCU_EXT_EVENT_RATE_REPORT ||
293 rxd->eid == MCU_EVENT_BSS_BEACON_LOSS ||
294 rxd->eid == MCU_EVENT_SCHED_SCAN_DONE ||
295 rxd->eid == MCU_EVENT_BSS_ABSENCE ||
296 rxd->eid == MCU_EVENT_SCAN_DONE ||
297 rxd->eid == MCU_EVENT_TX_DONE ||
298 rxd->eid == MCU_EVENT_DBG_MSG ||
299 rxd->eid == MCU_EVENT_COREDUMP ||
300 rxd->eid == MCU_EVENT_LP_INFO ||
302 mt7921_mcu_rx_unsolicited_event(dev, skb);
304 mt76_mcu_rx_event(&dev->mt76, skb);
307 /** starec & wtbl **/
308 int mt7921_mcu_uni_tx_ba(struct mt7921_dev *dev,
309 struct ieee80211_ampdu_params *params,
312 struct mt7921_sta *msta = (struct mt7921_sta *)params->sta->drv_priv;
314 if (enable && !params->amsdu)
315 msta->wcid.amsdu = false;
317 return mt76_connac_mcu_sta_ba(&dev->mt76, &msta->vif->mt76, params,
318 MCU_UNI_CMD(STA_REC_UPDATE),
322 int mt7921_mcu_uni_rx_ba(struct mt7921_dev *dev,
323 struct ieee80211_ampdu_params *params,
326 struct mt7921_sta *msta = (struct mt7921_sta *)params->sta->drv_priv;
328 return mt76_connac_mcu_sta_ba(&dev->mt76, &msta->vif->mt76, params,
329 MCU_UNI_CMD(STA_REC_UPDATE),
333 static char *mt7921_patch_name(struct mt7921_dev *dev)
337 if (is_mt7922(&dev->mt76))
338 ret = MT7922_ROM_PATCH;
340 ret = MT7921_ROM_PATCH;
345 static char *mt7921_ram_name(struct mt7921_dev *dev)
349 if (is_mt7922(&dev->mt76))
350 ret = MT7922_FIRMWARE_WM;
352 ret = MT7921_FIRMWARE_WM;
357 static int mt7921_load_firmware(struct mt7921_dev *dev)
361 ret = mt76_get_field(dev, MT_CONN_ON_MISC, MT_TOP_MISC2_FW_N9_RDY);
362 if (ret && mt76_is_mmio(&dev->mt76)) {
363 dev_dbg(dev->mt76.dev, "Firmware is already download\n");
367 ret = mt76_connac2_load_patch(&dev->mt76, mt7921_patch_name(dev));
371 if (mt76_is_sdio(&dev->mt76)) {
373 ret = __mt7921_mcu_fw_pmctrl(dev);
375 ret = __mt7921_mcu_drv_pmctrl(dev);
378 ret = mt76_connac2_load_ram(&dev->mt76, mt7921_ram_name(dev), NULL);
382 if (!mt76_poll_msec(dev, MT_CONN_ON_MISC, MT_TOP_MISC2_FW_N9_RDY,
383 MT_TOP_MISC2_FW_N9_RDY, 1500)) {
384 dev_err(dev->mt76.dev, "Timeout for initializing firmware\n");
392 dev->mt76.hw->wiphy->wowlan = &mt76_connac_wowlan_support;
393 #endif /* CONFIG_PM */
395 dev_dbg(dev->mt76.dev, "Firmware init done\n");
400 int mt7921_mcu_fw_log_2_host(struct mt7921_dev *dev, u8 ctrl)
409 return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(FWLOG_2_HOST),
410 &data, sizeof(data), false);
413 int mt7921_run_firmware(struct mt7921_dev *dev)
417 err = mt7921_load_firmware(dev);
421 err = mt76_connac_mcu_get_nic_capability(&dev->mphy);
425 set_bit(MT76_STATE_MCU_RUNNING, &dev->mphy.state);
426 return mt7921_mcu_fw_log_2_host(dev, 1);
428 EXPORT_SYMBOL_GPL(mt7921_run_firmware);
430 int mt7921_mcu_set_tx(struct mt7921_dev *dev, struct ieee80211_vif *vif)
432 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
441 struct mt7921_mcu_tx {
442 struct edca edca[IEEE80211_NUM_ACS];
448 .bss_idx = mvif->mt76.idx,
449 .qos = vif->bss_conf.qos,
450 .wmm_idx = mvif->mt76.wmm_idx,
460 struct mt7921_mcu_mu_tx {
468 struct mu_edca edca[IEEE80211_NUM_ACS];
470 } __packed req_mu = {
471 .bss_idx = mvif->mt76.idx,
472 .qos = vif->bss_conf.qos,
473 .wmm_idx = mvif->mt76.wmm_idx,
475 static const int to_aci[] = { 1, 0, 2, 3 };
478 for (ac = 0; ac < IEEE80211_NUM_ACS; ac++) {
479 struct ieee80211_tx_queue_params *q = &mvif->queue_params[ac];
480 struct edca *e = &req.edca[to_aci[ac]];
482 e->aifs = cpu_to_le16(q->aifs);
483 e->txop = cpu_to_le16(q->txop);
486 e->cw_min = cpu_to_le16(q->cw_min);
488 e->cw_min = cpu_to_le16(5);
491 e->cw_max = cpu_to_le16(q->cw_max);
493 e->cw_max = cpu_to_le16(10);
496 ret = mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_EDCA_PARMS), &req,
501 if (!vif->bss_conf.he_support)
504 for (ac = 0; ac < IEEE80211_NUM_ACS; ac++) {
505 struct ieee80211_he_mu_edca_param_ac_rec *q;
508 if (!mvif->queue_params[ac].mu_edca)
511 q = &mvif->queue_params[ac].mu_edca_param_rec;
512 e = &(req_mu.edca[to_aci[ac]]);
514 e->cw_min = q->ecw_min_max & 0xf;
515 e->cw_max = (q->ecw_min_max & 0xf0) >> 4;
517 e->timer = q->mu_edca_timer;
520 return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_MU_EDCA_PARMS),
521 &req_mu, sizeof(req_mu), false);
524 int mt7921_mcu_set_chan_info(struct mt7921_phy *phy, int cmd)
526 struct mt7921_dev *dev = phy->dev;
527 struct cfg80211_chan_def *chandef = &phy->mt76->chandef;
528 int freq1 = chandef->center_freq1;
534 u8 rx_streams; /* mask or num */
537 u8 center_ch2; /* for 80+80 only */
547 .control_ch = chandef->chan->hw_value,
548 .center_ch = ieee80211_frequency_to_channel(freq1),
549 .bw = mt76_connac_chan_bw(chandef),
550 .tx_streams_num = hweight8(phy->mt76->antenna_mask),
551 .rx_streams = phy->mt76->antenna_mask,
552 .band_idx = phy != &dev->phy,
555 if (chandef->chan->band == NL80211_BAND_6GHZ)
556 req.channel_band = 2;
558 req.channel_band = chandef->chan->band;
560 if (cmd == MCU_EXT_CMD(SET_RX_PATH) ||
561 dev->mt76.hw->conf.flags & IEEE80211_CONF_MONITOR)
562 req.switch_reason = CH_SWITCH_NORMAL;
563 else if (dev->mt76.hw->conf.flags & IEEE80211_CONF_OFFCHANNEL)
564 req.switch_reason = CH_SWITCH_SCAN_BYPASS_DPD;
565 else if (!cfg80211_reg_can_beacon(dev->mt76.hw->wiphy, chandef,
567 req.switch_reason = CH_SWITCH_DFS;
569 req.switch_reason = CH_SWITCH_NORMAL;
571 if (cmd == MCU_EXT_CMD(CHANNEL_SWITCH))
572 req.rx_streams = hweight8(req.rx_streams);
574 if (chandef->width == NL80211_CHAN_WIDTH_80P80) {
575 int freq2 = chandef->center_freq2;
577 req.center_ch2 = ieee80211_frequency_to_channel(freq2);
580 return mt76_mcu_send_msg(&dev->mt76, cmd, &req, sizeof(req), true);
583 int mt7921_mcu_set_eeprom(struct mt7921_dev *dev)
590 .buffer_mode = EE_MODE_EFUSE,
591 .format = EE_FORMAT_WHOLE,
594 return mt76_mcu_send_msg(&dev->mt76, MCU_EXT_CMD(EFUSE_BUFFER_MODE),
595 &req, sizeof(req), true);
597 EXPORT_SYMBOL_GPL(mt7921_mcu_set_eeprom);
599 int mt7921_mcu_uni_bss_ps(struct mt7921_dev *dev, struct ieee80211_vif *vif)
601 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
610 u8 ps_state; /* 0: device awake
611 * 1: static power save
612 * 2: dynamic power saving
613 * 3: enter TWT power saving
614 * 4: leave TWT power saving
618 } __packed ps_req = {
620 .bss_idx = mvif->mt76.idx,
623 .tag = cpu_to_le16(UNI_BSS_INFO_PS),
624 .len = cpu_to_le16(sizeof(struct ps_tlv)),
625 .ps_state = vif->bss_conf.ps ? 2 : 0,
629 if (vif->type != NL80211_IFTYPE_STATION)
632 return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
633 &ps_req, sizeof(ps_req), true);
637 mt7921_mcu_uni_bss_bcnft(struct mt7921_dev *dev, struct ieee80211_vif *vif,
640 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
653 } __packed bcnft_req = {
655 .bss_idx = mvif->mt76.idx,
658 .tag = cpu_to_le16(UNI_BSS_INFO_BCNFT),
659 .len = cpu_to_le16(sizeof(struct bcnft_tlv)),
660 .bcn_interval = cpu_to_le16(vif->bss_conf.beacon_int),
661 .dtim_period = vif->bss_conf.dtim_period,
665 if (vif->type != NL80211_IFTYPE_STATION)
668 return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
669 &bcnft_req, sizeof(bcnft_req), true);
673 mt7921_mcu_set_bss_pm(struct mt7921_dev *dev, struct ieee80211_vif *vif,
676 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
688 .bss_idx = mvif->mt76.idx,
689 .aid = cpu_to_le16(vif->cfg.aid),
690 .dtim_period = vif->bss_conf.dtim_period,
691 .bcn_interval = cpu_to_le16(vif->bss_conf.beacon_int),
697 .bss_idx = mvif->mt76.idx,
701 err = mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_BSS_ABORT),
702 &req_hdr, sizeof(req_hdr), false);
703 if (err < 0 || !enable)
706 return mt76_mcu_send_msg(&dev->mt76, MCU_CE_CMD(SET_BSS_CONNECTED),
707 &req, sizeof(req), false);
710 int mt7921_mcu_sta_update(struct mt7921_dev *dev, struct ieee80211_sta *sta,
711 struct ieee80211_vif *vif, bool enable,
712 enum mt76_sta_info_state state)
714 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
715 int rssi = -ewma_rssi_read(&mvif->rssi);
716 struct mt76_sta_cmd_info info = {
720 .cmd = MCU_UNI_CMD(STA_REC_UPDATE),
723 .rcpi = to_rcpi(rssi),
725 struct mt7921_sta *msta;
727 msta = sta ? (struct mt7921_sta *)sta->drv_priv : NULL;
728 info.wcid = msta ? &msta->wcid : &mvif->sta.wcid;
729 info.newly = msta ? state != MT76_STA_INFO_STATE_ASSOC : true;
731 return mt76_connac_mcu_sta_cmd(&dev->mphy, &info);
734 int mt7921_mcu_drv_pmctrl(struct mt7921_dev *dev)
736 struct mt76_phy *mphy = &dev->mt76.phy;
737 struct mt76_connac_pm *pm = &dev->pm;
740 mutex_lock(&pm->mutex);
742 if (!test_bit(MT76_STATE_PM, &mphy->state))
745 err = __mt7921_mcu_drv_pmctrl(dev);
747 mutex_unlock(&pm->mutex);
750 mt7921_reset(&dev->mt76);
754 EXPORT_SYMBOL_GPL(mt7921_mcu_drv_pmctrl);
756 int mt7921_mcu_fw_pmctrl(struct mt7921_dev *dev)
758 struct mt76_phy *mphy = &dev->mt76.phy;
759 struct mt76_connac_pm *pm = &dev->pm;
762 mutex_lock(&pm->mutex);
764 if (mt76_connac_skip_fw_pmctrl(mphy, pm))
767 err = __mt7921_mcu_fw_pmctrl(dev);
769 mutex_unlock(&pm->mutex);
772 mt7921_reset(&dev->mt76);
776 EXPORT_SYMBOL_GPL(mt7921_mcu_fw_pmctrl);
778 int mt7921_mcu_set_beacon_filter(struct mt7921_dev *dev,
779 struct ieee80211_vif *vif,
782 struct ieee80211_hw *hw = mt76_hw(dev);
786 err = mt7921_mcu_uni_bss_bcnft(dev, vif, true);
790 vif->driver_flags |= IEEE80211_VIF_BEACON_FILTER;
791 ieee80211_hw_set(hw, CONNECTION_MONITOR);
792 mt76_set(dev, MT_WF_RFCR(0), MT_WF_RFCR_DROP_OTHER_BEACON);
797 err = mt7921_mcu_set_bss_pm(dev, vif, false);
801 vif->driver_flags &= ~IEEE80211_VIF_BEACON_FILTER;
802 __clear_bit(IEEE80211_HW_CONNECTION_MONITOR, hw->flags);
803 mt76_clear(dev, MT_WF_RFCR(0), MT_WF_RFCR_DROP_OTHER_BEACON);
808 int mt7921_get_txpwr_info(struct mt7921_dev *dev, struct mt7921_txpwr *txpwr)
810 struct mt7921_txpwr_event *event;
811 struct mt7921_txpwr_req req = {
817 ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_CE_CMD(GET_TXPWR),
818 &req, sizeof(req), true, &skb);
822 event = (struct mt7921_txpwr_event *)skb->data;
823 WARN_ON(skb->len != le16_to_cpu(event->len));
824 memcpy(txpwr, &event->txpwr, sizeof(event->txpwr));
831 int mt7921_mcu_set_sniffer(struct mt7921_dev *dev, struct ieee80211_vif *vif,
834 struct mt76_vif *mvif = (struct mt76_vif *)vif->drv_priv;
840 struct sniffer_enable_tlv {
848 .band_idx = mvif->band_idx,
851 .tag = cpu_to_le16(0),
852 .len = cpu_to_le16(sizeof(struct sniffer_enable_tlv)),
857 return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(SNIFFER), &req, sizeof(req),
862 mt7921_mcu_uni_add_beacon_offload(struct mt7921_dev *dev,
863 struct ieee80211_hw *hw,
864 struct ieee80211_vif *vif,
867 struct mt7921_vif *mvif = (struct mt7921_vif *)vif->drv_priv;
868 struct mt76_wcid *wcid = &dev->mt76.global_wcid;
869 struct ieee80211_mutable_offsets offs;
875 struct bcn_content_tlv {
881 /* 0: disable beacon offload
882 * 1: enable beacon offload
883 * 2: update probe respond offload
886 /* 0: legacy format (TXD + payload)
887 * 1: only cap field IE
892 } __packed beacon_tlv;
895 .bss_idx = mvif->mt76.idx,
898 .tag = cpu_to_le16(UNI_BSS_INFO_BCN_CONTENT),
899 .len = cpu_to_le16(sizeof(struct bcn_content_tlv)),
905 /* support enable/update process only
906 * disable flow would be handled in bss stop handler automatically
911 skb = ieee80211_beacon_get_template(mt76_hw(dev), vif, &offs, 0);
915 if (skb->len > 512 - MT_TXD_SIZE) {
916 dev_err(dev->mt76.dev, "beacon size limit exceed\n");
921 mt76_connac2_mac_write_txwi(&dev->mt76, (__le32 *)(req.beacon_tlv.pkt),
922 skb, wcid, NULL, 0, 0, BSS_CHANGED_BEACON);
923 memcpy(req.beacon_tlv.pkt + MT_TXD_SIZE, skb->data, skb->len);
924 req.beacon_tlv.pkt_len = cpu_to_le16(MT_TXD_SIZE + skb->len);
925 req.beacon_tlv.tim_ie_pos = cpu_to_le16(MT_TXD_SIZE + offs.tim_offset);
927 if (offs.cntdwn_counter_offs[0]) {
930 csa_offs = MT_TXD_SIZE + offs.cntdwn_counter_offs[0] - 4;
931 req.beacon_tlv.csa_ie_pos = cpu_to_le16(csa_offs);
935 return mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(BSS_INFO_UPDATE),
936 &req, sizeof(req), true);