Commit 6d6dc980 authored by Ryder Lee's avatar Ryder Lee Committed by Felix Fietkau

mt76: mt7915: add implicit Tx beamforming support

Add ht/vht implicit Tx beamforming support and enable it via debugfs.
Tested-by: default avatarShayne Chen <shayne.chen@mediatek.com>
Signed-off-by: default avatarRyder Lee <ryder.lee@mediatek.com>
Signed-off-by: default avatarFelix Fietkau <nbd@nbd.name>
parent 248ac948
...@@ -6,6 +6,32 @@ ...@@ -6,6 +6,32 @@
/** global debugfs **/ /** global debugfs **/
static int
mt7915_implicit_txbf_set(void *data, u64 val)
{
struct mt7915_dev *dev = data;
if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state))
return -EBUSY;
dev->ibf = !!val;
return mt7915_mcu_set_txbf_type(dev);
}
static int
mt7915_implicit_txbf_get(void *data, u64 *val)
{
struct mt7915_dev *dev = data;
*val = dev->ibf;
return 0;
}
DEFINE_DEBUGFS_ATTRIBUTE(fops_implicit_txbf, mt7915_implicit_txbf_get,
mt7915_implicit_txbf_set, "%lld\n");
/* test knob of system layer 1/2 error recovery */ /* test knob of system layer 1/2 error recovery */
static int mt7915_ser_trigger_set(void *data, u64 val) static int mt7915_ser_trigger_set(void *data, u64 val)
{ {
...@@ -355,6 +381,8 @@ int mt7915_init_debugfs(struct mt7915_dev *dev) ...@@ -355,6 +381,8 @@ int mt7915_init_debugfs(struct mt7915_dev *dev)
mt7915_queues_acq); mt7915_queues_acq);
debugfs_create_file("tx_stats", 0400, dir, dev, &fops_tx_stats); debugfs_create_file("tx_stats", 0400, dir, dev, &fops_tx_stats);
debugfs_create_file("fw_debug", 0600, dir, dev, &fops_fw_debug); debugfs_create_file("fw_debug", 0600, dir, dev, &fops_fw_debug);
debugfs_create_file("implicit_txbf", 0600, dir, dev,
&fops_implicit_txbf);
debugfs_create_u32("dfs_hw_pattern", 0400, dir, &dev->hw_pattern); debugfs_create_u32("dfs_hw_pattern", 0400, dir, &dev->hw_pattern);
/* test knobs */ /* test knobs */
debugfs_create_file("radar_trigger", 0200, dir, dev, debugfs_create_file("radar_trigger", 0200, dir, dev,
......
...@@ -169,18 +169,12 @@ static int mt7915_txbf_init(struct mt7915_dev *dev) ...@@ -169,18 +169,12 @@ static int mt7915_txbf_init(struct mt7915_dev *dev)
{ {
int ret; int ret;
/*
* TODO: DBDC & check whether iBF phase calibration data has
* been stored in eeprom offset 0x651~0x7b8, then write down
* 0x1111 into 0x651 and 0x651 to trigger iBF.
*/
/* trigger sounding packets */ /* trigger sounding packets */
ret = mt7915_mcu_set_txbf_sounding(dev); ret = mt7915_mcu_set_txbf_sounding(dev);
if (ret) if (ret)
return ret; return ret;
/* enable iBF & eBF */ /* enable eBF */
return mt7915_mcu_set_txbf_type(dev); return mt7915_mcu_set_txbf_type(dev);
} }
......
...@@ -1729,6 +1729,7 @@ int mt7915_mcu_add_smps(struct mt7915_dev *dev, struct ieee80211_vif *vif, ...@@ -1729,6 +1729,7 @@ int mt7915_mcu_add_smps(struct mt7915_dev *dev, struct ieee80211_vif *vif,
static void static void
mt7915_mcu_sta_sounding_rate(struct sta_rec_bf *bf) mt7915_mcu_sta_sounding_rate(struct sta_rec_bf *bf)
{ {
bf->bf_cap = MT_EBF;
bf->sounding_phy = MT_PHY_TYPE_OFDM; bf->sounding_phy = MT_PHY_TYPE_OFDM;
bf->ndp_rate = 0; /* mcs0 */ bf->ndp_rate = 0; /* mcs0 */
bf->ndpa_rate = MT7915_CFEND_RATE_DEFAULT; /* ofdm 24m */ bf->ndpa_rate = MT7915_CFEND_RATE_DEFAULT; /* ofdm 24m */
...@@ -1736,13 +1737,14 @@ mt7915_mcu_sta_sounding_rate(struct sta_rec_bf *bf) ...@@ -1736,13 +1737,14 @@ mt7915_mcu_sta_sounding_rate(struct sta_rec_bf *bf)
} }
static void static void
mt7915_mcu_sta_bfer_ht(struct ieee80211_sta *sta, struct sta_rec_bf *bf) mt7915_mcu_sta_bfer_ht(struct ieee80211_sta *sta, struct mt7915_phy *phy,
struct sta_rec_bf *bf)
{ {
struct ieee80211_mcs_info *mcs = &sta->ht_cap.mcs; struct ieee80211_mcs_info *mcs = &sta->ht_cap.mcs;
u8 n = 0; u8 n = 0;
bf->tx_mode = MT_PHY_TYPE_HT; bf->tx_mode = MT_PHY_TYPE_HT;
bf->bf_cap |= MT_IBF; bf->bf_cap = MT_IBF;
if (mcs->tx_params & IEEE80211_HT_MCS_TX_RX_DIFF && if (mcs->tx_params & IEEE80211_HT_MCS_TX_RX_DIFF &&
(mcs->tx_params & IEEE80211_HT_MCS_TX_DEFINED)) (mcs->tx_params & IEEE80211_HT_MCS_TX_DEFINED))
...@@ -1755,43 +1757,46 @@ mt7915_mcu_sta_bfer_ht(struct ieee80211_sta *sta, struct sta_rec_bf *bf) ...@@ -1755,43 +1757,46 @@ mt7915_mcu_sta_bfer_ht(struct ieee80211_sta *sta, struct sta_rec_bf *bf)
else if (mcs->rx_mask[1]) else if (mcs->rx_mask[1])
n = 1; n = 1;
bf->nr = hweight8(phy->mt76->chainmask) - 1;
bf->nc = min_t(u8, bf->nr, n); bf->nc = min_t(u8, bf->nr, n);
bf->ibf_ncol = bf->nc; bf->ibf_ncol = n;
if (sta->bandwidth <= IEEE80211_STA_RX_BW_40 && !bf->nc)
bf->ibf_timeout = 0x48;
} }
static void static void
mt7915_mcu_sta_bfer_vht(struct ieee80211_sta *sta, struct mt7915_phy *phy, mt7915_mcu_sta_bfer_vht(struct ieee80211_sta *sta, struct mt7915_phy *phy,
struct sta_rec_bf *bf) struct sta_rec_bf *bf, bool explicit)
{ {
struct ieee80211_sta_vht_cap *pc = &sta->vht_cap; struct ieee80211_sta_vht_cap *pc = &sta->vht_cap;
struct ieee80211_sta_vht_cap *vc = &phy->mt76->sband_5g.sband.vht_cap; struct ieee80211_sta_vht_cap *vc = &phy->mt76->sband_5g.sband.vht_cap;
u8 bfee_nr, bfer_nr, n, tx_ant = hweight8(phy->mt76->chainmask) - 1; u16 mcs_map = le16_to_cpu(pc->vht_mcs.rx_mcs_map);
u16 mcs_map; u8 nss_mcs = mt7915_mcu_get_sta_nss(mcs_map);
u8 tx_ant = hweight8(phy->mt76->chainmask) - 1;
bf->tx_mode = MT_PHY_TYPE_VHT; bf->tx_mode = MT_PHY_TYPE_VHT;
bf->bf_cap |= MT_EBF;
mt7915_mcu_sta_sounding_rate(bf); if (explicit) {
u8 bfee_nr, bfer_nr;
bfee_nr = FIELD_GET(IEEE80211_VHT_CAP_BEAMFORMEE_STS_MASK, mt7915_mcu_sta_sounding_rate(bf);
pc->cap); bfee_nr = FIELD_GET(IEEE80211_VHT_CAP_BEAMFORMEE_STS_MASK,
bfer_nr = FIELD_GET(IEEE80211_VHT_CAP_SOUNDING_DIMENSIONS_MASK, pc->cap);
vc->cap); bfer_nr = FIELD_GET(IEEE80211_VHT_CAP_SOUNDING_DIMENSIONS_MASK,
mcs_map = le16_to_cpu(pc->vht_mcs.rx_mcs_map); vc->cap);
bf->nr = min_t(u8, min_t(u8, bfer_nr, bfee_nr), tx_ant);
bf->nc = min_t(u8, nss_mcs, bf->nr);
bf->ibf_ncol = bf->nc;
n = min_t(u8, bfer_nr, bfee_nr); if (sta->bandwidth == IEEE80211_STA_RX_BW_160)
bf->nr = min_t(u8, n, tx_ant); bf->nr = 1;
n = mt7915_mcu_get_sta_nss(mcs_map); } else {
bf->bf_cap = MT_IBF;
bf->nc = min_t(u8, n, bf->nr); bf->nr = tx_ant;
bf->ibf_ncol = bf->nc; bf->nc = min_t(u8, nss_mcs, bf->nr);
bf->ibf_ncol = nss_mcs;
/* force nr from 4 to 2 */ if (sta->bandwidth == IEEE80211_STA_RX_BW_160)
if (sta->bandwidth == IEEE80211_STA_RX_BW_160) bf->ibf_nrow = 1;
bf->nr = 1; }
} }
static void static void
...@@ -1800,19 +1805,14 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif, ...@@ -1800,19 +1805,14 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif,
{ {
struct ieee80211_sta_he_cap *pc = &sta->he_cap; struct ieee80211_sta_he_cap *pc = &sta->he_cap;
struct ieee80211_he_cap_elem *pe = &pc->he_cap_elem; struct ieee80211_he_cap_elem *pe = &pc->he_cap_elem;
const struct ieee80211_he_cap_elem *ve; const struct ieee80211_sta_he_cap *vc = mt7915_get_he_phy_cap(phy, vif);
const struct ieee80211_sta_he_cap *vc; const struct ieee80211_he_cap_elem *ve = &vc->he_cap_elem;
u8 bfee_nr, bfer_nr, nss_mcs; u16 mcs_map = le16_to_cpu(pc->he_mcs_nss_supp.rx_mcs_80);
u16 mcs_map; u8 nss_mcs = mt7915_mcu_get_sta_nss(mcs_map);
u8 bfee_nr, bfer_nr;
vc = mt7915_get_he_phy_cap(phy, vif);
ve = &vc->he_cap_elem;
bf->tx_mode = MT_PHY_TYPE_HE_SU; bf->tx_mode = MT_PHY_TYPE_HE_SU;
bf->bf_cap |= MT_EBF;
mt7915_mcu_sta_sounding_rate(bf); mt7915_mcu_sta_sounding_rate(bf);
bf->trigger_su = HE_PHY(CAP6_TRIG_SU_BEAMFORMER_FB, bf->trigger_su = HE_PHY(CAP6_TRIG_SU_BEAMFORMER_FB,
pe->phy_cap_info[6]); pe->phy_cap_info[6]);
bf->trigger_mu = HE_PHY(CAP6_TRIG_MU_BEAMFORMER_FB, bf->trigger_mu = HE_PHY(CAP6_TRIG_MU_BEAMFORMER_FB,
...@@ -1821,10 +1821,6 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif, ...@@ -1821,10 +1821,6 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif,
ve->phy_cap_info[5]); ve->phy_cap_info[5]);
bfee_nr = HE_PHY(CAP4_BEAMFORMEE_MAX_STS_UNDER_80MHZ_MASK, bfee_nr = HE_PHY(CAP4_BEAMFORMEE_MAX_STS_UNDER_80MHZ_MASK,
pe->phy_cap_info[4]); pe->phy_cap_info[4]);
mcs_map = le16_to_cpu(pc->he_mcs_nss_supp.tx_mcs_80);
nss_mcs = mt7915_mcu_get_sta_nss(mcs_map);
bf->nr = min_t(u8, bfer_nr, bfee_nr); bf->nr = min_t(u8, bfer_nr, bfee_nr);
bf->nc = min_t(u8, nss_mcs, bf->nr); bf->nc = min_t(u8, nss_mcs, bf->nr);
bf->ibf_ncol = bf->nc; bf->ibf_ncol = bf->nc;
...@@ -1863,7 +1859,7 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif, ...@@ -1863,7 +1859,7 @@ mt7915_mcu_sta_bfer_he(struct ieee80211_sta *sta, struct ieee80211_vif *vif,
static void static void
mt7915_mcu_sta_bfer_tlv(struct sk_buff *skb, struct ieee80211_sta *sta, mt7915_mcu_sta_bfer_tlv(struct sk_buff *skb, struct ieee80211_sta *sta,
struct ieee80211_vif *vif, struct mt7915_phy *phy, struct ieee80211_vif *vif, struct mt7915_phy *phy,
bool enable) bool enable, bool explicit)
{ {
int tx_ant = hweight8(phy->mt76->chainmask) - 1; int tx_ant = hweight8(phy->mt76->chainmask) - 1;
struct sta_rec_bf *bf; struct sta_rec_bf *bf;
...@@ -1885,19 +1881,29 @@ mt7915_mcu_sta_bfer_tlv(struct sk_buff *skb, struct ieee80211_sta *sta, ...@@ -1885,19 +1881,29 @@ mt7915_mcu_sta_bfer_tlv(struct sk_buff *skb, struct ieee80211_sta *sta,
return; return;
} }
/* he: eBF only, in accordance with spec
* vht: support eBF and iBF
* ht: iBF only, since mac80211 lacks of eBF support
*/
if (sta->he_cap.has_he && explicit)
mt7915_mcu_sta_bfer_he(sta, vif, phy, bf);
else if (sta->vht_cap.vht_supported)
mt7915_mcu_sta_bfer_vht(sta, phy, bf, explicit);
else if (sta->ht_cap.ht_supported)
mt7915_mcu_sta_bfer_ht(sta, phy, bf);
else
return;
bf->bw = sta->bandwidth; bf->bw = sta->bandwidth;
bf->ibf_dbw = sta->bandwidth; bf->ibf_dbw = sta->bandwidth;
bf->ibf_nrow = tx_ant; bf->ibf_nrow = tx_ant;
bf->ibf_timeout = 0x18;
if (sta->he_cap.has_he) if (!explicit && sta->bandwidth <= IEEE80211_STA_RX_BW_40 && !bf->nc)
mt7915_mcu_sta_bfer_he(sta, vif, phy, bf); bf->ibf_timeout = 0x48;
else if (sta->vht_cap.vht_supported) else
mt7915_mcu_sta_bfer_vht(sta, phy, bf); bf->ibf_timeout = 0x18;
else if (sta->ht_cap.ht_supported)
mt7915_mcu_sta_bfer_ht(sta, bf);
if (bf->bf_cap & MT_EBF && bf->nr != tx_ant) if (explicit && bf->nr != tx_ant)
bf->mem_20m = matrix[tx_ant][bf->nc]; bf->mem_20m = matrix[tx_ant][bf->nc];
else else
bf->mem_20m = matrix[bf->nr][bf->nc]; bf->mem_20m = matrix[bf->nr][bf->nc];
...@@ -1995,14 +2001,14 @@ mt7915_mcu_add_txbf(struct mt7915_dev *dev, struct ieee80211_vif *vif, ...@@ -1995,14 +2001,14 @@ mt7915_mcu_add_txbf(struct mt7915_dev *dev, struct ieee80211_vif *vif,
/* must keep each tag independent */ /* must keep each tag independent */
/* starec bf */ /* starec bf */
if (ebf) { if (ebf || dev->ibf) {
len = sizeof(struct sta_req_hdr) + sizeof(struct sta_rec_bf); len = sizeof(struct sta_req_hdr) + sizeof(struct sta_rec_bf);
skb = mt7915_mcu_alloc_sta_req(dev, mvif, msta, len); skb = mt7915_mcu_alloc_sta_req(dev, mvif, msta, len);
if (IS_ERR(skb)) if (IS_ERR(skb))
return PTR_ERR(skb); return PTR_ERR(skb);
mt7915_mcu_sta_bfer_tlv(skb, sta, vif, phy, enable); mt7915_mcu_sta_bfer_tlv(skb, sta, vif, phy, enable, ebf);
r = mt76_mcu_skb_send_msg(&dev->mt76, skb, r = mt76_mcu_skb_send_msg(&dev->mt76, skb,
MCU_EXT_CMD_STA_REC_UPDATE, true); MCU_EXT_CMD_STA_REC_UPDATE, true);
...@@ -3427,7 +3433,7 @@ int mt7915_mcu_set_txbf_type(struct mt7915_dev *dev) ...@@ -3427,7 +3433,7 @@ int mt7915_mcu_set_txbf_type(struct mt7915_dev *dev)
} __packed req = { } __packed req = {
.action = MT_BF_TYPE_UPDATE, .action = MT_BF_TYPE_UPDATE,
.ebf = true, .ebf = true,
.ibf = false, .ibf = dev->ibf,
}; };
return mt76_mcu_send_msg(&dev->mt76, MCU_EXT_CMD_TXBF_ACTION, &req, return mt76_mcu_send_msg(&dev->mt76, MCU_EXT_CMD_TXBF_ACTION, &req,
......
...@@ -188,6 +188,7 @@ struct mt7915_dev { ...@@ -188,6 +188,7 @@ struct mt7915_dev {
bool dbdc_support; bool dbdc_support;
bool flash_mode; bool flash_mode;
bool fw_debug; bool fw_debug;
bool ibf;
}; };
enum { enum {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment