phydm_kfree.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. /******************************************************************************
  2. *
  3. * Copyright(c) 2007 - 2011 Realtek Corporation. All rights reserved.
  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. *
  19. ******************************************************************************/
  20. /*============================================================*/
  21. /*include files*/
  22. /*============================================================*/
  23. #include "mp_precomp.h"
  24. #include "phydm_precomp.h"
  25. /*<YuChen, 150720> Add for KFree Feature Requested by RF David.*/
  26. /*This is a phydm API*/
  27. void
  28. phydm_set_kfree_to_rf_8814a(
  29. void *p_dm_void,
  30. u8 e_rf_path,
  31. u8 data
  32. )
  33. {
  34. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  35. struct odm_rf_calibration_structure *p_rf_calibrate_info = &(p_dm_odm->rf_calibrate_info);
  36. boolean is_odd;
  37. if ((data % 2) != 0) { /*odd->positive*/
  38. data = data - 1;
  39. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(19), 1);
  40. is_odd = true;
  41. } else { /*even->negative*/
  42. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(19), 0);
  43. is_odd = false;
  44. }
  45. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("phy_ConfigKFree8814A(): RF_0x55[19]= %d\n", is_odd));
  46. switch (data) {
  47. case 0:
  48. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  49. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 0);
  50. p_rf_calibrate_info->kfree_offset[e_rf_path] = 0;
  51. break;
  52. case 2:
  53. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 1);
  54. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 0);
  55. p_rf_calibrate_info->kfree_offset[e_rf_path] = 0;
  56. break;
  57. case 4:
  58. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  59. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 1);
  60. p_rf_calibrate_info->kfree_offset[e_rf_path] = 1;
  61. break;
  62. case 6:
  63. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 1);
  64. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 1);
  65. p_rf_calibrate_info->kfree_offset[e_rf_path] = 1;
  66. break;
  67. case 8:
  68. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  69. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 2);
  70. p_rf_calibrate_info->kfree_offset[e_rf_path] = 2;
  71. break;
  72. case 10:
  73. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 1);
  74. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 2);
  75. p_rf_calibrate_info->kfree_offset[e_rf_path] = 2;
  76. break;
  77. case 12:
  78. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  79. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 3);
  80. p_rf_calibrate_info->kfree_offset[e_rf_path] = 3;
  81. break;
  82. case 14:
  83. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 1);
  84. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 3);
  85. p_rf_calibrate_info->kfree_offset[e_rf_path] = 3;
  86. break;
  87. case 16:
  88. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  89. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 4);
  90. p_rf_calibrate_info->kfree_offset[e_rf_path] = 4;
  91. break;
  92. case 18:
  93. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 1);
  94. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 4);
  95. p_rf_calibrate_info->kfree_offset[e_rf_path] = 4;
  96. break;
  97. case 20:
  98. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(14), 0);
  99. odm_set_rf_reg(p_dm_odm, e_rf_path, REG_RF_TX_GAIN_OFFSET, BIT(17) | BIT(16) | BIT(15), 5);
  100. p_rf_calibrate_info->kfree_offset[e_rf_path] = 5;
  101. break;
  102. default:
  103. break;
  104. }
  105. if (is_odd == false) {
  106. /*that means Kfree offset is negative, we need to record it.*/
  107. p_rf_calibrate_info->kfree_offset[e_rf_path] = (-1) * p_rf_calibrate_info->kfree_offset[e_rf_path];
  108. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("phy_ConfigKFree8814A(): kfree_offset = %d\n", p_rf_calibrate_info->kfree_offset[e_rf_path]));
  109. } else
  110. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("phy_ConfigKFree8814A(): kfree_offset = %d\n", p_rf_calibrate_info->kfree_offset[e_rf_path]));
  111. }
  112. void
  113. phydm_get_thermal_trim_offset_8821c(
  114. void *p_dm_void
  115. )
  116. {
  117. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  118. struct odm_power_trim_data *p_power_trim_info = &(p_dm_odm->power_trim_data);
  119. u8 pg_therm = 0xff;
  120. odm_efuse_one_byte_read(p_dm_odm, PPG_THERMAL_OFFSET_8821C, &pg_therm, false);
  121. if (pg_therm != 0xff) {
  122. pg_therm = pg_therm & 0x1f;
  123. if ((pg_therm & BIT0) == 0)
  124. p_power_trim_info->thermal = (-1 * (pg_therm >> 1));
  125. else
  126. p_power_trim_info->thermal = (pg_therm >> 1);
  127. p_power_trim_info->flag |= KFREE_FLAG_THERMAL_K_ON;
  128. }
  129. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] power trim flag:0x%02x\n", p_power_trim_info->flag));
  130. if (p_power_trim_info->flag & KFREE_FLAG_THERMAL_K_ON)
  131. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] thermal:%d\n", p_power_trim_info->thermal));
  132. }
  133. void
  134. phydm_get_power_trim_offset_8821c(
  135. void *p_dm_void
  136. )
  137. {
  138. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  139. struct odm_power_trim_data *p_power_trim_info = &(p_dm_odm->power_trim_data);
  140. u8 pg_power = 0xff, i;
  141. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_2G_TXA_OFFSET_8821C, &pg_power, false);
  142. if (pg_power != 0xff) {
  143. p_power_trim_info->bb_gain[0][0] = pg_power;
  144. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_5GL1_TXA_OFFSET_8821C, &pg_power, false);
  145. p_power_trim_info->bb_gain[1][0] = pg_power;
  146. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_5GL2_TXA_OFFSET_8821C, &pg_power, false);
  147. p_power_trim_info->bb_gain[2][0] = pg_power;
  148. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_5GM1_TXA_OFFSET_8821C, &pg_power, false);
  149. p_power_trim_info->bb_gain[3][0] = pg_power;
  150. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_5GM2_TXA_OFFSET_8821C, &pg_power, false);
  151. p_power_trim_info->bb_gain[4][0] = pg_power;
  152. odm_efuse_one_byte_read(p_dm_odm, PPG_BB_GAIN_5GH1_TXA_OFFSET_8821C, &pg_power, false);
  153. p_power_trim_info->bb_gain[5][0] = pg_power;
  154. p_power_trim_info->flag |= KFREE_FLAG_ON;
  155. }
  156. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] power trim flag:0x%02x\n", p_power_trim_info->flag));
  157. if (p_power_trim_info->flag & KFREE_FLAG_ON) {
  158. for (i = 0; i < 6; i++)
  159. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] power_trim_data->bb_gain[%d][0]=0x%X\n", i, p_power_trim_info->bb_gain[i][0]));
  160. }
  161. }
  162. void
  163. phydm_set_kfree_to_rf_8821c(
  164. void *p_dm_void,
  165. u8 e_rf_path,
  166. boolean wlg_btg,
  167. u8 data
  168. )
  169. {
  170. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  171. struct odm_rf_calibration_structure *p_rf_calibrate_info = &(p_dm_odm->rf_calibrate_info);
  172. boolean is_odd;
  173. u8 wlg, btg;
  174. odm_set_rf_reg(p_dm_odm, e_rf_path, 0xde, BIT(0), 1);
  175. odm_set_rf_reg(p_dm_odm, e_rf_path, 0xde, BIT(5), 1);
  176. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x55, BIT(6), 1);
  177. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x65, BIT(6), 1);
  178. if (wlg_btg == true) {
  179. wlg = data & 0xf;
  180. btg = (data & 0xf0) >> 4;
  181. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x55, BIT(19), (wlg & BIT(0)));
  182. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x55, (BIT(18) | BIT(17) | BIT(16) | BIT(15) | BIT(14)), (wlg >> 1));
  183. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x65, BIT(19), (btg & BIT(0)));
  184. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x65, (BIT(18) | BIT(17) | BIT(16) | BIT(15) | BIT(14)), (btg >> 1));
  185. } else {
  186. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x55, BIT(19), (data & BIT(0)));
  187. odm_set_rf_reg(p_dm_odm, e_rf_path, 0x55, (BIT(18) | BIT(17) | BIT(16) | BIT(15) | BIT(14)), ((data & 0x1f) >> 1));
  188. }
  189. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD,
  190. ("[kfree] 0x55[19:14]=0x%X 0x65[19:14]=0x%X\n",
  191. odm_get_rf_reg(p_dm_odm, e_rf_path, 0x55, (BIT(19) | BIT(18) | BIT(17) | BIT(16) | BIT(15) | BIT(14))),
  192. odm_get_rf_reg(p_dm_odm, e_rf_path, 0x65, (BIT(19) | BIT(18) | BIT(17) | BIT(16) | BIT(15) | BIT(14)))
  193. ));
  194. }
  195. void
  196. phydm_set_kfree_to_rf(
  197. void *p_dm_void,
  198. u8 e_rf_path,
  199. u8 data
  200. )
  201. {
  202. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  203. if (p_dm_odm->support_ic_type & ODM_RTL8814A)
  204. phydm_set_kfree_to_rf_8814a(p_dm_odm, e_rf_path, data);
  205. if ((p_dm_odm->support_ic_type & ODM_RTL8821C) && (*p_dm_odm->p_band_type == ODM_BAND_2_4G))
  206. phydm_set_kfree_to_rf_8821c(p_dm_odm, e_rf_path, true, data);
  207. else if (p_dm_odm->support_ic_type & ODM_RTL8821C)
  208. phydm_set_kfree_to_rf_8821c(p_dm_odm, e_rf_path, false, data);
  209. }
  210. void
  211. phydm_get_thermal_trim_offset(
  212. void *p_dm_void
  213. )
  214. {
  215. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  216. #if (DM_ODM_SUPPORT_TYPE & ODM_WIN)
  217. struct _ADAPTER *adapter = p_dm_odm->adapter;
  218. HAL_DATA_TYPE *p_hal_data = GET_HAL_DATA(adapter);
  219. PEFUSE_HAL pEfuseHal = &(p_hal_data->EfuseHal);
  220. u1Byte eFuseContent[DCMD_EFUSE_MAX_SECTION_NUM * EFUSE_MAX_WORD_UNIT * 2];
  221. if (HAL_MAC_Dump_EFUSE(&GET_HAL_MAC_INFO(adapter), EFUSE_WIFI, eFuseContent, pEfuseHal->PhysicalLen_WiFi, HAL_MAC_EFUSE_PHYSICAL, HAL_MAC_EFUSE_PARSE_DRV) != RT_STATUS_SUCCESS)
  222. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] dump efuse fail !!!\n"));
  223. #endif
  224. if (p_dm_odm->support_ic_type & ODM_RTL8821C)
  225. phydm_get_thermal_trim_offset_8821c(p_dm_void);
  226. }
  227. void
  228. phydm_get_power_trim_offset(
  229. void *p_dm_void
  230. )
  231. {
  232. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  233. #if (DM_ODM_SUPPORT_TYPE & ODM_WIN)
  234. struct _ADAPTER *adapter = p_dm_odm->adapter;
  235. HAL_DATA_TYPE *p_hal_data = GET_HAL_DATA(adapter);
  236. PEFUSE_HAL pEfuseHal = &(p_hal_data->EfuseHal);
  237. u1Byte eFuseContent[DCMD_EFUSE_MAX_SECTION_NUM * EFUSE_MAX_WORD_UNIT * 2];
  238. if (HAL_MAC_Dump_EFUSE(&GET_HAL_MAC_INFO(adapter), EFUSE_WIFI, eFuseContent, pEfuseHal->PhysicalLen_WiFi, HAL_MAC_EFUSE_PHYSICAL, HAL_MAC_EFUSE_PARSE_DRV) != RT_STATUS_SUCCESS)
  239. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] dump efuse fail !!!\n"));
  240. #endif
  241. if (p_dm_odm->support_ic_type & ODM_RTL8821C)
  242. phydm_get_power_trim_offset_8821c(p_dm_void);
  243. }
  244. s8
  245. phydm_get_thermal_offset(
  246. void *p_dm_void
  247. )
  248. {
  249. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  250. struct odm_power_trim_data *p_power_trim_info = &(p_dm_odm->power_trim_data);
  251. if (p_power_trim_info->flag & KFREE_FLAG_THERMAL_K_ON)
  252. return p_power_trim_info->thermal;
  253. else
  254. return 0;
  255. }
  256. void
  257. phydm_config_kfree(
  258. void *p_dm_void,
  259. u8 channel_to_sw
  260. )
  261. {
  262. struct PHY_DM_STRUCT *p_dm_odm = (struct PHY_DM_STRUCT *)p_dm_void;
  263. struct odm_rf_calibration_structure *p_rf_calibrate_info = &(p_dm_odm->rf_calibrate_info);
  264. struct odm_power_trim_data *p_power_trim_info = &(p_dm_odm->power_trim_data);
  265. u8 rfpath = 0, max_rf_path = 0;
  266. u8 channel_idx = 0, i;
  267. if (p_dm_odm->support_ic_type & ODM_RTL8814A)
  268. max_rf_path = 4; /*0~3*/
  269. else if (p_dm_odm->support_ic_type & (ODM_RTL8812 | ODM_RTL8192E | ODM_RTL8822B))
  270. max_rf_path = 2; /*0~1*/
  271. else if (p_dm_odm->support_ic_type & ODM_RTL8821C)
  272. max_rf_path = 1;
  273. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("===>[kfree] phy_ConfigKFree()\n"));
  274. if (p_rf_calibrate_info->reg_rf_kfree_enable == 2) {
  275. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] phy_ConfigKFree(): reg_rf_kfree_enable == 2, Disable\n"));
  276. return;
  277. } else if (p_rf_calibrate_info->reg_rf_kfree_enable == 1 || p_rf_calibrate_info->reg_rf_kfree_enable == 0) {
  278. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] phy_ConfigKFree(): reg_rf_kfree_enable == true\n"));
  279. /*Make sure the targetval is defined*/
  280. if (p_power_trim_info->flag & KFREE_FLAG_ON) {
  281. /*if kfree_table[0] == 0xff, means no Kfree*/
  282. if (*p_dm_odm->p_band_type == ODM_BAND_2_4G) {
  283. if (channel_to_sw >= 1 && channel_to_sw <= 14)
  284. channel_idx = PHYDM_2G;
  285. } else if (*p_dm_odm->p_band_type == ODM_BAND_5G) {
  286. if (channel_to_sw >= 36 && channel_to_sw <= 48)
  287. channel_idx = PHYDM_5GLB1;
  288. if (channel_to_sw >= 52 && channel_to_sw <= 64)
  289. channel_idx = PHYDM_5GLB2;
  290. if (channel_to_sw >= 100 && channel_to_sw <= 120)
  291. channel_idx = PHYDM_5GMB1;
  292. if (channel_to_sw >= 122 && channel_to_sw <= 144)
  293. channel_idx = PHYDM_5GMB2;
  294. if (channel_to_sw >= 149 && channel_to_sw <= 177)
  295. channel_idx = PHYDM_5GHB;
  296. }
  297. for (i = 0; i < (max_rf_path * BB_GAIN_NUM); i++)
  298. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] p_power_trim_info->bb_gain[%d][%d]=0x%X\n", i, max_rf_path - 1, p_power_trim_info->bb_gain[i][max_rf_path - 1]));
  299. for (rfpath = ODM_RF_PATH_A; rfpath < max_rf_path; rfpath++) {
  300. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] phydm_kfree(): channel_to_sw=%d PATH_%d: 0x%X\n", channel_to_sw, rfpath, p_power_trim_info->bb_gain[channel_idx][rfpath]));
  301. phydm_set_kfree_to_rf(p_dm_odm, rfpath, p_power_trim_info->bb_gain[channel_idx][rfpath]);
  302. }
  303. } else {
  304. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("[kfree] phy_ConfigKFree(): targetval not defined, Don't execute KFree Process.\n"));
  305. return;
  306. }
  307. }
  308. ODM_RT_TRACE(p_dm_odm, ODM_COMP_MP, ODM_DBG_LOUD, ("<===[kfree] phy_ConfigKFree()\n"));
  309. }