phy.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605
  1. /******************************************************************************
  2. *
  3. * Copyright(c) 2009-2010 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. * You should have received a copy of the GNU General Public License along with
  15. * this program; if not, write to the Free Software Foundation, Inc.,
  16. * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
  17. *
  18. * The full GNU General Public License is included in this distribution in the
  19. * file called LICENSE.
  20. *
  21. * Contact Information:
  22. * wlanfae <wlanfae@realtek.com>
  23. * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park,
  24. * Hsinchu 300, Taiwan.
  25. *
  26. * Larry Finger <Larry.Finger@lwfinger.net>
  27. *
  28. *****************************************************************************/
  29. #include "../wifi.h"
  30. #include "../pci.h"
  31. #include "../ps.h"
  32. #include "reg.h"
  33. #include "def.h"
  34. #include "phy.h"
  35. #include "rf.h"
  36. #include "dm.h"
  37. #include "table.h"
  38. u32 rtl92cu_phy_query_rf_reg(struct ieee80211_hw *hw,
  39. enum radio_path rfpath, u32 regaddr, u32 bitmask)
  40. {
  41. struct rtl_priv *rtlpriv = rtl_priv(hw);
  42. u32 original_value, readback_value, bitshift;
  43. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  44. RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), "
  45. "rfpath(%#x), bitmask(%#x)\n",
  46. regaddr, rfpath, bitmask));
  47. if (rtlphy->rf_mode != RF_OP_BY_FW) {
  48. original_value = _rtl92c_phy_rf_serial_read(hw,
  49. rfpath, regaddr);
  50. } else {
  51. original_value = _rtl92c_phy_fw_rf_serial_read(hw,
  52. rfpath, regaddr);
  53. }
  54. bitshift = _rtl92c_phy_calculate_bit_shift(bitmask);
  55. readback_value = (original_value & bitmask) >> bitshift;
  56. RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE,
  57. ("regaddr(%#x), rfpath(%#x), "
  58. "bitmask(%#x), original_value(%#x)\n",
  59. regaddr, rfpath, bitmask, original_value));
  60. return readback_value;
  61. }
  62. void rtl92cu_phy_set_rf_reg(struct ieee80211_hw *hw,
  63. enum radio_path rfpath,
  64. u32 regaddr, u32 bitmask, u32 data)
  65. {
  66. struct rtl_priv *rtlpriv = rtl_priv(hw);
  67. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  68. u32 original_value, bitshift;
  69. RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE,
  70. ("regaddr(%#x), bitmask(%#x), data(%#x), rfpath(%#x)\n",
  71. regaddr, bitmask, data, rfpath));
  72. if (rtlphy->rf_mode != RF_OP_BY_FW) {
  73. if (bitmask != RFREG_OFFSET_MASK) {
  74. original_value = _rtl92c_phy_rf_serial_read(hw,
  75. rfpath,
  76. regaddr);
  77. bitshift = _rtl92c_phy_calculate_bit_shift(bitmask);
  78. data =
  79. ((original_value & (~bitmask)) |
  80. (data << bitshift));
  81. }
  82. _rtl92c_phy_rf_serial_write(hw, rfpath, regaddr, data);
  83. } else {
  84. if (bitmask != RFREG_OFFSET_MASK) {
  85. original_value = _rtl92c_phy_fw_rf_serial_read(hw,
  86. rfpath,
  87. regaddr);
  88. bitshift = _rtl92c_phy_calculate_bit_shift(bitmask);
  89. data =
  90. ((original_value & (~bitmask)) |
  91. (data << bitshift));
  92. }
  93. _rtl92c_phy_fw_rf_serial_write(hw, rfpath, regaddr, data);
  94. }
  95. RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), "
  96. "bitmask(%#x), data(%#x), rfpath(%#x)\n",
  97. regaddr, bitmask, data, rfpath));
  98. }
  99. bool rtl92cu_phy_mac_config(struct ieee80211_hw *hw)
  100. {
  101. bool rtstatus;
  102. struct rtl_priv *rtlpriv = rtl_priv(hw);
  103. struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
  104. bool is92c = IS_92C_SERIAL(rtlhal->version);
  105. rtstatus = _rtl92cu_phy_config_mac_with_headerfile(hw);
  106. if (is92c && IS_HARDWARE_TYPE_8192CE(rtlhal))
  107. rtl_write_byte(rtlpriv, 0x14, 0x71);
  108. return rtstatus;
  109. }
  110. bool rtl92cu_phy_bb_config(struct ieee80211_hw *hw)
  111. {
  112. bool rtstatus = true;
  113. struct rtl_priv *rtlpriv = rtl_priv(hw);
  114. struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
  115. u16 regval;
  116. u8 b_reg_hwparafile = 1;
  117. _rtl92c_phy_init_bb_rf_register_definition(hw);
  118. regval = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN);
  119. rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, regval | BIT(13) |
  120. BIT(0) | BIT(1));
  121. rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL, 0x83);
  122. rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL + 1, 0xdb);
  123. rtl_write_byte(rtlpriv, REG_RF_CTRL, RF_EN | RF_RSTB | RF_SDMRSTB);
  124. if (IS_HARDWARE_TYPE_8192CE(rtlhal)) {
  125. rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, FEN_PPLL | FEN_PCIEA |
  126. FEN_DIO_PCIE | FEN_BB_GLB_RSTn | FEN_BBRSTB);
  127. } else if (IS_HARDWARE_TYPE_8192CU(rtlhal)) {
  128. rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, FEN_USBA | FEN_USBD |
  129. FEN_BB_GLB_RSTn | FEN_BBRSTB);
  130. rtl_write_byte(rtlpriv, REG_LDOHCI12_CTRL, 0x0f);
  131. }
  132. rtl_write_byte(rtlpriv, REG_AFE_XTAL_CTRL + 1, 0x80);
  133. if (b_reg_hwparafile == 1)
  134. rtstatus = _rtl92c_phy_bb8192c_config_parafile(hw);
  135. return rtstatus;
  136. }
  137. bool _rtl92cu_phy_config_mac_with_headerfile(struct ieee80211_hw *hw)
  138. {
  139. struct rtl_priv *rtlpriv = rtl_priv(hw);
  140. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  141. u32 i;
  142. u32 arraylength;
  143. u32 *ptrarray;
  144. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Read Rtl819XMACPHY_Array\n"));
  145. arraylength = rtlphy->hwparam_tables[MAC_REG].length ;
  146. ptrarray = rtlphy->hwparam_tables[MAC_REG].pdata;
  147. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  148. ("Img:RTL8192CEMAC_2T_ARRAY\n"));
  149. for (i = 0; i < arraylength; i = i + 2)
  150. rtl_write_byte(rtlpriv, ptrarray[i], (u8) ptrarray[i + 1]);
  151. return true;
  152. }
  153. bool _rtl92cu_phy_config_bb_with_headerfile(struct ieee80211_hw *hw,
  154. u8 configtype)
  155. {
  156. int i;
  157. u32 *phy_regarray_table;
  158. u32 *agctab_array_table;
  159. u16 phy_reg_arraylen, agctab_arraylen;
  160. struct rtl_priv *rtlpriv = rtl_priv(hw);
  161. struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
  162. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  163. if (IS_92C_SERIAL(rtlhal->version)) {
  164. agctab_arraylen = rtlphy->hwparam_tables[AGCTAB_2T].length;
  165. agctab_array_table = rtlphy->hwparam_tables[AGCTAB_2T].pdata;
  166. phy_reg_arraylen = rtlphy->hwparam_tables[PHY_REG_2T].length;
  167. phy_regarray_table = rtlphy->hwparam_tables[PHY_REG_2T].pdata;
  168. } else {
  169. agctab_arraylen = rtlphy->hwparam_tables[AGCTAB_1T].length;
  170. agctab_array_table = rtlphy->hwparam_tables[AGCTAB_1T].pdata;
  171. phy_reg_arraylen = rtlphy->hwparam_tables[PHY_REG_1T].length;
  172. phy_regarray_table = rtlphy->hwparam_tables[PHY_REG_1T].pdata;
  173. }
  174. if (configtype == BASEBAND_CONFIG_PHY_REG) {
  175. for (i = 0; i < phy_reg_arraylen; i = i + 2) {
  176. if (phy_regarray_table[i] == 0xfe)
  177. mdelay(50);
  178. else if (phy_regarray_table[i] == 0xfd)
  179. mdelay(5);
  180. else if (phy_regarray_table[i] == 0xfc)
  181. mdelay(1);
  182. else if (phy_regarray_table[i] == 0xfb)
  183. udelay(50);
  184. else if (phy_regarray_table[i] == 0xfa)
  185. udelay(5);
  186. else if (phy_regarray_table[i] == 0xf9)
  187. udelay(1);
  188. rtl_set_bbreg(hw, phy_regarray_table[i], MASKDWORD,
  189. phy_regarray_table[i + 1]);
  190. udelay(1);
  191. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  192. ("The phy_regarray_table[0] is %x"
  193. " Rtl819XPHY_REGArray[1] is %x\n",
  194. phy_regarray_table[i],
  195. phy_regarray_table[i + 1]));
  196. }
  197. } else if (configtype == BASEBAND_CONFIG_AGC_TAB) {
  198. for (i = 0; i < agctab_arraylen; i = i + 2) {
  199. rtl_set_bbreg(hw, agctab_array_table[i], MASKDWORD,
  200. agctab_array_table[i + 1]);
  201. udelay(1);
  202. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  203. ("The agctab_array_table[0] is "
  204. "%x Rtl819XPHY_REGArray[1] is %x\n",
  205. agctab_array_table[i],
  206. agctab_array_table[i + 1]));
  207. }
  208. }
  209. return true;
  210. }
  211. bool _rtl92cu_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw,
  212. u8 configtype)
  213. {
  214. struct rtl_priv *rtlpriv = rtl_priv(hw);
  215. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  216. int i;
  217. u32 *phy_regarray_table_pg;
  218. u16 phy_regarray_pg_len;
  219. rtlphy->pwrgroup_cnt = 0;
  220. phy_regarray_pg_len = rtlphy->hwparam_tables[PHY_REG_PG].length;
  221. phy_regarray_table_pg = rtlphy->hwparam_tables[PHY_REG_PG].pdata;
  222. if (configtype == BASEBAND_CONFIG_PHY_REG) {
  223. for (i = 0; i < phy_regarray_pg_len; i = i + 3) {
  224. if (phy_regarray_table_pg[i] == 0xfe)
  225. mdelay(50);
  226. else if (phy_regarray_table_pg[i] == 0xfd)
  227. mdelay(5);
  228. else if (phy_regarray_table_pg[i] == 0xfc)
  229. mdelay(1);
  230. else if (phy_regarray_table_pg[i] == 0xfb)
  231. udelay(50);
  232. else if (phy_regarray_table_pg[i] == 0xfa)
  233. udelay(5);
  234. else if (phy_regarray_table_pg[i] == 0xf9)
  235. udelay(1);
  236. _rtl92c_store_pwrIndex_diffrate_offset(hw,
  237. phy_regarray_table_pg[i],
  238. phy_regarray_table_pg[i + 1],
  239. phy_regarray_table_pg[i + 2]);
  240. }
  241. } else {
  242. RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE,
  243. ("configtype != BaseBand_Config_PHY_REG\n"));
  244. }
  245. return true;
  246. }
  247. bool rtl92cu_phy_config_rf_with_headerfile(struct ieee80211_hw *hw,
  248. enum radio_path rfpath)
  249. {
  250. int i;
  251. u32 *radioa_array_table;
  252. u32 *radiob_array_table;
  253. u16 radioa_arraylen, radiob_arraylen;
  254. struct rtl_priv *rtlpriv = rtl_priv(hw);
  255. struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
  256. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  257. if (IS_92C_SERIAL(rtlhal->version)) {
  258. radioa_arraylen = rtlphy->hwparam_tables[RADIOA_2T].length;
  259. radioa_array_table = rtlphy->hwparam_tables[RADIOA_2T].pdata;
  260. radiob_arraylen = rtlphy->hwparam_tables[RADIOB_2T].length;
  261. radiob_array_table = rtlphy->hwparam_tables[RADIOB_2T].pdata;
  262. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  263. ("Radio_A:RTL8192CERADIOA_2TARRAY\n"));
  264. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  265. ("Radio_B:RTL8192CE_RADIOB_2TARRAY\n"));
  266. } else {
  267. radioa_arraylen = rtlphy->hwparam_tables[RADIOA_1T].length;
  268. radioa_array_table = rtlphy->hwparam_tables[RADIOA_1T].pdata;
  269. radiob_arraylen = rtlphy->hwparam_tables[RADIOB_1T].length;
  270. radiob_array_table = rtlphy->hwparam_tables[RADIOB_1T].pdata;
  271. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  272. ("Radio_A:RTL8192CE_RADIOA_1TARRAY\n"));
  273. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE,
  274. ("Radio_B:RTL8192CE_RADIOB_1TARRAY\n"));
  275. }
  276. RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Radio No %x\n", rfpath));
  277. switch (rfpath) {
  278. case RF90_PATH_A:
  279. for (i = 0; i < radioa_arraylen; i = i + 2) {
  280. if (radioa_array_table[i] == 0xfe)
  281. mdelay(50);
  282. else if (radioa_array_table[i] == 0xfd)
  283. mdelay(5);
  284. else if (radioa_array_table[i] == 0xfc)
  285. mdelay(1);
  286. else if (radioa_array_table[i] == 0xfb)
  287. udelay(50);
  288. else if (radioa_array_table[i] == 0xfa)
  289. udelay(5);
  290. else if (radioa_array_table[i] == 0xf9)
  291. udelay(1);
  292. else {
  293. rtl_set_rfreg(hw, rfpath, radioa_array_table[i],
  294. RFREG_OFFSET_MASK,
  295. radioa_array_table[i + 1]);
  296. udelay(1);
  297. }
  298. }
  299. break;
  300. case RF90_PATH_B:
  301. for (i = 0; i < radiob_arraylen; i = i + 2) {
  302. if (radiob_array_table[i] == 0xfe) {
  303. mdelay(50);
  304. } else if (radiob_array_table[i] == 0xfd)
  305. mdelay(5);
  306. else if (radiob_array_table[i] == 0xfc)
  307. mdelay(1);
  308. else if (radiob_array_table[i] == 0xfb)
  309. udelay(50);
  310. else if (radiob_array_table[i] == 0xfa)
  311. udelay(5);
  312. else if (radiob_array_table[i] == 0xf9)
  313. udelay(1);
  314. else {
  315. rtl_set_rfreg(hw, rfpath, radiob_array_table[i],
  316. RFREG_OFFSET_MASK,
  317. radiob_array_table[i + 1]);
  318. udelay(1);
  319. }
  320. }
  321. break;
  322. case RF90_PATH_C:
  323. RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG,
  324. ("switch case not process\n"));
  325. break;
  326. case RF90_PATH_D:
  327. RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG,
  328. ("switch case not process\n"));
  329. break;
  330. }
  331. return true;
  332. }
  333. void rtl92cu_phy_set_bw_mode_callback(struct ieee80211_hw *hw)
  334. {
  335. struct rtl_priv *rtlpriv = rtl_priv(hw);
  336. struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw));
  337. struct rtl_phy *rtlphy = &(rtlpriv->phy);
  338. struct rtl_mac *mac = rtl_mac(rtl_priv(hw));
  339. u8 reg_bw_opmode;
  340. u8 reg_prsr_rsc;
  341. RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE,
  342. ("Switch to %s bandwidth\n",
  343. rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20 ?
  344. "20MHz" : "40MHz"))
  345. if (is_hal_stop(rtlhal)) {
  346. rtlphy->set_bwmode_inprogress = false;
  347. return;
  348. }
  349. reg_bw_opmode = rtl_read_byte(rtlpriv, REG_BWOPMODE);
  350. reg_prsr_rsc = rtl_read_byte(rtlpriv, REG_RRSR + 2);
  351. switch (rtlphy->current_chan_bw) {
  352. case HT_CHANNEL_WIDTH_20:
  353. reg_bw_opmode |= BW_OPMODE_20MHZ;
  354. rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode);
  355. break;
  356. case HT_CHANNEL_WIDTH_20_40:
  357. reg_bw_opmode &= ~BW_OPMODE_20MHZ;
  358. rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode);
  359. reg_prsr_rsc =
  360. (reg_prsr_rsc & 0x90) | (mac->cur_40_prime_sc << 5);
  361. rtl_write_byte(rtlpriv, REG_RRSR + 2, reg_prsr_rsc);
  362. break;
  363. default:
  364. RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG,
  365. ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw));
  366. break;
  367. }
  368. switch (rtlphy->current_chan_bw) {
  369. case HT_CHANNEL_WIDTH_20:
  370. rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x0);
  371. rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x0);
  372. rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 1);
  373. break;
  374. case HT_CHANNEL_WIDTH_20_40:
  375. rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x1);
  376. rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x1);
  377. rtl_set_bbreg(hw, RCCK0_SYSTEM, BCCK_SIDEBAND,
  378. (mac->cur_40_prime_sc >> 1));
  379. rtl_set_bbreg(hw, ROFDM1_LSTF, 0xC00, mac->cur_40_prime_sc);
  380. rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 0);
  381. rtl_set_bbreg(hw, 0x818, (BIT(26) | BIT(27)),
  382. (mac->cur_40_prime_sc ==
  383. HAL_PRIME_CHNL_OFFSET_LOWER) ? 2 : 1);
  384. break;
  385. default:
  386. RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG,
  387. ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw));
  388. break;
  389. }
  390. rtl92cu_phy_rf6052_set_bandwidth(hw, rtlphy->current_chan_bw);
  391. rtlphy->set_bwmode_inprogress = false;
  392. RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n"));
  393. }
  394. void rtl92cu_bb_block_on(struct ieee80211_hw *hw)
  395. {
  396. struct rtl_priv *rtlpriv = rtl_priv(hw);
  397. mutex_lock(&rtlpriv->io.bb_mutex);
  398. rtl_set_bbreg(hw, RFPGA0_RFMOD, BCCKEN, 0x1);
  399. rtl_set_bbreg(hw, RFPGA0_RFMOD, BOFDMEN, 0x1);
  400. mutex_unlock(&rtlpriv->io.bb_mutex);
  401. }
  402. void _rtl92cu_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t)
  403. {
  404. u8 tmpreg;
  405. u32 rf_a_mode = 0, rf_b_mode = 0, lc_cal;
  406. struct rtl_priv *rtlpriv = rtl_priv(hw);
  407. tmpreg = rtl_read_byte(rtlpriv, 0xd03);
  408. if ((tmpreg & 0x70) != 0)
  409. rtl_write_byte(rtlpriv, 0xd03, tmpreg & 0x8F);
  410. else
  411. rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF);
  412. if ((tmpreg & 0x70) != 0) {
  413. rf_a_mode = rtl_get_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS);
  414. if (is2t)
  415. rf_b_mode = rtl_get_rfreg(hw, RF90_PATH_B, 0x00,
  416. MASK12BITS);
  417. rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS,
  418. (rf_a_mode & 0x8FFFF) | 0x10000);
  419. if (is2t)
  420. rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS,
  421. (rf_b_mode & 0x8FFFF) | 0x10000);
  422. }
  423. lc_cal = rtl_get_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS);
  424. rtl_set_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS, lc_cal | 0x08000);
  425. mdelay(100);
  426. if ((tmpreg & 0x70) != 0) {
  427. rtl_write_byte(rtlpriv, 0xd03, tmpreg);
  428. rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS, rf_a_mode);
  429. if (is2t)
  430. rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS,
  431. rf_b_mode);
  432. } else {
  433. rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00);
  434. }
  435. }
  436. static bool _rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw,
  437. enum rf_pwrstate rfpwr_state)
  438. {
  439. struct rtl_priv *rtlpriv = rtl_priv(hw);
  440. struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw);
  441. struct rtl_mac *mac = rtl_mac(rtl_priv(hw));
  442. struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw));
  443. bool bresult = true;
  444. u8 i, queue_id;
  445. struct rtl8192_tx_ring *ring = NULL;
  446. switch (rfpwr_state) {
  447. case ERFON:
  448. if ((ppsc->rfpwr_state == ERFOFF) &&
  449. RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC)) {
  450. bool rtstatus;
  451. u32 InitializeCount = 0;
  452. do {
  453. InitializeCount++;
  454. RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG,
  455. ("IPS Set eRf nic enable\n"));
  456. rtstatus = rtl_ps_enable_nic(hw);
  457. } while ((rtstatus != true)
  458. && (InitializeCount < 10));
  459. RT_CLEAR_PS_LEVEL(ppsc,
  460. RT_RF_OFF_LEVL_HALT_NIC);
  461. } else {
  462. RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG,
  463. ("Set ERFON sleeped:%d ms\n",
  464. jiffies_to_msecs(jiffies -
  465. ppsc->
  466. last_sleep_jiffies)));
  467. ppsc->last_awake_jiffies = jiffies;
  468. rtl92ce_phy_set_rf_on(hw);
  469. }
  470. if (mac->link_state == MAC80211_LINKED) {
  471. rtlpriv->cfg->ops->led_control(hw,
  472. LED_CTL_LINK);
  473. } else {
  474. rtlpriv->cfg->ops->led_control(hw,
  475. LED_CTL_NO_LINK);
  476. }
  477. break;
  478. case ERFOFF:
  479. for (queue_id = 0, i = 0;
  480. queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) {
  481. ring = &pcipriv->dev.tx_ring[queue_id];
  482. if (skb_queue_len(&ring->queue) == 0 ||
  483. queue_id == BEACON_QUEUE) {
  484. queue_id++;
  485. continue;
  486. } else {
  487. RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING,
  488. ("eRf Off/Sleep: %d times "
  489. "TcbBusyQueue[%d] "
  490. "=%d before doze!\n", (i + 1),
  491. queue_id,
  492. skb_queue_len(&ring->queue)));
  493. udelay(10);
  494. i++;
  495. }
  496. if (i >= MAX_DOZE_WAITING_TIMES_9x) {
  497. RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING,
  498. ("\nERFOFF: %d times "
  499. "TcbBusyQueue[%d] = %d !\n",
  500. MAX_DOZE_WAITING_TIMES_9x,
  501. queue_id,
  502. skb_queue_len(&ring->queue)));
  503. break;
  504. }
  505. }
  506. if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC) {
  507. RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG,
  508. ("IPS Set eRf nic disable\n"));
  509. rtl_ps_disable_nic(hw);
  510. RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC);
  511. } else {
  512. if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS) {
  513. rtlpriv->cfg->ops->led_control(hw,
  514. LED_CTL_NO_LINK);
  515. } else {
  516. rtlpriv->cfg->ops->led_control(hw,
  517. LED_CTL_POWER_OFF);
  518. }
  519. }
  520. break;
  521. case ERFSLEEP:
  522. if (ppsc->rfpwr_state == ERFOFF)
  523. break;
  524. for (queue_id = 0, i = 0;
  525. queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) {
  526. ring = &pcipriv->dev.tx_ring[queue_id];
  527. if (skb_queue_len(&ring->queue) == 0) {
  528. queue_id++;
  529. continue;
  530. } else {
  531. RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING,
  532. ("eRf Off/Sleep: %d times "
  533. "TcbBusyQueue[%d] =%d before "
  534. "doze!\n", (i + 1), queue_id,
  535. skb_queue_len(&ring->queue)));
  536. udelay(10);
  537. i++;
  538. }
  539. if (i >= MAX_DOZE_WAITING_TIMES_9x) {
  540. RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING,
  541. ("\n ERFSLEEP: %d times "
  542. "TcbBusyQueue[%d] = %d !\n",
  543. MAX_DOZE_WAITING_TIMES_9x,
  544. queue_id,
  545. skb_queue_len(&ring->queue)));
  546. break;
  547. }
  548. }
  549. RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG,
  550. ("Set ERFSLEEP awaked:%d ms\n",
  551. jiffies_to_msecs(jiffies -
  552. ppsc->last_awake_jiffies)));
  553. ppsc->last_sleep_jiffies = jiffies;
  554. _rtl92c_phy_set_rf_sleep(hw);
  555. break;
  556. default:
  557. RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG,
  558. ("switch case not process\n"));
  559. bresult = false;
  560. break;
  561. }
  562. if (bresult)
  563. ppsc->rfpwr_state = rfpwr_state;
  564. return bresult;
  565. }
  566. bool rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw,
  567. enum rf_pwrstate rfpwr_state)
  568. {
  569. struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw));
  570. bool bresult = false;
  571. if (rfpwr_state == ppsc->rfpwr_state)
  572. return bresult;
  573. bresult = _rtl92cu_phy_set_rf_power_state(hw, rfpwr_state);
  574. return bresult;
  575. }