1 /****************************************************************************** 2 * 3 * Copyright(c) 2009-2014 Realtek Corporation. 4 * 5 * This program is free software; you can redistribute it and/or modify it 6 * under the terms of version 2 of the GNU General Public License as 7 * published by the Free Software Foundation. 8 * 9 * This program is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for 12 * more details. 13 * 14 * The full GNU General Public License is included in this distribution in the 15 * file called LICENSE. 16 * 17 * Contact Information: 18 * wlanfae <wlanfae@realtek.com> 19 * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, 20 * Hsinchu 300, Taiwan. 21 * 22 * Larry Finger <Larry.Finger@lwfinger.net> 23 * 24 *****************************************************************************/ 25 26 #include "../wifi.h" 27 #include "phy_common.h" 28 #include "../rtl8723ae/reg.h" 29 #include <linux/module.h> 30 31 /* These routines are common to RTL8723AE and RTL8723bE */ 32 33 u32 rtl8723_phy_query_bb_reg(struct ieee80211_hw *hw, 34 u32 regaddr, u32 bitmask) 35 { 36 struct rtl_priv *rtlpriv = rtl_priv(hw); 37 u32 returnvalue, originalvalue, bitshift; 38 39 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 40 "regaddr(%#x), bitmask(%#x)\n", regaddr, bitmask); 41 originalvalue = rtl_read_dword(rtlpriv, regaddr); 42 bitshift = rtl8723_phy_calculate_bit_shift(bitmask); 43 returnvalue = (originalvalue & bitmask) >> bitshift; 44 45 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 46 "BBR MASK=0x%x Addr[0x%x]=0x%x\n", bitmask, 47 regaddr, originalvalue); 48 return returnvalue; 49 } 50 EXPORT_SYMBOL_GPL(rtl8723_phy_query_bb_reg); 51 52 void rtl8723_phy_set_bb_reg(struct ieee80211_hw *hw, u32 regaddr, 53 u32 bitmask, u32 data) 54 { 55 struct rtl_priv *rtlpriv = rtl_priv(hw); 56 u32 originalvalue, bitshift; 57 58 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 59 "regaddr(%#x), bitmask(%#x), data(%#x)\n", regaddr, bitmask, 60 data); 61 62 if (bitmask != MASKDWORD) { 63 originalvalue = rtl_read_dword(rtlpriv, regaddr); 64 bitshift = rtl8723_phy_calculate_bit_shift(bitmask); 65 data = ((originalvalue & (~bitmask)) | (data << bitshift)); 66 } 67 68 rtl_write_dword(rtlpriv, regaddr, data); 69 70 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 71 "regaddr(%#x), bitmask(%#x), data(%#x)\n", 72 regaddr, bitmask, data); 73 } 74 EXPORT_SYMBOL_GPL(rtl8723_phy_set_bb_reg); 75 76 u32 rtl8723_phy_calculate_bit_shift(u32 bitmask) 77 { 78 u32 i; 79 80 for (i = 0; i <= 31; i++) { 81 if (((bitmask >> i) & 0x1) == 1) 82 break; 83 } 84 return i; 85 } 86 EXPORT_SYMBOL_GPL(rtl8723_phy_calculate_bit_shift); 87 88 u32 rtl8723_phy_rf_serial_read(struct ieee80211_hw *hw, 89 enum radio_path rfpath, u32 offset) 90 { 91 struct rtl_priv *rtlpriv = rtl_priv(hw); 92 struct rtl_phy *rtlphy = &(rtlpriv->phy); 93 struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; 94 u32 newoffset; 95 u32 tmplong, tmplong2; 96 u8 rfpi_enable = 0; 97 u32 retvalue; 98 99 offset &= 0xff; 100 newoffset = offset; 101 if (RT_CANNOT_IO(hw)) { 102 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, "return all one\n"); 103 return 0xFFFFFFFF; 104 } 105 tmplong = rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD); 106 if (rfpath == RF90_PATH_A) 107 tmplong2 = tmplong; 108 else 109 tmplong2 = rtl_get_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD); 110 tmplong2 = (tmplong2 & (~BLSSIREADADDRESS)) | 111 (newoffset << 23) | BLSSIREADEDGE; 112 rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, 113 tmplong & (~BLSSIREADEDGE)); 114 mdelay(1); 115 rtl_set_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD, tmplong2); 116 mdelay(1); 117 rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, 118 tmplong | BLSSIREADEDGE); 119 mdelay(1); 120 if (rfpath == RF90_PATH_A) 121 rfpi_enable = (u8) rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER1, 122 BIT(8)); 123 else if (rfpath == RF90_PATH_B) 124 rfpi_enable = (u8) rtl_get_bbreg(hw, RFPGA0_XB_HSSIPARAMETER1, 125 BIT(8)); 126 if (rfpi_enable) 127 retvalue = rtl_get_bbreg(hw, pphyreg->rf_rbpi, 128 BLSSIREADBACKDATA); 129 else 130 retvalue = rtl_get_bbreg(hw, pphyreg->rf_rb, 131 BLSSIREADBACKDATA); 132 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 133 "RFR-%d Addr[0x%x]=0x%x\n", 134 rfpath, pphyreg->rf_rb, retvalue); 135 return retvalue; 136 } 137 EXPORT_SYMBOL_GPL(rtl8723_phy_rf_serial_read); 138 139 void rtl8723_phy_rf_serial_write(struct ieee80211_hw *hw, 140 enum radio_path rfpath, 141 u32 offset, u32 data) 142 { 143 u32 data_and_addr; 144 u32 newoffset; 145 struct rtl_priv *rtlpriv = rtl_priv(hw); 146 struct rtl_phy *rtlphy = &(rtlpriv->phy); 147 struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; 148 149 if (RT_CANNOT_IO(hw)) { 150 RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, "stop\n"); 151 return; 152 } 153 offset &= 0xff; 154 newoffset = offset; 155 data_and_addr = ((newoffset << 20) | (data & 0x000fffff)) & 0x0fffffff; 156 rtl_set_bbreg(hw, pphyreg->rf3wire_offset, MASKDWORD, data_and_addr); 157 RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, 158 "RFW-%d Addr[0x%x]=0x%x\n", 159 rfpath, pphyreg->rf3wire_offset, 160 data_and_addr); 161 } 162 EXPORT_SYMBOL_GPL(rtl8723_phy_rf_serial_write); 163 164 long rtl8723_phy_txpwr_idx_to_dbm(struct ieee80211_hw *hw, 165 enum wireless_mode wirelessmode, 166 u8 txpwridx) 167 { 168 long offset; 169 long pwrout_dbm; 170 171 switch (wirelessmode) { 172 case WIRELESS_MODE_B: 173 offset = -7; 174 break; 175 case WIRELESS_MODE_G: 176 case WIRELESS_MODE_N_24G: 177 offset = -8; 178 break; 179 default: 180 offset = -8; 181 break; 182 } 183 pwrout_dbm = txpwridx / 2 + offset; 184 return pwrout_dbm; 185 } 186 EXPORT_SYMBOL_GPL(rtl8723_phy_txpwr_idx_to_dbm); 187 188 void rtl8723_phy_init_bb_rf_reg_def(struct ieee80211_hw *hw) 189 { 190 struct rtl_priv *rtlpriv = rtl_priv(hw); 191 struct rtl_phy *rtlphy = &(rtlpriv->phy); 192 193 rtlphy->phyreg_def[RF90_PATH_A].rfintfs = RFPGA0_XAB_RFINTERFACESW; 194 rtlphy->phyreg_def[RF90_PATH_B].rfintfs = RFPGA0_XAB_RFINTERFACESW; 195 rtlphy->phyreg_def[RF90_PATH_C].rfintfs = RFPGA0_XCD_RFINTERFACESW; 196 rtlphy->phyreg_def[RF90_PATH_D].rfintfs = RFPGA0_XCD_RFINTERFACESW; 197 198 rtlphy->phyreg_def[RF90_PATH_A].rfintfi = RFPGA0_XAB_RFINTERFACERB; 199 rtlphy->phyreg_def[RF90_PATH_B].rfintfi = RFPGA0_XAB_RFINTERFACERB; 200 rtlphy->phyreg_def[RF90_PATH_C].rfintfi = RFPGA0_XCD_RFINTERFACERB; 201 rtlphy->phyreg_def[RF90_PATH_D].rfintfi = RFPGA0_XCD_RFINTERFACERB; 202 203 rtlphy->phyreg_def[RF90_PATH_A].rfintfo = RFPGA0_XA_RFINTERFACEOE; 204 rtlphy->phyreg_def[RF90_PATH_B].rfintfo = RFPGA0_XB_RFINTERFACEOE; 205 206 rtlphy->phyreg_def[RF90_PATH_A].rfintfe = RFPGA0_XA_RFINTERFACEOE; 207 rtlphy->phyreg_def[RF90_PATH_B].rfintfe = RFPGA0_XB_RFINTERFACEOE; 208 209 rtlphy->phyreg_def[RF90_PATH_A].rf3wire_offset = 210 RFPGA0_XA_LSSIPARAMETER; 211 rtlphy->phyreg_def[RF90_PATH_B].rf3wire_offset = 212 RFPGA0_XB_LSSIPARAMETER; 213 214 rtlphy->phyreg_def[RF90_PATH_A].rflssi_select = RFPGA0_XAB_RFPARAMETER; 215 rtlphy->phyreg_def[RF90_PATH_B].rflssi_select = RFPGA0_XAB_RFPARAMETER; 216 rtlphy->phyreg_def[RF90_PATH_C].rflssi_select = RFPGA0_XCD_RFPARAMETER; 217 rtlphy->phyreg_def[RF90_PATH_D].rflssi_select = RFPGA0_XCD_RFPARAMETER; 218 219 rtlphy->phyreg_def[RF90_PATH_A].rftxgain_stage = RFPGA0_TXGAINSTAGE; 220 rtlphy->phyreg_def[RF90_PATH_B].rftxgain_stage = RFPGA0_TXGAINSTAGE; 221 rtlphy->phyreg_def[RF90_PATH_C].rftxgain_stage = RFPGA0_TXGAINSTAGE; 222 rtlphy->phyreg_def[RF90_PATH_D].rftxgain_stage = RFPGA0_TXGAINSTAGE; 223 224 rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para1 = RFPGA0_XA_HSSIPARAMETER1; 225 rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para1 = RFPGA0_XB_HSSIPARAMETER1; 226 227 rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para2 = RFPGA0_XA_HSSIPARAMETER2; 228 rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para2 = RFPGA0_XB_HSSIPARAMETER2; 229 230 rtlphy->phyreg_def[RF90_PATH_A].rfsw_ctrl = RFPGA0_XAB_SWITCHCONTROL; 231 rtlphy->phyreg_def[RF90_PATH_B].rfsw_ctrl = RFPGA0_XAB_SWITCHCONTROL; 232 rtlphy->phyreg_def[RF90_PATH_C].rfsw_ctrl = RFPGA0_XCD_SWITCHCONTROL; 233 rtlphy->phyreg_def[RF90_PATH_D].rfsw_ctrl = RFPGA0_XCD_SWITCHCONTROL; 234 235 rtlphy->phyreg_def[RF90_PATH_A].rfagc_control1 = ROFDM0_XAAGCCORE1; 236 rtlphy->phyreg_def[RF90_PATH_B].rfagc_control1 = ROFDM0_XBAGCCORE1; 237 rtlphy->phyreg_def[RF90_PATH_C].rfagc_control1 = ROFDM0_XCAGCCORE1; 238 rtlphy->phyreg_def[RF90_PATH_D].rfagc_control1 = ROFDM0_XDAGCCORE1; 239 240 rtlphy->phyreg_def[RF90_PATH_A].rfagc_control2 = ROFDM0_XAAGCCORE2; 241 rtlphy->phyreg_def[RF90_PATH_B].rfagc_control2 = ROFDM0_XBAGCCORE2; 242 rtlphy->phyreg_def[RF90_PATH_C].rfagc_control2 = ROFDM0_XCAGCCORE2; 243 rtlphy->phyreg_def[RF90_PATH_D].rfagc_control2 = ROFDM0_XDAGCCORE2; 244 245 rtlphy->phyreg_def[RF90_PATH_A].rfrxiq_imbal = ROFDM0_XARXIQIMBALANCE; 246 rtlphy->phyreg_def[RF90_PATH_B].rfrxiq_imbal = ROFDM0_XBRXIQIMBALANCE; 247 rtlphy->phyreg_def[RF90_PATH_C].rfrxiq_imbal = ROFDM0_XCRXIQIMBANLANCE; 248 rtlphy->phyreg_def[RF90_PATH_D].rfrxiq_imbal = ROFDM0_XDRXIQIMBALANCE; 249 250 rtlphy->phyreg_def[RF90_PATH_A].rfrx_afe = ROFDM0_XARXAFE; 251 rtlphy->phyreg_def[RF90_PATH_B].rfrx_afe = ROFDM0_XBRXAFE; 252 rtlphy->phyreg_def[RF90_PATH_C].rfrx_afe = ROFDM0_XCRXAFE; 253 rtlphy->phyreg_def[RF90_PATH_D].rfrx_afe = ROFDM0_XDRXAFE; 254 255 rtlphy->phyreg_def[RF90_PATH_A].rftxiq_imbal = ROFDM0_XATXIQIMBALANCE; 256 rtlphy->phyreg_def[RF90_PATH_B].rftxiq_imbal = ROFDM0_XBTXIQIMBALANCE; 257 rtlphy->phyreg_def[RF90_PATH_C].rftxiq_imbal = ROFDM0_XCTXIQIMBALANCE; 258 rtlphy->phyreg_def[RF90_PATH_D].rftxiq_imbal = ROFDM0_XDTXIQIMBALANCE; 259 260 rtlphy->phyreg_def[RF90_PATH_A].rftx_afe = ROFDM0_XATXAFE; 261 rtlphy->phyreg_def[RF90_PATH_B].rftx_afe = ROFDM0_XBTXAFE; 262 rtlphy->phyreg_def[RF90_PATH_C].rftx_afe = ROFDM0_XCTXAFE; 263 rtlphy->phyreg_def[RF90_PATH_D].rftx_afe = ROFDM0_XDTXAFE; 264 265 rtlphy->phyreg_def[RF90_PATH_A].rf_rb = RFPGA0_XA_LSSIREADBACK; 266 rtlphy->phyreg_def[RF90_PATH_B].rf_rb = RFPGA0_XB_LSSIREADBACK; 267 rtlphy->phyreg_def[RF90_PATH_C].rf_rb = RFPGA0_XC_LSSIREADBACK; 268 rtlphy->phyreg_def[RF90_PATH_D].rf_rb = RFPGA0_XD_LSSIREADBACK; 269 270 rtlphy->phyreg_def[RF90_PATH_A].rf_rbpi = TRANSCEIVEA_HSPI_READBACK; 271 rtlphy->phyreg_def[RF90_PATH_B].rf_rbpi = TRANSCEIVEB_HSPI_READBACK; 272 273 } 274 EXPORT_SYMBOL_GPL(rtl8723_phy_init_bb_rf_reg_def); 275 276 bool rtl8723_phy_set_sw_chnl_cmdarray(struct swchnlcmd *cmdtable, 277 u32 cmdtableidx, 278 u32 cmdtablesz, 279 enum swchnlcmd_id cmdid, 280 u32 para1, u32 para2, 281 u32 msdelay) 282 { 283 struct swchnlcmd *pcmd; 284 285 if (cmdtable == NULL) { 286 RT_ASSERT(false, "cmdtable cannot be NULL.\n"); 287 return false; 288 } 289 290 if (cmdtableidx >= cmdtablesz) 291 return false; 292 293 pcmd = cmdtable + cmdtableidx; 294 pcmd->cmdid = cmdid; 295 pcmd->para1 = para1; 296 pcmd->para2 = para2; 297 pcmd->msdelay = msdelay; 298 return true; 299 } 300 EXPORT_SYMBOL_GPL(rtl8723_phy_set_sw_chnl_cmdarray); 301 302 void rtl8723_phy_path_a_fill_iqk_matrix(struct ieee80211_hw *hw, 303 bool iqk_ok, 304 long result[][8], 305 u8 final_candidate, 306 bool btxonly) 307 { 308 u32 oldval_0, x, tx0_a, reg; 309 long y, tx0_c; 310 311 if (final_candidate == 0xFF) { 312 return; 313 } else if (iqk_ok) { 314 oldval_0 = (rtl_get_bbreg(hw, ROFDM0_XATXIQIMBALANCE, 315 MASKDWORD) >> 22) & 0x3FF; 316 x = result[final_candidate][0]; 317 if ((x & 0x00000200) != 0) 318 x = x | 0xFFFFFC00; 319 tx0_a = (x * oldval_0) >> 8; 320 rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, 0x3FF, tx0_a); 321 rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(31), 322 ((x * oldval_0 >> 7) & 0x1)); 323 y = result[final_candidate][1]; 324 if ((y & 0x00000200) != 0) 325 y = y | 0xFFFFFC00; 326 tx0_c = (y * oldval_0) >> 8; 327 rtl_set_bbreg(hw, ROFDM0_XCTXAFE, 0xF0000000, 328 ((tx0_c & 0x3C0) >> 6)); 329 rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, 0x003F0000, 330 (tx0_c & 0x3F)); 331 rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(29), 332 ((y * oldval_0 >> 7) & 0x1)); 333 if (btxonly) 334 return; 335 reg = result[final_candidate][2]; 336 rtl_set_bbreg(hw, ROFDM0_XARXIQIMBALANCE, 0x3FF, reg); 337 reg = result[final_candidate][3] & 0x3F; 338 rtl_set_bbreg(hw, ROFDM0_XARXIQIMBALANCE, 0xFC00, reg); 339 reg = (result[final_candidate][3] >> 6) & 0xF; 340 rtl_set_bbreg(hw, 0xca0, 0xF0000000, reg); 341 } 342 } 343 EXPORT_SYMBOL_GPL(rtl8723_phy_path_a_fill_iqk_matrix); 344 345 void rtl8723_save_adda_registers(struct ieee80211_hw *hw, u32 *addareg, 346 u32 *addabackup, u32 registernum) 347 { 348 u32 i; 349 350 for (i = 0; i < registernum; i++) 351 addabackup[i] = rtl_get_bbreg(hw, addareg[i], MASKDWORD); 352 } 353 EXPORT_SYMBOL_GPL(rtl8723_save_adda_registers); 354 355 void rtl8723_phy_save_mac_registers(struct ieee80211_hw *hw, 356 u32 *macreg, u32 *macbackup) 357 { 358 struct rtl_priv *rtlpriv = rtl_priv(hw); 359 u32 i; 360 361 for (i = 0; i < (IQK_MAC_REG_NUM - 1); i++) 362 macbackup[i] = rtl_read_byte(rtlpriv, macreg[i]); 363 macbackup[i] = rtl_read_dword(rtlpriv, macreg[i]); 364 } 365 EXPORT_SYMBOL_GPL(rtl8723_phy_save_mac_registers); 366 367 void rtl8723_phy_reload_adda_registers(struct ieee80211_hw *hw, 368 u32 *addareg, u32 *addabackup, 369 u32 regiesternum) 370 { 371 u32 i; 372 373 for (i = 0; i < regiesternum; i++) 374 rtl_set_bbreg(hw, addareg[i], MASKDWORD, addabackup[i]); 375 } 376 EXPORT_SYMBOL_GPL(rtl8723_phy_reload_adda_registers); 377 378 void rtl8723_phy_reload_mac_registers(struct ieee80211_hw *hw, 379 u32 *macreg, u32 *macbackup) 380 { 381 struct rtl_priv *rtlpriv = rtl_priv(hw); 382 u32 i; 383 384 for (i = 0; i < (IQK_MAC_REG_NUM - 1); i++) 385 rtl_write_byte(rtlpriv, macreg[i], (u8) macbackup[i]); 386 rtl_write_dword(rtlpriv, macreg[i], macbackup[i]); 387 } 388 EXPORT_SYMBOL_GPL(rtl8723_phy_reload_mac_registers); 389 390 void rtl8723_phy_path_adda_on(struct ieee80211_hw *hw, u32 *addareg, 391 bool is_patha_on, bool is2t) 392 { 393 struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); 394 u32 pathon; 395 u32 i; 396 397 if (rtlhal->hw_type == HARDWARE_TYPE_RTL8723AE) { 398 pathon = is_patha_on ? 0x04db25a4 : 0x0b1b25a4; 399 if (!is2t) { 400 pathon = 0x0bdb25a0; 401 rtl_set_bbreg(hw, addareg[0], MASKDWORD, 0x0b1b25a0); 402 } else { 403 rtl_set_bbreg(hw, addareg[0], MASKDWORD, pathon); 404 } 405 } else { 406 /* rtl8723be */ 407 pathon = 0x01c00014; 408 rtl_set_bbreg(hw, addareg[0], MASKDWORD, pathon); 409 } 410 411 for (i = 1; i < IQK_ADDA_REG_NUM; i++) 412 rtl_set_bbreg(hw, addareg[i], MASKDWORD, pathon); 413 } 414 EXPORT_SYMBOL_GPL(rtl8723_phy_path_adda_on); 415 416 void rtl8723_phy_mac_setting_calibration(struct ieee80211_hw *hw, 417 u32 *macreg, u32 *macbackup) 418 { 419 struct rtl_priv *rtlpriv = rtl_priv(hw); 420 u32 i = 0; 421 422 rtl_write_byte(rtlpriv, macreg[i], 0x3F); 423 424 for (i = 1; i < (IQK_MAC_REG_NUM - 1); i++) 425 rtl_write_byte(rtlpriv, macreg[i], 426 (u8) (macbackup[i] & (~BIT(3)))); 427 rtl_write_byte(rtlpriv, macreg[i], (u8) (macbackup[i] & (~BIT(5)))); 428 } 429 EXPORT_SYMBOL_GPL(rtl8723_phy_mac_setting_calibration); 430 431 void rtl8723_phy_path_a_standby(struct ieee80211_hw *hw) 432 { 433 rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x0); 434 rtl_set_bbreg(hw, 0x840, MASKDWORD, 0x00010000); 435 rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x80800000); 436 } 437 EXPORT_SYMBOL_GPL(rtl8723_phy_path_a_standby); 438 439 void rtl8723_phy_pi_mode_switch(struct ieee80211_hw *hw, bool pi_mode) 440 { 441 u32 mode; 442 443 mode = pi_mode ? 0x01000100 : 0x01000000; 444 rtl_set_bbreg(hw, 0x820, MASKDWORD, mode); 445 rtl_set_bbreg(hw, 0x828, MASKDWORD, mode); 446 } 447 EXPORT_SYMBOL_GPL(rtl8723_phy_pi_mode_switch); 448