linux/drivers/net/wireless/realtek/rtlwifi/rtl8723com/phy_common.c
<<
>>
Prefs
   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
  33u32 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}
  50EXPORT_SYMBOL_GPL(rtl8723_phy_query_bb_reg);
  51
  52void 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}
  74EXPORT_SYMBOL_GPL(rtl8723_phy_set_bb_reg);
  75
  76u32 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}
  86EXPORT_SYMBOL_GPL(rtl8723_phy_calculate_bit_shift);
  87
  88u32 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                pr_err("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}
 137EXPORT_SYMBOL_GPL(rtl8723_phy_rf_serial_read);
 138
 139void 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                pr_err("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}
 162EXPORT_SYMBOL_GPL(rtl8723_phy_rf_serial_write);
 163
 164long 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}
 186EXPORT_SYMBOL_GPL(rtl8723_phy_txpwr_idx_to_dbm);
 187
 188void 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}
 274EXPORT_SYMBOL_GPL(rtl8723_phy_init_bb_rf_reg_def);
 275
 276bool 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                WARN_ONCE(true, "rtl8723-common: 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}
 300EXPORT_SYMBOL_GPL(rtl8723_phy_set_sw_chnl_cmdarray);
 301
 302void 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}
 343EXPORT_SYMBOL_GPL(rtl8723_phy_path_a_fill_iqk_matrix);
 344
 345void 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}
 353EXPORT_SYMBOL_GPL(rtl8723_save_adda_registers);
 354
 355void 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}
 365EXPORT_SYMBOL_GPL(rtl8723_phy_save_mac_registers);
 366
 367void 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}
 376EXPORT_SYMBOL_GPL(rtl8723_phy_reload_adda_registers);
 377
 378void 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}
 388EXPORT_SYMBOL_GPL(rtl8723_phy_reload_mac_registers);
 389
 390void 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}
 414EXPORT_SYMBOL_GPL(rtl8723_phy_path_adda_on);
 415
 416void 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}
 429EXPORT_SYMBOL_GPL(rtl8723_phy_mac_setting_calibration);
 430
 431void 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}
 437EXPORT_SYMBOL_GPL(rtl8723_phy_path_a_standby);
 438
 439void 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}
 447EXPORT_SYMBOL_GPL(rtl8723_phy_pi_mode_switch);
 448