18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: ISC 28c2ecf20Sopenharmony_ci/* Copyright (C) 2020 MediaTek Inc. */ 38c2ecf20Sopenharmony_ci 48c2ecf20Sopenharmony_ci#include <linux/etherdevice.h> 58c2ecf20Sopenharmony_ci#include "mt7915.h" 68c2ecf20Sopenharmony_ci#include "mac.h" 78c2ecf20Sopenharmony_ci#include "eeprom.h" 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_cistatic void 108c2ecf20Sopenharmony_cimt7915_mac_init_band(struct mt7915_dev *dev, u8 band) 118c2ecf20Sopenharmony_ci{ 128c2ecf20Sopenharmony_ci u32 mask, set; 138c2ecf20Sopenharmony_ci 148c2ecf20Sopenharmony_ci mt76_rmw_field(dev, MT_TMAC_CTCR0(band), 158c2ecf20Sopenharmony_ci MT_TMAC_CTCR0_INS_DDLMT_REFTIME, 0x3f); 168c2ecf20Sopenharmony_ci mt76_set(dev, MT_TMAC_CTCR0(band), 178c2ecf20Sopenharmony_ci MT_TMAC_CTCR0_INS_DDLMT_VHT_SMPDU_EN | 188c2ecf20Sopenharmony_ci MT_TMAC_CTCR0_INS_DDLMT_EN); 198c2ecf20Sopenharmony_ci 208c2ecf20Sopenharmony_ci mask = MT_MDP_RCFR0_MCU_RX_MGMT | 218c2ecf20Sopenharmony_ci MT_MDP_RCFR0_MCU_RX_CTL_NON_BAR | 228c2ecf20Sopenharmony_ci MT_MDP_RCFR0_MCU_RX_CTL_BAR; 238c2ecf20Sopenharmony_ci set = FIELD_PREP(MT_MDP_RCFR0_MCU_RX_MGMT, MT_MDP_TO_HIF) | 248c2ecf20Sopenharmony_ci FIELD_PREP(MT_MDP_RCFR0_MCU_RX_CTL_NON_BAR, MT_MDP_TO_HIF) | 258c2ecf20Sopenharmony_ci FIELD_PREP(MT_MDP_RCFR0_MCU_RX_CTL_BAR, MT_MDP_TO_HIF); 268c2ecf20Sopenharmony_ci mt76_rmw(dev, MT_MDP_BNRCFR0(band), mask, set); 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci mask = MT_MDP_RCFR1_MCU_RX_BYPASS | 298c2ecf20Sopenharmony_ci MT_MDP_RCFR1_RX_DROPPED_UCAST | 308c2ecf20Sopenharmony_ci MT_MDP_RCFR1_RX_DROPPED_MCAST; 318c2ecf20Sopenharmony_ci set = FIELD_PREP(MT_MDP_RCFR1_MCU_RX_BYPASS, MT_MDP_TO_HIF) | 328c2ecf20Sopenharmony_ci FIELD_PREP(MT_MDP_RCFR1_RX_DROPPED_UCAST, MT_MDP_TO_HIF) | 338c2ecf20Sopenharmony_ci FIELD_PREP(MT_MDP_RCFR1_RX_DROPPED_MCAST, MT_MDP_TO_HIF); 348c2ecf20Sopenharmony_ci mt76_rmw(dev, MT_MDP_BNRCFR1(band), mask, set); 358c2ecf20Sopenharmony_ci 368c2ecf20Sopenharmony_ci mt76_set(dev, MT_WF_RMAC_MIB_TIME0(band), MT_WF_RMAC_MIB_RXTIME_EN); 378c2ecf20Sopenharmony_ci mt76_set(dev, MT_WF_RMAC_MIB_AIRTIME0(band), MT_WF_RMAC_MIB_RXTIME_EN); 388c2ecf20Sopenharmony_ci} 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_cistatic void mt7915_mac_init(struct mt7915_dev *dev) 418c2ecf20Sopenharmony_ci{ 428c2ecf20Sopenharmony_ci int i; 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_ci mt76_rmw_field(dev, MT_DMA_DCR0, MT_DMA_DCR0_MAX_RX_LEN, 1536); 458c2ecf20Sopenharmony_ci mt76_rmw_field(dev, MT_MDP_DCR1, MT_MDP_DCR1_MAX_RX_LEN, 1536); 468c2ecf20Sopenharmony_ci /* enable rx rate report */ 478c2ecf20Sopenharmony_ci mt76_set(dev, MT_DMA_DCR0, MT_DMA_DCR0_RXD_G5_EN); 488c2ecf20Sopenharmony_ci /* disable hardware de-agg */ 498c2ecf20Sopenharmony_ci mt76_clear(dev, MT_MDP_DCR0, MT_MDP_DCR0_DAMSDU_EN); 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci for (i = 0; i < MT7915_WTBL_SIZE; i++) 528c2ecf20Sopenharmony_ci mt7915_mac_wtbl_update(dev, i, 538c2ecf20Sopenharmony_ci MT_WTBL_UPDATE_ADM_COUNT_CLEAR); 548c2ecf20Sopenharmony_ci 558c2ecf20Sopenharmony_ci mt7915_mac_init_band(dev, 0); 568c2ecf20Sopenharmony_ci mt7915_mac_init_band(dev, 1); 578c2ecf20Sopenharmony_ci mt7915_mcu_set_rts_thresh(&dev->phy, 0x92b); 588c2ecf20Sopenharmony_ci} 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_cistatic int mt7915_txbf_init(struct mt7915_dev *dev) 618c2ecf20Sopenharmony_ci{ 628c2ecf20Sopenharmony_ci int ret; 638c2ecf20Sopenharmony_ci 648c2ecf20Sopenharmony_ci /* 658c2ecf20Sopenharmony_ci * TODO: DBDC & check whether iBF phase calibration data has 668c2ecf20Sopenharmony_ci * been stored in eeprom offset 0x651~0x7b8, then write down 678c2ecf20Sopenharmony_ci * 0x1111 into 0x651 and 0x651 to trigger iBF. 688c2ecf20Sopenharmony_ci */ 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_ci /* trigger sounding packets */ 718c2ecf20Sopenharmony_ci ret = mt7915_mcu_set_txbf_sounding(dev); 728c2ecf20Sopenharmony_ci if (ret) 738c2ecf20Sopenharmony_ci return ret; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci /* enable iBF & eBF */ 768c2ecf20Sopenharmony_ci return mt7915_mcu_set_txbf_type(dev); 778c2ecf20Sopenharmony_ci} 788c2ecf20Sopenharmony_ci 798c2ecf20Sopenharmony_cistatic void 808c2ecf20Sopenharmony_cimt7915_init_txpower_band(struct mt7915_dev *dev, 818c2ecf20Sopenharmony_ci struct ieee80211_supported_band *sband) 828c2ecf20Sopenharmony_ci{ 838c2ecf20Sopenharmony_ci int i, n_chains = hweight8(dev->mphy.antenna_mask); 848c2ecf20Sopenharmony_ci 858c2ecf20Sopenharmony_ci for (i = 0; i < sband->n_channels; i++) { 868c2ecf20Sopenharmony_ci struct ieee80211_channel *chan = &sband->channels[i]; 878c2ecf20Sopenharmony_ci u32 target_power = 0; 888c2ecf20Sopenharmony_ci int j; 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci for (j = 0; j < n_chains; j++) { 918c2ecf20Sopenharmony_ci u32 val; 928c2ecf20Sopenharmony_ci 938c2ecf20Sopenharmony_ci val = mt7915_eeprom_get_target_power(dev, chan, j); 948c2ecf20Sopenharmony_ci target_power = max(target_power, val); 958c2ecf20Sopenharmony_ci } 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci chan->max_power = min_t(int, chan->max_reg_power, 988c2ecf20Sopenharmony_ci target_power / 2); 998c2ecf20Sopenharmony_ci chan->orig_mpwr = target_power / 2; 1008c2ecf20Sopenharmony_ci } 1018c2ecf20Sopenharmony_ci} 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_cistatic void mt7915_init_txpower(struct mt7915_dev *dev) 1048c2ecf20Sopenharmony_ci{ 1058c2ecf20Sopenharmony_ci mt7915_init_txpower_band(dev, &dev->mphy.sband_2g.sband); 1068c2ecf20Sopenharmony_ci mt7915_init_txpower_band(dev, &dev->mphy.sband_5g.sband); 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci mt7915_eeprom_init_sku(dev); 1098c2ecf20Sopenharmony_ci} 1108c2ecf20Sopenharmony_ci 1118c2ecf20Sopenharmony_cistatic void mt7915_init_work(struct work_struct *work) 1128c2ecf20Sopenharmony_ci{ 1138c2ecf20Sopenharmony_ci struct mt7915_dev *dev = container_of(work, struct mt7915_dev, 1148c2ecf20Sopenharmony_ci init_work); 1158c2ecf20Sopenharmony_ci 1168c2ecf20Sopenharmony_ci mt7915_mcu_set_eeprom(dev); 1178c2ecf20Sopenharmony_ci mt7915_mac_init(dev); 1188c2ecf20Sopenharmony_ci mt7915_init_txpower(dev); 1198c2ecf20Sopenharmony_ci mt7915_txbf_init(dev); 1208c2ecf20Sopenharmony_ci} 1218c2ecf20Sopenharmony_ci 1228c2ecf20Sopenharmony_cistatic int mt7915_init_hardware(struct mt7915_dev *dev) 1238c2ecf20Sopenharmony_ci{ 1248c2ecf20Sopenharmony_ci int ret, idx; 1258c2ecf20Sopenharmony_ci 1268c2ecf20Sopenharmony_ci mt76_wr(dev, MT_INT_SOURCE_CSR, ~0); 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_ci INIT_WORK(&dev->init_work, mt7915_init_work); 1298c2ecf20Sopenharmony_ci spin_lock_init(&dev->token_lock); 1308c2ecf20Sopenharmony_ci idr_init(&dev->token); 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci ret = mt7915_dma_init(dev); 1338c2ecf20Sopenharmony_ci if (ret) 1348c2ecf20Sopenharmony_ci return ret; 1358c2ecf20Sopenharmony_ci 1368c2ecf20Sopenharmony_ci set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state); 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci /* 1398c2ecf20Sopenharmony_ci * force firmware operation mode into normal state, 1408c2ecf20Sopenharmony_ci * which should be set before firmware download stage. 1418c2ecf20Sopenharmony_ci */ 1428c2ecf20Sopenharmony_ci mt76_wr(dev, MT_SWDEF_MODE, MT_SWDEF_NORMAL_MODE); 1438c2ecf20Sopenharmony_ci 1448c2ecf20Sopenharmony_ci ret = mt7915_mcu_init(dev); 1458c2ecf20Sopenharmony_ci if (ret) 1468c2ecf20Sopenharmony_ci return ret; 1478c2ecf20Sopenharmony_ci 1488c2ecf20Sopenharmony_ci ret = mt7915_eeprom_init(dev); 1498c2ecf20Sopenharmony_ci if (ret < 0) 1508c2ecf20Sopenharmony_ci return ret; 1518c2ecf20Sopenharmony_ci 1528c2ecf20Sopenharmony_ci /* Beacon and mgmt frames should occupy wcid 0 */ 1538c2ecf20Sopenharmony_ci idx = mt76_wcid_alloc(dev->mt76.wcid_mask, MT7915_WTBL_STA - 1); 1548c2ecf20Sopenharmony_ci if (idx) 1558c2ecf20Sopenharmony_ci return -ENOSPC; 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_ci dev->mt76.global_wcid.idx = idx; 1588c2ecf20Sopenharmony_ci dev->mt76.global_wcid.hw_key_idx = -1; 1598c2ecf20Sopenharmony_ci dev->mt76.global_wcid.tx_info |= MT_WCID_TX_INFO_SET; 1608c2ecf20Sopenharmony_ci rcu_assign_pointer(dev->mt76.wcid[idx], &dev->mt76.global_wcid); 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_ci return 0; 1638c2ecf20Sopenharmony_ci} 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci#define CCK_RATE(_idx, _rate) { \ 1668c2ecf20Sopenharmony_ci .bitrate = _rate, \ 1678c2ecf20Sopenharmony_ci .flags = IEEE80211_RATE_SHORT_PREAMBLE, \ 1688c2ecf20Sopenharmony_ci .hw_value = (MT_PHY_TYPE_CCK << 8) | (_idx), \ 1698c2ecf20Sopenharmony_ci .hw_value_short = (MT_PHY_TYPE_CCK << 8) | (4 + (_idx)), \ 1708c2ecf20Sopenharmony_ci} 1718c2ecf20Sopenharmony_ci 1728c2ecf20Sopenharmony_ci#define OFDM_RATE(_idx, _rate) { \ 1738c2ecf20Sopenharmony_ci .bitrate = _rate, \ 1748c2ecf20Sopenharmony_ci .hw_value = (MT_PHY_TYPE_OFDM << 8) | (_idx), \ 1758c2ecf20Sopenharmony_ci .hw_value_short = (MT_PHY_TYPE_OFDM << 8) | (_idx), \ 1768c2ecf20Sopenharmony_ci} 1778c2ecf20Sopenharmony_ci 1788c2ecf20Sopenharmony_cistatic struct ieee80211_rate mt7915_rates[] = { 1798c2ecf20Sopenharmony_ci CCK_RATE(0, 10), 1808c2ecf20Sopenharmony_ci CCK_RATE(1, 20), 1818c2ecf20Sopenharmony_ci CCK_RATE(2, 55), 1828c2ecf20Sopenharmony_ci CCK_RATE(3, 110), 1838c2ecf20Sopenharmony_ci OFDM_RATE(11, 60), 1848c2ecf20Sopenharmony_ci OFDM_RATE(15, 90), 1858c2ecf20Sopenharmony_ci OFDM_RATE(10, 120), 1868c2ecf20Sopenharmony_ci OFDM_RATE(14, 180), 1878c2ecf20Sopenharmony_ci OFDM_RATE(9, 240), 1888c2ecf20Sopenharmony_ci OFDM_RATE(13, 360), 1898c2ecf20Sopenharmony_ci OFDM_RATE(8, 480), 1908c2ecf20Sopenharmony_ci OFDM_RATE(12, 540), 1918c2ecf20Sopenharmony_ci}; 1928c2ecf20Sopenharmony_ci 1938c2ecf20Sopenharmony_cistatic const struct ieee80211_iface_limit if_limits[] = { 1948c2ecf20Sopenharmony_ci { 1958c2ecf20Sopenharmony_ci .max = 1, 1968c2ecf20Sopenharmony_ci .types = BIT(NL80211_IFTYPE_ADHOC) 1978c2ecf20Sopenharmony_ci }, { 1988c2ecf20Sopenharmony_ci .max = MT7915_MAX_INTERFACES, 1998c2ecf20Sopenharmony_ci .types = BIT(NL80211_IFTYPE_AP) | 2008c2ecf20Sopenharmony_ci#ifdef CONFIG_MAC80211_MESH 2018c2ecf20Sopenharmony_ci BIT(NL80211_IFTYPE_MESH_POINT) | 2028c2ecf20Sopenharmony_ci#endif 2038c2ecf20Sopenharmony_ci BIT(NL80211_IFTYPE_STATION) 2048c2ecf20Sopenharmony_ci } 2058c2ecf20Sopenharmony_ci}; 2068c2ecf20Sopenharmony_ci 2078c2ecf20Sopenharmony_cistatic const struct ieee80211_iface_combination if_comb[] = { 2088c2ecf20Sopenharmony_ci { 2098c2ecf20Sopenharmony_ci .limits = if_limits, 2108c2ecf20Sopenharmony_ci .n_limits = ARRAY_SIZE(if_limits), 2118c2ecf20Sopenharmony_ci .max_interfaces = 4, 2128c2ecf20Sopenharmony_ci .num_different_channels = 1, 2138c2ecf20Sopenharmony_ci .beacon_int_infra_match = true, 2148c2ecf20Sopenharmony_ci .radar_detect_widths = BIT(NL80211_CHAN_WIDTH_20_NOHT) | 2158c2ecf20Sopenharmony_ci BIT(NL80211_CHAN_WIDTH_20) | 2168c2ecf20Sopenharmony_ci BIT(NL80211_CHAN_WIDTH_40) | 2178c2ecf20Sopenharmony_ci BIT(NL80211_CHAN_WIDTH_80) | 2188c2ecf20Sopenharmony_ci BIT(NL80211_CHAN_WIDTH_160) | 2198c2ecf20Sopenharmony_ci BIT(NL80211_CHAN_WIDTH_80P80), 2208c2ecf20Sopenharmony_ci } 2218c2ecf20Sopenharmony_ci}; 2228c2ecf20Sopenharmony_ci 2238c2ecf20Sopenharmony_cistatic void 2248c2ecf20Sopenharmony_cimt7915_regd_notifier(struct wiphy *wiphy, 2258c2ecf20Sopenharmony_ci struct regulatory_request *request) 2268c2ecf20Sopenharmony_ci{ 2278c2ecf20Sopenharmony_ci struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy); 2288c2ecf20Sopenharmony_ci struct mt7915_dev *dev = mt7915_hw_dev(hw); 2298c2ecf20Sopenharmony_ci struct mt76_phy *mphy = hw->priv; 2308c2ecf20Sopenharmony_ci struct mt7915_phy *phy = mphy->priv; 2318c2ecf20Sopenharmony_ci struct cfg80211_chan_def *chandef = &mphy->chandef; 2328c2ecf20Sopenharmony_ci 2338c2ecf20Sopenharmony_ci dev->mt76.region = request->dfs_region; 2348c2ecf20Sopenharmony_ci 2358c2ecf20Sopenharmony_ci if (!(chandef->chan->flags & IEEE80211_CHAN_RADAR)) 2368c2ecf20Sopenharmony_ci return; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci mt7915_dfs_init_radar_detector(phy); 2398c2ecf20Sopenharmony_ci} 2408c2ecf20Sopenharmony_ci 2418c2ecf20Sopenharmony_cistatic void 2428c2ecf20Sopenharmony_cimt7915_init_wiphy(struct ieee80211_hw *hw) 2438c2ecf20Sopenharmony_ci{ 2448c2ecf20Sopenharmony_ci struct mt7915_phy *phy = mt7915_hw_phy(hw); 2458c2ecf20Sopenharmony_ci struct wiphy *wiphy = hw->wiphy; 2468c2ecf20Sopenharmony_ci 2478c2ecf20Sopenharmony_ci hw->queues = 4; 2488c2ecf20Sopenharmony_ci hw->max_rx_aggregation_subframes = IEEE80211_MAX_AMPDU_BUF; 2498c2ecf20Sopenharmony_ci hw->max_tx_aggregation_subframes = IEEE80211_MAX_AMPDU_BUF; 2508c2ecf20Sopenharmony_ci 2518c2ecf20Sopenharmony_ci phy->slottime = 9; 2528c2ecf20Sopenharmony_ci 2538c2ecf20Sopenharmony_ci hw->sta_data_size = sizeof(struct mt7915_sta); 2548c2ecf20Sopenharmony_ci hw->vif_data_size = sizeof(struct mt7915_vif); 2558c2ecf20Sopenharmony_ci 2568c2ecf20Sopenharmony_ci wiphy->iface_combinations = if_comb; 2578c2ecf20Sopenharmony_ci wiphy->n_iface_combinations = ARRAY_SIZE(if_comb); 2588c2ecf20Sopenharmony_ci wiphy->reg_notifier = mt7915_regd_notifier; 2598c2ecf20Sopenharmony_ci wiphy->flags |= WIPHY_FLAG_HAS_CHANNEL_SWITCH; 2608c2ecf20Sopenharmony_ci 2618c2ecf20Sopenharmony_ci wiphy_ext_feature_set(wiphy, NL80211_EXT_FEATURE_VHT_IBSS); 2628c2ecf20Sopenharmony_ci 2638c2ecf20Sopenharmony_ci ieee80211_hw_set(hw, HAS_RATE_CONTROL); 2648c2ecf20Sopenharmony_ci 2658c2ecf20Sopenharmony_ci hw->max_tx_fragments = 4; 2668c2ecf20Sopenharmony_ci} 2678c2ecf20Sopenharmony_ci 2688c2ecf20Sopenharmony_civoid mt7915_set_stream_vht_txbf_caps(struct mt7915_phy *phy) 2698c2ecf20Sopenharmony_ci{ 2708c2ecf20Sopenharmony_ci int nss = hweight8(phy->chainmask); 2718c2ecf20Sopenharmony_ci u32 *cap = &phy->mt76->sband_5g.sband.vht_cap.cap; 2728c2ecf20Sopenharmony_ci 2738c2ecf20Sopenharmony_ci *cap |= IEEE80211_VHT_CAP_SU_BEAMFORMEE_CAPABLE | 2748c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_MU_BEAMFORMEE_CAPABLE | 2758c2ecf20Sopenharmony_ci (3 << IEEE80211_VHT_CAP_BEAMFORMEE_STS_SHIFT); 2768c2ecf20Sopenharmony_ci 2778c2ecf20Sopenharmony_ci *cap &= ~(IEEE80211_VHT_CAP_SOUNDING_DIMENSIONS_MASK | 2788c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_SU_BEAMFORMER_CAPABLE | 2798c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_MU_BEAMFORMER_CAPABLE); 2808c2ecf20Sopenharmony_ci 2818c2ecf20Sopenharmony_ci if (nss < 2) 2828c2ecf20Sopenharmony_ci return; 2838c2ecf20Sopenharmony_ci 2848c2ecf20Sopenharmony_ci *cap |= IEEE80211_VHT_CAP_SU_BEAMFORMER_CAPABLE | 2858c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_MU_BEAMFORMER_CAPABLE | 2868c2ecf20Sopenharmony_ci FIELD_PREP(IEEE80211_VHT_CAP_SOUNDING_DIMENSIONS_MASK, 2878c2ecf20Sopenharmony_ci nss - 1); 2888c2ecf20Sopenharmony_ci} 2898c2ecf20Sopenharmony_ci 2908c2ecf20Sopenharmony_cistatic void 2918c2ecf20Sopenharmony_cimt7915_set_stream_he_txbf_caps(struct ieee80211_sta_he_cap *he_cap, 2928c2ecf20Sopenharmony_ci int vif, int nss) 2938c2ecf20Sopenharmony_ci{ 2948c2ecf20Sopenharmony_ci struct ieee80211_he_cap_elem *elem = &he_cap->he_cap_elem; 2958c2ecf20Sopenharmony_ci struct ieee80211_he_mcs_nss_supp *mcs = &he_cap->he_mcs_nss_supp; 2968c2ecf20Sopenharmony_ci u8 c; 2978c2ecf20Sopenharmony_ci 2988c2ecf20Sopenharmony_ci#ifdef CONFIG_MAC80211_MESH 2998c2ecf20Sopenharmony_ci if (vif == NL80211_IFTYPE_MESH_POINT) 3008c2ecf20Sopenharmony_ci return; 3018c2ecf20Sopenharmony_ci#endif 3028c2ecf20Sopenharmony_ci 3038c2ecf20Sopenharmony_ci elem->phy_cap_info[3] &= ~IEEE80211_HE_PHY_CAP3_SU_BEAMFORMER; 3048c2ecf20Sopenharmony_ci elem->phy_cap_info[4] &= ~IEEE80211_HE_PHY_CAP4_MU_BEAMFORMER; 3058c2ecf20Sopenharmony_ci 3068c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP5_BEAMFORMEE_NUM_SND_DIM_UNDER_80MHZ_MASK | 3078c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP5_BEAMFORMEE_NUM_SND_DIM_ABOVE_80MHZ_MASK; 3088c2ecf20Sopenharmony_ci elem->phy_cap_info[5] &= ~c; 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP6_TRIG_SU_BEAMFORMER_FB | 3118c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_TRIG_MU_BEAMFORMER_FB; 3128c2ecf20Sopenharmony_ci elem->phy_cap_info[6] &= ~c; 3138c2ecf20Sopenharmony_ci 3148c2ecf20Sopenharmony_ci elem->phy_cap_info[7] &= ~IEEE80211_HE_PHY_CAP7_MAX_NC_MASK; 3158c2ecf20Sopenharmony_ci 3168c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP2_NDP_4x_LTF_AND_3_2US | 3178c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP2_UL_MU_FULL_MU_MIMO | 3188c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP2_UL_MU_PARTIAL_MU_MIMO; 3198c2ecf20Sopenharmony_ci elem->phy_cap_info[2] |= c; 3208c2ecf20Sopenharmony_ci 3218c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP4_SU_BEAMFORMEE | 3228c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP4_BEAMFORMEE_MAX_STS_UNDER_80MHZ_4 | 3238c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP4_BEAMFORMEE_MAX_STS_ABOVE_80MHZ_4; 3248c2ecf20Sopenharmony_ci elem->phy_cap_info[4] |= c; 3258c2ecf20Sopenharmony_ci 3268c2ecf20Sopenharmony_ci /* do not support NG16 due to spec D4.0 changes subcarrier idx */ 3278c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP6_CODEBOOK_SIZE_42_SU | 3288c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_CODEBOOK_SIZE_75_MU; 3298c2ecf20Sopenharmony_ci 3308c2ecf20Sopenharmony_ci if (vif == NL80211_IFTYPE_STATION) 3318c2ecf20Sopenharmony_ci c |= IEEE80211_HE_PHY_CAP6_PARTIAL_BANDWIDTH_DL_MUMIMO; 3328c2ecf20Sopenharmony_ci 3338c2ecf20Sopenharmony_ci elem->phy_cap_info[6] |= c; 3348c2ecf20Sopenharmony_ci 3358c2ecf20Sopenharmony_ci if (nss < 2) 3368c2ecf20Sopenharmony_ci return; 3378c2ecf20Sopenharmony_ci 3388c2ecf20Sopenharmony_ci /* the maximum cap is 4 x 3, (Nr, Nc) = (3, 2) */ 3398c2ecf20Sopenharmony_ci elem->phy_cap_info[7] |= min_t(int, nss - 1, 2) << 3; 3408c2ecf20Sopenharmony_ci 3418c2ecf20Sopenharmony_ci if (vif != NL80211_IFTYPE_AP) 3428c2ecf20Sopenharmony_ci return; 3438c2ecf20Sopenharmony_ci 3448c2ecf20Sopenharmony_ci elem->phy_cap_info[3] |= IEEE80211_HE_PHY_CAP3_SU_BEAMFORMER; 3458c2ecf20Sopenharmony_ci elem->phy_cap_info[4] |= IEEE80211_HE_PHY_CAP4_MU_BEAMFORMER; 3468c2ecf20Sopenharmony_ci 3478c2ecf20Sopenharmony_ci /* num_snd_dim */ 3488c2ecf20Sopenharmony_ci c = (nss - 1) | (max_t(int, mcs->tx_mcs_160, 1) << 3); 3498c2ecf20Sopenharmony_ci elem->phy_cap_info[5] |= c; 3508c2ecf20Sopenharmony_ci 3518c2ecf20Sopenharmony_ci c = IEEE80211_HE_PHY_CAP6_TRIG_SU_BEAMFORMER_FB | 3528c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_TRIG_MU_BEAMFORMER_FB; 3538c2ecf20Sopenharmony_ci elem->phy_cap_info[6] |= c; 3548c2ecf20Sopenharmony_ci} 3558c2ecf20Sopenharmony_ci 3568c2ecf20Sopenharmony_cistatic void 3578c2ecf20Sopenharmony_cimt7915_gen_ppe_thresh(u8 *he_ppet) 3588c2ecf20Sopenharmony_ci{ 3598c2ecf20Sopenharmony_ci int ru, nss, max_nss = 1, max_ru = 3; 3608c2ecf20Sopenharmony_ci u8 bit = 7, ru_bit_mask = 0x7; 3618c2ecf20Sopenharmony_ci u8 ppet16_ppet8_ru3_ru0[] = {0x1c, 0xc7, 0x71}; 3628c2ecf20Sopenharmony_ci 3638c2ecf20Sopenharmony_ci he_ppet[0] = max_nss & IEEE80211_PPE_THRES_NSS_MASK; 3648c2ecf20Sopenharmony_ci he_ppet[0] |= (ru_bit_mask << 3658c2ecf20Sopenharmony_ci IEEE80211_PPE_THRES_RU_INDEX_BITMASK_POS) & 3668c2ecf20Sopenharmony_ci IEEE80211_PPE_THRES_RU_INDEX_BITMASK_MASK; 3678c2ecf20Sopenharmony_ci 3688c2ecf20Sopenharmony_ci for (nss = 0; nss <= max_nss; nss++) { 3698c2ecf20Sopenharmony_ci for (ru = 0; ru < max_ru; ru++) { 3708c2ecf20Sopenharmony_ci u8 val; 3718c2ecf20Sopenharmony_ci int i; 3728c2ecf20Sopenharmony_ci 3738c2ecf20Sopenharmony_ci if (!(ru_bit_mask & BIT(ru))) 3748c2ecf20Sopenharmony_ci continue; 3758c2ecf20Sopenharmony_ci 3768c2ecf20Sopenharmony_ci val = (ppet16_ppet8_ru3_ru0[nss] >> (ru * 6)) & 3778c2ecf20Sopenharmony_ci 0x3f; 3788c2ecf20Sopenharmony_ci val = ((val >> 3) & 0x7) | ((val & 0x7) << 3); 3798c2ecf20Sopenharmony_ci for (i = 5; i >= 0; i--) { 3808c2ecf20Sopenharmony_ci he_ppet[bit / 8] |= 3818c2ecf20Sopenharmony_ci ((val >> i) & 0x1) << ((bit % 8)); 3828c2ecf20Sopenharmony_ci bit++; 3838c2ecf20Sopenharmony_ci } 3848c2ecf20Sopenharmony_ci } 3858c2ecf20Sopenharmony_ci } 3868c2ecf20Sopenharmony_ci} 3878c2ecf20Sopenharmony_ci 3888c2ecf20Sopenharmony_cistatic int 3898c2ecf20Sopenharmony_cimt7915_init_he_caps(struct mt7915_phy *phy, enum nl80211_band band, 3908c2ecf20Sopenharmony_ci struct ieee80211_sband_iftype_data *data) 3918c2ecf20Sopenharmony_ci{ 3928c2ecf20Sopenharmony_ci int i, idx = 0; 3938c2ecf20Sopenharmony_ci int nss = hweight8(phy->chainmask); 3948c2ecf20Sopenharmony_ci u16 mcs_map = 0; 3958c2ecf20Sopenharmony_ci 3968c2ecf20Sopenharmony_ci for (i = 0; i < 8; i++) { 3978c2ecf20Sopenharmony_ci if (i < nss) 3988c2ecf20Sopenharmony_ci mcs_map |= (IEEE80211_HE_MCS_SUPPORT_0_11 << (i * 2)); 3998c2ecf20Sopenharmony_ci else 4008c2ecf20Sopenharmony_ci mcs_map |= (IEEE80211_HE_MCS_NOT_SUPPORTED << (i * 2)); 4018c2ecf20Sopenharmony_ci } 4028c2ecf20Sopenharmony_ci 4038c2ecf20Sopenharmony_ci for (i = 0; i < NUM_NL80211_IFTYPES; i++) { 4048c2ecf20Sopenharmony_ci struct ieee80211_sta_he_cap *he_cap = &data[idx].he_cap; 4058c2ecf20Sopenharmony_ci struct ieee80211_he_cap_elem *he_cap_elem = 4068c2ecf20Sopenharmony_ci &he_cap->he_cap_elem; 4078c2ecf20Sopenharmony_ci struct ieee80211_he_mcs_nss_supp *he_mcs = 4088c2ecf20Sopenharmony_ci &he_cap->he_mcs_nss_supp; 4098c2ecf20Sopenharmony_ci 4108c2ecf20Sopenharmony_ci switch (i) { 4118c2ecf20Sopenharmony_ci case NL80211_IFTYPE_STATION: 4128c2ecf20Sopenharmony_ci case NL80211_IFTYPE_AP: 4138c2ecf20Sopenharmony_ci#ifdef CONFIG_MAC80211_MESH 4148c2ecf20Sopenharmony_ci case NL80211_IFTYPE_MESH_POINT: 4158c2ecf20Sopenharmony_ci#endif 4168c2ecf20Sopenharmony_ci break; 4178c2ecf20Sopenharmony_ci default: 4188c2ecf20Sopenharmony_ci continue; 4198c2ecf20Sopenharmony_ci } 4208c2ecf20Sopenharmony_ci 4218c2ecf20Sopenharmony_ci data[idx].types_mask = BIT(i); 4228c2ecf20Sopenharmony_ci he_cap->has_he = true; 4238c2ecf20Sopenharmony_ci 4248c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[0] = 4258c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP0_HTC_HE; 4268c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[3] = 4278c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP3_OMI_CONTROL | 4288c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP3_MAX_AMPDU_LEN_EXP_RESERVED; 4298c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[4] = 4308c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP4_AMDSU_IN_AMPDU; 4318c2ecf20Sopenharmony_ci 4328c2ecf20Sopenharmony_ci if (band == NL80211_BAND_2GHZ) 4338c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[0] = 4348c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_IN_2G; 4358c2ecf20Sopenharmony_ci else if (band == NL80211_BAND_5GHZ) 4368c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[0] = 4378c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_40MHZ_80MHZ_IN_5G | 4388c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_160MHZ_IN_5G | 4398c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_80PLUS80_MHZ_IN_5G; 4408c2ecf20Sopenharmony_ci 4418c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[1] = 4428c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP1_LDPC_CODING_IN_PAYLOAD; 4438c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[2] = 4448c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP2_STBC_TX_UNDER_80MHZ | 4458c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP2_STBC_RX_UNDER_80MHZ; 4468c2ecf20Sopenharmony_ci 4478c2ecf20Sopenharmony_ci switch (i) { 4488c2ecf20Sopenharmony_ci case NL80211_IFTYPE_AP: 4498c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[0] |= 4508c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP0_TWT_RES; 4518c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[2] |= 4528c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP2_BSR; 4538c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[4] |= 4548c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP4_BQR; 4558c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[5] |= 4568c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP5_OM_CTRL_UL_MU_DATA_DIS_RX; 4578c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[3] |= 4588c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_TX_QPSK | 4598c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_RX_QPSK; 4608c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[6] |= 4618c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_PPE_THRESHOLD_PRESENT; 4628c2ecf20Sopenharmony_ci break; 4638c2ecf20Sopenharmony_ci case NL80211_IFTYPE_STATION: 4648c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[0] |= 4658c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP0_TWT_REQ; 4668c2ecf20Sopenharmony_ci he_cap_elem->mac_cap_info[1] |= 4678c2ecf20Sopenharmony_ci IEEE80211_HE_MAC_CAP1_TF_MAC_PAD_DUR_16US; 4688c2ecf20Sopenharmony_ci 4698c2ecf20Sopenharmony_ci if (band == NL80211_BAND_2GHZ) 4708c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[0] |= 4718c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_RU_MAPPING_IN_2G; 4728c2ecf20Sopenharmony_ci else if (band == NL80211_BAND_5GHZ) 4738c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[0] |= 4748c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP0_CHANNEL_WIDTH_SET_RU_MAPPING_IN_5G; 4758c2ecf20Sopenharmony_ci 4768c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[1] |= 4778c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP1_DEVICE_CLASS_A | 4788c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP1_HE_LTF_AND_GI_FOR_HE_PPDUS_0_8US; 4798c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[3] |= 4808c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_TX_QPSK | 4818c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP3_DCM_MAX_CONST_RX_QPSK; 4828c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[6] |= 4838c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_TRIG_CQI_FB | 4848c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_PARTIAL_BW_EXT_RANGE | 4858c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_PPE_THRESHOLD_PRESENT; 4868c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[7] |= 4878c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP7_POWER_BOOST_FACTOR_AR | 4888c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP7_HE_SU_MU_PPDU_4XLTF_AND_08_US_GI; 4898c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[8] |= 4908c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP8_20MHZ_IN_40MHZ_HE_PPDU_IN_2G | 4918c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP8_20MHZ_IN_160MHZ_HE_PPDU | 4928c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP8_80MHZ_IN_160MHZ_HE_PPDU | 4938c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP8_DCM_MAX_RU_484; 4948c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[9] |= 4958c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_LONGER_THAN_16_SIGB_OFDM_SYM | 4968c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_NON_TRIGGERED_CQI_FEEDBACK | 4978c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_TX_1024_QAM_LESS_THAN_242_TONE_RU | 4988c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_RX_1024_QAM_LESS_THAN_242_TONE_RU | 4998c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_RX_FULL_BW_SU_USING_MU_WITH_COMP_SIGB | 5008c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_RX_FULL_BW_SU_USING_MU_WITH_NON_COMP_SIGB; 5018c2ecf20Sopenharmony_ci break; 5028c2ecf20Sopenharmony_ci } 5038c2ecf20Sopenharmony_ci 5048c2ecf20Sopenharmony_ci he_mcs->rx_mcs_80 = cpu_to_le16(mcs_map); 5058c2ecf20Sopenharmony_ci he_mcs->tx_mcs_80 = cpu_to_le16(mcs_map); 5068c2ecf20Sopenharmony_ci he_mcs->rx_mcs_160 = cpu_to_le16(mcs_map); 5078c2ecf20Sopenharmony_ci he_mcs->tx_mcs_160 = cpu_to_le16(mcs_map); 5088c2ecf20Sopenharmony_ci he_mcs->rx_mcs_80p80 = cpu_to_le16(mcs_map); 5098c2ecf20Sopenharmony_ci he_mcs->tx_mcs_80p80 = cpu_to_le16(mcs_map); 5108c2ecf20Sopenharmony_ci 5118c2ecf20Sopenharmony_ci mt7915_set_stream_he_txbf_caps(he_cap, i, nss); 5128c2ecf20Sopenharmony_ci 5138c2ecf20Sopenharmony_ci memset(he_cap->ppe_thres, 0, sizeof(he_cap->ppe_thres)); 5148c2ecf20Sopenharmony_ci if (he_cap_elem->phy_cap_info[6] & 5158c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP6_PPE_THRESHOLD_PRESENT) { 5168c2ecf20Sopenharmony_ci mt7915_gen_ppe_thresh(he_cap->ppe_thres); 5178c2ecf20Sopenharmony_ci } else { 5188c2ecf20Sopenharmony_ci he_cap_elem->phy_cap_info[9] |= 5198c2ecf20Sopenharmony_ci IEEE80211_HE_PHY_CAP9_NOMIMAL_PKT_PADDING_16US; 5208c2ecf20Sopenharmony_ci } 5218c2ecf20Sopenharmony_ci idx++; 5228c2ecf20Sopenharmony_ci } 5238c2ecf20Sopenharmony_ci 5248c2ecf20Sopenharmony_ci return idx; 5258c2ecf20Sopenharmony_ci} 5268c2ecf20Sopenharmony_ci 5278c2ecf20Sopenharmony_civoid mt7915_set_stream_he_caps(struct mt7915_phy *phy) 5288c2ecf20Sopenharmony_ci{ 5298c2ecf20Sopenharmony_ci struct ieee80211_sband_iftype_data *data; 5308c2ecf20Sopenharmony_ci struct ieee80211_supported_band *band; 5318c2ecf20Sopenharmony_ci int n; 5328c2ecf20Sopenharmony_ci 5338c2ecf20Sopenharmony_ci if (phy->mt76->cap.has_2ghz) { 5348c2ecf20Sopenharmony_ci data = phy->iftype[NL80211_BAND_2GHZ]; 5358c2ecf20Sopenharmony_ci n = mt7915_init_he_caps(phy, NL80211_BAND_2GHZ, data); 5368c2ecf20Sopenharmony_ci 5378c2ecf20Sopenharmony_ci band = &phy->mt76->sband_2g.sband; 5388c2ecf20Sopenharmony_ci band->iftype_data = data; 5398c2ecf20Sopenharmony_ci band->n_iftype_data = n; 5408c2ecf20Sopenharmony_ci } 5418c2ecf20Sopenharmony_ci 5428c2ecf20Sopenharmony_ci if (phy->mt76->cap.has_5ghz) { 5438c2ecf20Sopenharmony_ci data = phy->iftype[NL80211_BAND_5GHZ]; 5448c2ecf20Sopenharmony_ci n = mt7915_init_he_caps(phy, NL80211_BAND_5GHZ, data); 5458c2ecf20Sopenharmony_ci 5468c2ecf20Sopenharmony_ci band = &phy->mt76->sband_5g.sband; 5478c2ecf20Sopenharmony_ci band->iftype_data = data; 5488c2ecf20Sopenharmony_ci band->n_iftype_data = n; 5498c2ecf20Sopenharmony_ci } 5508c2ecf20Sopenharmony_ci} 5518c2ecf20Sopenharmony_ci 5528c2ecf20Sopenharmony_cistatic void 5538c2ecf20Sopenharmony_cimt7915_cap_dbdc_enable(struct mt7915_dev *dev) 5548c2ecf20Sopenharmony_ci{ 5558c2ecf20Sopenharmony_ci dev->mphy.sband_5g.sband.vht_cap.cap &= 5568c2ecf20Sopenharmony_ci ~(IEEE80211_VHT_CAP_SHORT_GI_160 | 5578c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ); 5588c2ecf20Sopenharmony_ci 5598c2ecf20Sopenharmony_ci if (dev->chainmask == 0xf) 5608c2ecf20Sopenharmony_ci dev->mphy.antenna_mask = dev->chainmask >> 2; 5618c2ecf20Sopenharmony_ci else 5628c2ecf20Sopenharmony_ci dev->mphy.antenna_mask = dev->chainmask >> 1; 5638c2ecf20Sopenharmony_ci 5648c2ecf20Sopenharmony_ci dev->phy.chainmask = dev->mphy.antenna_mask; 5658c2ecf20Sopenharmony_ci dev->mphy.hw->wiphy->available_antennas_rx = dev->phy.chainmask; 5668c2ecf20Sopenharmony_ci dev->mphy.hw->wiphy->available_antennas_tx = dev->phy.chainmask; 5678c2ecf20Sopenharmony_ci 5688c2ecf20Sopenharmony_ci mt76_set_stream_caps(&dev->mphy, true); 5698c2ecf20Sopenharmony_ci mt7915_set_stream_vht_txbf_caps(&dev->phy); 5708c2ecf20Sopenharmony_ci mt7915_set_stream_he_caps(&dev->phy); 5718c2ecf20Sopenharmony_ci} 5728c2ecf20Sopenharmony_ci 5738c2ecf20Sopenharmony_cistatic void 5748c2ecf20Sopenharmony_cimt7915_cap_dbdc_disable(struct mt7915_dev *dev) 5758c2ecf20Sopenharmony_ci{ 5768c2ecf20Sopenharmony_ci dev->mphy.sband_5g.sband.vht_cap.cap |= 5778c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_SHORT_GI_160 | 5788c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160_80PLUS80MHZ; 5798c2ecf20Sopenharmony_ci 5808c2ecf20Sopenharmony_ci dev->mphy.antenna_mask = dev->chainmask; 5818c2ecf20Sopenharmony_ci dev->phy.chainmask = dev->chainmask; 5828c2ecf20Sopenharmony_ci dev->mphy.hw->wiphy->available_antennas_rx = dev->chainmask; 5838c2ecf20Sopenharmony_ci dev->mphy.hw->wiphy->available_antennas_tx = dev->chainmask; 5848c2ecf20Sopenharmony_ci 5858c2ecf20Sopenharmony_ci mt76_set_stream_caps(&dev->mphy, true); 5868c2ecf20Sopenharmony_ci mt7915_set_stream_vht_txbf_caps(&dev->phy); 5878c2ecf20Sopenharmony_ci mt7915_set_stream_he_caps(&dev->phy); 5888c2ecf20Sopenharmony_ci} 5898c2ecf20Sopenharmony_ci 5908c2ecf20Sopenharmony_ciint mt7915_register_ext_phy(struct mt7915_dev *dev) 5918c2ecf20Sopenharmony_ci{ 5928c2ecf20Sopenharmony_ci struct mt7915_phy *phy = mt7915_ext_phy(dev); 5938c2ecf20Sopenharmony_ci struct mt76_phy *mphy; 5948c2ecf20Sopenharmony_ci int ret; 5958c2ecf20Sopenharmony_ci bool bound; 5968c2ecf20Sopenharmony_ci 5978c2ecf20Sopenharmony_ci /* TODO: enble DBDC */ 5988c2ecf20Sopenharmony_ci bound = mt7915_l1_rr(dev, MT_HW_BOUND) & BIT(5); 5998c2ecf20Sopenharmony_ci if (!bound) 6008c2ecf20Sopenharmony_ci return -EINVAL; 6018c2ecf20Sopenharmony_ci 6028c2ecf20Sopenharmony_ci if (test_bit(MT76_STATE_RUNNING, &dev->mphy.state)) 6038c2ecf20Sopenharmony_ci return -EINVAL; 6048c2ecf20Sopenharmony_ci 6058c2ecf20Sopenharmony_ci if (phy) 6068c2ecf20Sopenharmony_ci return 0; 6078c2ecf20Sopenharmony_ci 6088c2ecf20Sopenharmony_ci mt7915_cap_dbdc_enable(dev); 6098c2ecf20Sopenharmony_ci mphy = mt76_alloc_phy(&dev->mt76, sizeof(*phy), &mt7915_ops); 6108c2ecf20Sopenharmony_ci if (!mphy) 6118c2ecf20Sopenharmony_ci return -ENOMEM; 6128c2ecf20Sopenharmony_ci 6138c2ecf20Sopenharmony_ci phy = mphy->priv; 6148c2ecf20Sopenharmony_ci phy->dev = dev; 6158c2ecf20Sopenharmony_ci phy->mt76 = mphy; 6168c2ecf20Sopenharmony_ci phy->chainmask = dev->chainmask & ~dev->phy.chainmask; 6178c2ecf20Sopenharmony_ci mphy->antenna_mask = BIT(hweight8(phy->chainmask)) - 1; 6188c2ecf20Sopenharmony_ci mt7915_init_wiphy(mphy->hw); 6198c2ecf20Sopenharmony_ci 6208c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&phy->stats_list); 6218c2ecf20Sopenharmony_ci INIT_DELAYED_WORK(&phy->mac_work, mt7915_mac_work); 6228c2ecf20Sopenharmony_ci 6238c2ecf20Sopenharmony_ci /* 6248c2ecf20Sopenharmony_ci * Make the secondary PHY MAC address local without overlapping with 6258c2ecf20Sopenharmony_ci * the usual MAC address allocation scheme on multiple virtual interfaces 6268c2ecf20Sopenharmony_ci */ 6278c2ecf20Sopenharmony_ci mphy->hw->wiphy->perm_addr[0] |= 2; 6288c2ecf20Sopenharmony_ci mphy->hw->wiphy->perm_addr[0] ^= BIT(7); 6298c2ecf20Sopenharmony_ci 6308c2ecf20Sopenharmony_ci /* The second interface does not get any packets unless it has a vif */ 6318c2ecf20Sopenharmony_ci ieee80211_hw_set(mphy->hw, WANT_MONITOR_VIF); 6328c2ecf20Sopenharmony_ci 6338c2ecf20Sopenharmony_ci ret = mt76_register_phy(mphy); 6348c2ecf20Sopenharmony_ci if (ret) 6358c2ecf20Sopenharmony_ci ieee80211_free_hw(mphy->hw); 6368c2ecf20Sopenharmony_ci 6378c2ecf20Sopenharmony_ci return ret; 6388c2ecf20Sopenharmony_ci} 6398c2ecf20Sopenharmony_ci 6408c2ecf20Sopenharmony_civoid mt7915_unregister_ext_phy(struct mt7915_dev *dev) 6418c2ecf20Sopenharmony_ci{ 6428c2ecf20Sopenharmony_ci struct mt7915_phy *phy = mt7915_ext_phy(dev); 6438c2ecf20Sopenharmony_ci struct mt76_phy *mphy = dev->mt76.phy2; 6448c2ecf20Sopenharmony_ci 6458c2ecf20Sopenharmony_ci if (!phy) 6468c2ecf20Sopenharmony_ci return; 6478c2ecf20Sopenharmony_ci 6488c2ecf20Sopenharmony_ci mt7915_cap_dbdc_disable(dev); 6498c2ecf20Sopenharmony_ci mt76_unregister_phy(mphy); 6508c2ecf20Sopenharmony_ci ieee80211_free_hw(mphy->hw); 6518c2ecf20Sopenharmony_ci} 6528c2ecf20Sopenharmony_ci 6538c2ecf20Sopenharmony_ciint mt7915_register_device(struct mt7915_dev *dev) 6548c2ecf20Sopenharmony_ci{ 6558c2ecf20Sopenharmony_ci struct ieee80211_hw *hw = mt76_hw(dev); 6568c2ecf20Sopenharmony_ci int ret; 6578c2ecf20Sopenharmony_ci 6588c2ecf20Sopenharmony_ci dev->phy.dev = dev; 6598c2ecf20Sopenharmony_ci dev->phy.mt76 = &dev->mt76.phy; 6608c2ecf20Sopenharmony_ci dev->mt76.phy.priv = &dev->phy; 6618c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&dev->phy.stats_list); 6628c2ecf20Sopenharmony_ci INIT_WORK(&dev->rc_work, mt7915_mac_sta_rc_work); 6638c2ecf20Sopenharmony_ci INIT_DELAYED_WORK(&dev->phy.mac_work, mt7915_mac_work); 6648c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&dev->sta_rc_list); 6658c2ecf20Sopenharmony_ci INIT_LIST_HEAD(&dev->sta_poll_list); 6668c2ecf20Sopenharmony_ci spin_lock_init(&dev->sta_poll_lock); 6678c2ecf20Sopenharmony_ci 6688c2ecf20Sopenharmony_ci init_waitqueue_head(&dev->reset_wait); 6698c2ecf20Sopenharmony_ci INIT_WORK(&dev->reset_work, mt7915_mac_reset_work); 6708c2ecf20Sopenharmony_ci 6718c2ecf20Sopenharmony_ci ret = mt7915_init_hardware(dev); 6728c2ecf20Sopenharmony_ci if (ret) 6738c2ecf20Sopenharmony_ci return ret; 6748c2ecf20Sopenharmony_ci 6758c2ecf20Sopenharmony_ci mt7915_init_wiphy(hw); 6768c2ecf20Sopenharmony_ci dev->mphy.sband_2g.sband.ht_cap.cap |= 6778c2ecf20Sopenharmony_ci IEEE80211_HT_CAP_LDPC_CODING | 6788c2ecf20Sopenharmony_ci IEEE80211_HT_CAP_MAX_AMSDU; 6798c2ecf20Sopenharmony_ci dev->mphy.sband_5g.sband.ht_cap.cap |= 6808c2ecf20Sopenharmony_ci IEEE80211_HT_CAP_LDPC_CODING | 6818c2ecf20Sopenharmony_ci IEEE80211_HT_CAP_MAX_AMSDU; 6828c2ecf20Sopenharmony_ci dev->mphy.sband_5g.sband.vht_cap.cap |= 6838c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_7991 | 6848c2ecf20Sopenharmony_ci IEEE80211_VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MASK; 6858c2ecf20Sopenharmony_ci mt7915_cap_dbdc_disable(dev); 6868c2ecf20Sopenharmony_ci dev->phy.dfs_state = -1; 6878c2ecf20Sopenharmony_ci 6888c2ecf20Sopenharmony_ci ret = mt76_register_device(&dev->mt76, true, mt7915_rates, 6898c2ecf20Sopenharmony_ci ARRAY_SIZE(mt7915_rates)); 6908c2ecf20Sopenharmony_ci if (ret) 6918c2ecf20Sopenharmony_ci return ret; 6928c2ecf20Sopenharmony_ci 6938c2ecf20Sopenharmony_ci ieee80211_queue_work(mt76_hw(dev), &dev->init_work); 6948c2ecf20Sopenharmony_ci 6958c2ecf20Sopenharmony_ci return mt7915_init_debugfs(dev); 6968c2ecf20Sopenharmony_ci} 6978c2ecf20Sopenharmony_ci 6988c2ecf20Sopenharmony_civoid mt7915_unregister_device(struct mt7915_dev *dev) 6998c2ecf20Sopenharmony_ci{ 7008c2ecf20Sopenharmony_ci struct mt76_txwi_cache *txwi; 7018c2ecf20Sopenharmony_ci int id; 7028c2ecf20Sopenharmony_ci 7038c2ecf20Sopenharmony_ci mt7915_unregister_ext_phy(dev); 7048c2ecf20Sopenharmony_ci mt76_unregister_device(&dev->mt76); 7058c2ecf20Sopenharmony_ci mt7915_mcu_exit(dev); 7068c2ecf20Sopenharmony_ci mt7915_dma_cleanup(dev); 7078c2ecf20Sopenharmony_ci 7088c2ecf20Sopenharmony_ci spin_lock_bh(&dev->token_lock); 7098c2ecf20Sopenharmony_ci idr_for_each_entry(&dev->token, txwi, id) { 7108c2ecf20Sopenharmony_ci mt7915_txp_skb_unmap(&dev->mt76, txwi); 7118c2ecf20Sopenharmony_ci if (txwi->skb) { 7128c2ecf20Sopenharmony_ci struct ieee80211_hw *hw; 7138c2ecf20Sopenharmony_ci 7148c2ecf20Sopenharmony_ci hw = mt76_tx_status_get_hw(&dev->mt76, txwi->skb); 7158c2ecf20Sopenharmony_ci ieee80211_free_txskb(hw, txwi->skb); 7168c2ecf20Sopenharmony_ci } 7178c2ecf20Sopenharmony_ci mt76_put_txwi(&dev->mt76, txwi); 7188c2ecf20Sopenharmony_ci } 7198c2ecf20Sopenharmony_ci spin_unlock_bh(&dev->token_lock); 7208c2ecf20Sopenharmony_ci idr_destroy(&dev->token); 7218c2ecf20Sopenharmony_ci 7228c2ecf20Sopenharmony_ci mt76_free_device(&dev->mt76); 7238c2ecf20Sopenharmony_ci} 724