// SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause /* Copyright(c) 2023 Realtek Corporation */ #include "chan.h" #include "debug.h" #include "mac.h" #include "phy.h" #include "reg.h" #include "rtw8922a.h" #include "rtw8922a_rfk.h" static void rtw8922a_tssi_cont_en(struct rtw89_dev *rtwdev, bool en, enum rtw89_rf_path path) { static const u32 tssi_trk_man[2] = {R_TSSI_PWR_P0, R_TSSI_PWR_P1}; if (en) rtw89_phy_write32_mask(rtwdev, tssi_trk_man[path], B_TSSI_CONT_EN, 0); else rtw89_phy_write32_mask(rtwdev, tssi_trk_man[path], B_TSSI_CONT_EN, 1); } void rtw8922a_tssi_cont_en_phyidx(struct rtw89_dev *rtwdev, bool en, u8 phy_idx) { if (rtwdev->mlo_dbcc_mode == MLO_1_PLUS_1_1RF) { if (phy_idx == RTW89_PHY_0) rtw8922a_tssi_cont_en(rtwdev, en, RF_PATH_A); else rtw8922a_tssi_cont_en(rtwdev, en, RF_PATH_B); } else { rtw8922a_tssi_cont_en(rtwdev, en, RF_PATH_A); rtw8922a_tssi_cont_en(rtwdev, en, RF_PATH_B); } } static void rtw8922a_ctl_band_ch_bw(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy, u8 central_ch, enum rtw89_band band, enum rtw89_bandwidth bw) { const u32 rf_addr[2] = {RR_CFGCH, RR_CFGCH_V1}; struct rtw89_hal *hal = &rtwdev->hal; u32 rf_reg[RF_PATH_NUM_8922A][2]; u8 synpath; u32 rf18; u8 kpath; u8 path; u8 i; rf_reg[RF_PATH_A][0] = rtw89_read_rf(rtwdev, RF_PATH_A, rf_addr[0], RFREG_MASK); rf_reg[RF_PATH_A][1] = rtw89_read_rf(rtwdev, RF_PATH_A, rf_addr[1], RFREG_MASK); rf_reg[RF_PATH_B][0] = rtw89_read_rf(rtwdev, RF_PATH_B, rf_addr[0], RFREG_MASK); rf_reg[RF_PATH_B][1] = rtw89_read_rf(rtwdev, RF_PATH_B, rf_addr[1], RFREG_MASK); kpath = rtw89_phy_get_kpath(rtwdev, phy); synpath = rtw89_phy_get_syn_sel(rtwdev, phy); rf18 = rtw89_read_rf(rtwdev, synpath, RR_CFGCH, RFREG_MASK); if (rf18 == INV_RF_DATA) { rtw89_warn(rtwdev, "[RFK] Invalid RF18 value\n"); return; } for (path = 0; path < RF_PATH_NUM_8922A; path++) { if (!(kpath & BIT(path))) continue; for (i = 0; i < 2; i++) { if (rf_reg[path][i] == INV_RF_DATA) { rtw89_warn(rtwdev, "[RFK] Invalid RF_0x18 for Path-%d\n", path); return; } rf_reg[path][i] &= ~(RR_CFGCH_BAND1 | RR_CFGCH_BW | RR_CFGCH_BAND0 | RR_CFGCH_CH); rf_reg[path][i] |= u32_encode_bits(central_ch, RR_CFGCH_CH); if (band == RTW89_BAND_2G) rtw89_write_rf(rtwdev, path, RR_SMD, RR_VCO2, 0x0); else rtw89_write_rf(rtwdev, path, RR_SMD, RR_VCO2, 0x1); switch (band) { case RTW89_BAND_2G: default: break; case RTW89_BAND_5G: rf_reg[path][i] |= u32_encode_bits(CFGCH_BAND1_5G, RR_CFGCH_BAND1) | u32_encode_bits(CFGCH_BAND0_5G, RR_CFGCH_BAND0); break; case RTW89_BAND_6G: rf_reg[path][i] |= u32_encode_bits(CFGCH_BAND1_6G, RR_CFGCH_BAND1) | u32_encode_bits(CFGCH_BAND0_6G, RR_CFGCH_BAND0); break; } switch (bw) { case RTW89_CHANNEL_WIDTH_5: case RTW89_CHANNEL_WIDTH_10: case RTW89_CHANNEL_WIDTH_20: default: break; case RTW89_CHANNEL_WIDTH_40: rf_reg[path][i] |= u32_encode_bits(CFGCH_BW_V2_40M, RR_CFGCH_BW_V2); break; case RTW89_CHANNEL_WIDTH_80: rf_reg[path][i] |= u32_encode_bits(CFGCH_BW_V2_80M, RR_CFGCH_BW_V2); break; case RTW89_CHANNEL_WIDTH_160: rf_reg[path][i] |= u32_encode_bits(CFGCH_BW_V2_160M, RR_CFGCH_BW_V2); break; case RTW89_CHANNEL_WIDTH_320: rf_reg[path][i] |= u32_encode_bits(CFGCH_BW_V2_320M, RR_CFGCH_BW_V2); break; } rtw89_write_rf(rtwdev, path, rf_addr[i], RFREG_MASK, rf_reg[path][i]); fsleep(100); } } if (hal->cv != CHIP_CAV) return; if (band == RTW89_BAND_2G) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWE, RFREG_MASK, 0x80000); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWA, RFREG_MASK, 0x00003); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWD1, RFREG_MASK, 0x0c990); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWD0, RFREG_MASK, 0xebe38); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWE, RFREG_MASK, 0x00000); } else { rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWE, RFREG_MASK, 0x80000); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWA, RFREG_MASK, 0x00003); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWD1, RFREG_MASK, 0x0c190); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWD0, RFREG_MASK, 0xebe38); rtw89_write_rf(rtwdev, RF_PATH_A, RR_LUTWE, RFREG_MASK, 0x00000); } } void rtw8922a_set_channel_rf(struct rtw89_dev *rtwdev, const struct rtw89_chan *chan, enum rtw89_phy_idx phy_idx) { rtw8922a_ctl_band_ch_bw(rtwdev, phy_idx, chan->channel, chan->band_type, chan->band_width); } enum _rf_syn_pow { RF_SYN_ON_OFF, RF_SYN_OFF_ON, RF_SYN_ALLON, RF_SYN_ALLOFF, }; static void rtw8922a_set_syn01_cav(struct rtw89_dev *rtwdev, enum _rf_syn_pow syn) { if (syn == RF_SYN_ALLON) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x2); rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x2); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x3); } else if (syn == RF_SYN_ON_OFF) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x2); rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x0); } else if (syn == RF_SYN_OFF_ON) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x0); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x3); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x2); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x3); } else if (syn == RF_SYN_ALLOFF) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN, 0x0); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN, 0x0); } } static void rtw8922a_set_syn01_cbv(struct rtw89_dev *rtwdev, enum _rf_syn_pow syn) { if (syn == RF_SYN_ALLON) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN_V1, 0xf); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN_V1, 0xf); } else if (syn == RF_SYN_ON_OFF) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN_V1, 0xf); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN_V1, 0x0); } else if (syn == RF_SYN_OFF_ON) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN_V1, 0x0); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN_V1, 0xf); } else if (syn == RF_SYN_ALLOFF) { rtw89_write_rf(rtwdev, RF_PATH_A, RR_POW, RR_POW_SYN_V1, 0x0); rtw89_write_rf(rtwdev, RF_PATH_B, RR_POW, RR_POW_SYN_V1, 0x0); } } static void rtw8922a_set_syn01(struct rtw89_dev *rtwdev, enum _rf_syn_pow syn) { struct rtw89_hal *hal = &rtwdev->hal; rtw89_debug(rtwdev, RTW89_DBG_RFK, "SYN config=%d\n", syn); if (hal->cv == CHIP_CAV) rtw8922a_set_syn01_cav(rtwdev, syn); else rtw8922a_set_syn01_cbv(rtwdev, syn); } static void rtw8922a_chlk_ktbl_sel(struct rtw89_dev *rtwdev, u8 kpath, u8 idx) { u32 tmp; if (idx > 2) { rtw89_warn(rtwdev, "[DBCC][ERROR]indx is out of limit!! index(%d)", idx); return; } if (kpath & RF_A) { rtw89_phy_write32_mask(rtwdev, R_COEF_SEL, B_COEF_SEL_EN, 0x1); rtw89_phy_write32_mask(rtwdev, R_COEF_SEL, B_COEF_SEL_IQC_V1, idx); rtw89_phy_write32_mask(rtwdev, R_COEF_SEL, B_COEF_SEL_MDPD_V1, idx); rtw89_write_rf(rtwdev, RF_PATH_A, RR_MODOPT, RR_TXG_SEL, 0x4 | idx); tmp = rtw89_phy_read32_mask(rtwdev, R_COEF_SEL, BIT(0)); rtw89_phy_write32_mask(rtwdev, R_CFIR_LUT, B_CFIR_LUT_G3, tmp); tmp = rtw89_phy_read32_mask(rtwdev, R_COEF_SEL, BIT(1)); rtw89_phy_write32_mask(rtwdev, R_CFIR_LUT, B_CFIR_LUT_G5, tmp); } if (kpath & RF_B) { rtw89_phy_write32_mask(rtwdev, R_COEF_SEL_C1, B_COEF_SEL_EN, 0x1); rtw89_phy_write32_mask(rtwdev, R_COEF_SEL_C1, B_COEF_SEL_IQC_V1, idx); rtw89_phy_write32_mask(rtwdev, R_COEF_SEL_C1, B_COEF_SEL_MDPD_V1, idx); rtw89_write_rf(rtwdev, RF_PATH_B, RR_MODOPT, RR_TXG_SEL, 0x4 | idx); tmp = rtw89_phy_read32_mask(rtwdev, R_COEF_SEL_C1, BIT(0)); rtw89_phy_write32_mask(rtwdev, R_CFIR_LUT_C1, B_CFIR_LUT_G3, tmp); tmp = rtw89_phy_read32_mask(rtwdev, R_COEF_SEL_C1, BIT(1)); rtw89_phy_write32_mask(rtwdev, R_CFIR_LUT_C1, B_CFIR_LUT_G5, tmp); } } static void rtw8922a_chlk_reload(struct rtw89_dev *rtwdev) { struct rtw89_rfk_mcc_info *rfk_mcc = &rtwdev->rfk_mcc; struct rtw89_rfk_chan_desc desc[__RTW89_RFK_CHS_NR_V1] = {}; enum rtw89_chanctx_idx chanctx_idx; const struct rtw89_chan *chan; enum rtw89_entity_mode mode; u8 s0_tbl, s1_tbl; u8 tbl_sel; mode = rtw89_get_entity_mode(rtwdev); switch (mode) { case RTW89_ENTITY_MODE_MCC_PREPARE: chanctx_idx = RTW89_CHANCTX_1; break; default: chanctx_idx = RTW89_CHANCTX_0; break; } chan = rtw89_chan_get(rtwdev, chanctx_idx); for (tbl_sel = 0; tbl_sel < ARRAY_SIZE(desc); tbl_sel++) { struct rtw89_rfk_chan_desc *p = &desc[tbl_sel]; p->ch = rfk_mcc->ch[tbl_sel]; p->has_band = true; p->band = rfk_mcc->band[tbl_sel]; p->has_bw = true; p->bw = rfk_mcc->bw[tbl_sel]; } tbl_sel = rtw89_rfk_chan_lookup(rtwdev, desc, ARRAY_SIZE(desc), chan); rfk_mcc->ch[tbl_sel] = chan->channel; rfk_mcc->band[tbl_sel] = chan->band_type; rfk_mcc->bw[tbl_sel] = chan->band_width; rfk_mcc->table_idx = tbl_sel; s0_tbl = tbl_sel; s1_tbl = tbl_sel; rtw8922a_chlk_ktbl_sel(rtwdev, RF_A, s0_tbl); rtw8922a_chlk_ktbl_sel(rtwdev, RF_B, s1_tbl); } static void rtw8922a_rfk_mlo_ctrl(struct rtw89_dev *rtwdev) { enum _rf_syn_pow syn_pow; if (!rtwdev->dbcc_en) goto set_rfk_reload; switch (rtwdev->mlo_dbcc_mode) { case MLO_0_PLUS_2_1RF: syn_pow = RF_SYN_OFF_ON; break; case MLO_0_PLUS_2_2RF: case MLO_1_PLUS_1_2RF: case MLO_2_PLUS_0_1RF: case MLO_2_PLUS_0_2RF: case MLO_2_PLUS_2_2RF: case MLO_DBCC_NOT_SUPPORT: default: syn_pow = RF_SYN_ON_OFF; break; case MLO_1_PLUS_1_1RF: case DBCC_LEGACY: syn_pow = RF_SYN_ALLON; break; } rtw8922a_set_syn01(rtwdev, syn_pow); set_rfk_reload: rtw8922a_chlk_reload(rtwdev); } static void rtw8922a_rfk_pll_init(struct rtw89_dev *rtwdev) { int ret; u8 tmp; ret = rtw89_mac_read_xtal_si(rtwdev, XTAL_SI_PLL_1, &tmp); if (ret) return; ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_PLL_1, tmp | 0xf8, 0xFF); if (ret) return; ret = rtw89_mac_read_xtal_si(rtwdev, XTAL_SI_APBT, &tmp); if (ret) return; ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_APBT, tmp & ~0x60, 0xFF); if (ret) return; ret = rtw89_mac_read_xtal_si(rtwdev, XTAL_SI_XTAL_PLL, &tmp); if (ret) return; ret = rtw89_mac_write_xtal_si(rtwdev, XTAL_SI_XTAL_PLL, tmp | 0x38, 0xFF); if (ret) return; } void rtw8922a_rfk_hw_init(struct rtw89_dev *rtwdev) { if (rtwdev->dbcc_en) rtw8922a_rfk_mlo_ctrl(rtwdev); rtw8922a_rfk_pll_init(rtwdev); } void rtw8922a_pre_set_channel_rf(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy_idx) { bool mlo_1_1; if (!rtwdev->dbcc_en) return; mlo_1_1 = rtw89_is_mlo_1_1(rtwdev); if (mlo_1_1) rtw8922a_set_syn01(rtwdev, RF_SYN_ALLON); else if (phy_idx == RTW89_PHY_0) rtw8922a_set_syn01(rtwdev, RF_SYN_ON_OFF); else rtw8922a_set_syn01(rtwdev, RF_SYN_OFF_ON); fsleep(1000); } void rtw8922a_post_set_channel_rf(struct rtw89_dev *rtwdev, enum rtw89_phy_idx phy_idx) { rtw8922a_rfk_mlo_ctrl(rtwdev); }