Path: blob/main/sys/contrib/dev/mediatek/mt76/mt7925/testmode.c
48525 views
// SPDX-License-Identifier: ISC12#include "mt7925.h"3#include "mcu.h"45#define MT7925_EVT_RSP_LEN 51267enum mt7925_testmode_attr {8MT7925_TM_ATTR_UNSPEC,9MT7925_TM_ATTR_SET,10MT7925_TM_ATTR_QUERY,11MT7925_TM_ATTR_RSP,1213/* keep last */14NUM_MT7925_TM_ATTRS,15MT7925_TM_ATTR_MAX = NUM_MT7925_TM_ATTRS - 1,16};1718struct mt7925_tm_cmd {19u8 padding[4];20struct uni_cmd_testmode_ctrl c;21} __packed;2223struct mt7925_tm_evt {24u32 param0;25u32 param1;26} __packed;2728static const struct nla_policy mt7925_tm_policy[NUM_MT7925_TM_ATTRS] = {29[MT7925_TM_ATTR_SET] = NLA_POLICY_EXACT_LEN(sizeof(struct mt7925_tm_cmd)),30[MT7925_TM_ATTR_QUERY] = NLA_POLICY_EXACT_LEN(sizeof(struct mt7925_tm_cmd)),31};3233static int34mt7925_tm_set(struct mt792x_dev *dev, struct mt7925_tm_cmd *req)35{36struct mt7925_rftest_cmd cmd;37struct mt7925_rftest_cmd *pcmd = &cmd;38bool testmode = false, normal = false;39struct mt76_connac_pm *pm = &dev->pm;40struct mt76_phy *phy = &dev->mphy;41int ret = -ENOTCONN;4243memset(pcmd, 0, sizeof(*pcmd));44memcpy(pcmd, req, sizeof(struct mt7925_tm_cmd));4546mutex_lock(&dev->mt76.mutex);4748if (pcmd->ctrl.action == CMD_TEST_CTRL_ACT_SWITCH_MODE) {49if (pcmd->ctrl.data.op_mode == CMD_TEST_CTRL_ACT_SWITCH_MODE_NORMAL)50normal = true;51else52testmode = true;53}5455if (testmode) {56/* Make sure testmode running on full power mode */57pm->enable = false;58cancel_delayed_work_sync(&pm->ps_work);59cancel_work_sync(&pm->wake_work);60__mt792x_mcu_drv_pmctrl(dev);6162phy->test.state = MT76_TM_STATE_ON;63}6465if (!mt76_testmode_enabled(phy))66goto out;6768ret = mt76_mcu_send_msg(&dev->mt76, MCU_UNI_CMD(TESTMODE_CTRL), &cmd,69sizeof(cmd), false);7071if (ret)72goto out;7374if (normal) {75/* Switch back to the normal world */76phy->test.state = MT76_TM_STATE_OFF;77pm->enable = true;78}79out:80mutex_unlock(&dev->mt76.mutex);8182return ret;83}8485static int86mt7925_tm_query(struct mt792x_dev *dev, struct mt7925_tm_cmd *req,87char *evt_resp)88{89struct mt7925_rftest_cmd cmd;90char *pcmd = (char *)&cmd;91struct sk_buff *skb = NULL;92int ret = 1;9394memset(pcmd, 0, sizeof(*pcmd));95memcpy(pcmd + 4, (char *)&req->c, sizeof(struct uni_cmd_testmode_ctrl));9697if (*((uint16_t *)req->padding) == MCU_UNI_CMD_TESTMODE_CTRL)98ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_QUERY(TESTMODE_CTRL),99&cmd, sizeof(cmd), true, &skb);100else if (*((uint16_t *)req->padding) == MCU_UNI_CMD_TESTMODE_RX_STAT)101ret = mt76_mcu_send_and_get_msg(&dev->mt76, MCU_UNI_QUERY(TESTMODE_RX_STAT),102&cmd, sizeof(cmd), true, &skb);103104if (ret)105goto out;106107memcpy((char *)evt_resp, (char *)skb->data + 8, MT7925_EVT_RSP_LEN);108109out:110dev_kfree_skb(skb);111112return ret;113}114115int mt7925_testmode_cmd(struct ieee80211_hw *hw, struct ieee80211_vif *vif,116void *data, int len)117{118struct nlattr *tb[NUM_MT76_TM_ATTRS];119struct mt76_phy *mphy = hw->priv;120struct mt792x_phy *phy = mphy->priv;121int err;122123if (!test_bit(MT76_STATE_RUNNING, &mphy->state) ||124!(hw->conf.flags & IEEE80211_CONF_MONITOR))125return -ENOTCONN;126127err = nla_parse_deprecated(tb, MT76_TM_ATTR_MAX, data, len,128mt76_tm_policy, NULL);129if (err)130return err;131132if (tb[MT76_TM_ATTR_DRV_DATA]) {133struct nlattr *drv_tb[NUM_MT7925_TM_ATTRS], *data;134int ret;135136data = tb[MT76_TM_ATTR_DRV_DATA];137ret = nla_parse_nested_deprecated(drv_tb,138MT7925_TM_ATTR_MAX,139data, mt7925_tm_policy,140NULL);141if (ret)142return ret;143144data = drv_tb[MT7925_TM_ATTR_SET];145if (data)146return mt7925_tm_set(phy->dev, nla_data(data));147}148149return -EINVAL;150}151152int mt7925_testmode_dump(struct ieee80211_hw *hw, struct sk_buff *msg,153struct netlink_callback *cb, void *data, int len)154{155struct nlattr *tb[NUM_MT76_TM_ATTRS];156struct mt76_phy *mphy = hw->priv;157struct mt792x_phy *phy = mphy->priv;158int err;159160if (!test_bit(MT76_STATE_RUNNING, &mphy->state) ||161!(hw->conf.flags & IEEE80211_CONF_MONITOR) ||162!mt76_testmode_enabled(mphy))163return -ENOTCONN;164165if (cb->args[2]++ > 0)166return -ENOENT;167168err = nla_parse_deprecated(tb, MT76_TM_ATTR_MAX, data, len,169mt76_tm_policy, NULL);170if (err)171return err;172173if (tb[MT76_TM_ATTR_DRV_DATA]) {174struct nlattr *drv_tb[NUM_MT7925_TM_ATTRS], *data;175int ret;176177data = tb[MT76_TM_ATTR_DRV_DATA];178ret = nla_parse_nested_deprecated(drv_tb,179MT7925_TM_ATTR_MAX,180data, mt7925_tm_policy,181NULL);182if (ret)183return ret;184185data = drv_tb[MT7925_TM_ATTR_QUERY];186if (data) {187char evt_resp[MT7925_EVT_RSP_LEN];188189err = mt7925_tm_query(phy->dev, nla_data(data),190evt_resp);191if (err)192return err;193194return nla_put(msg, MT7925_TM_ATTR_RSP,195sizeof(evt_resp), evt_resp);196}197}198199return -EINVAL;200}201202203