blob: 71d054900b8455e6a1fe7381e2d3494c007e4697 [file] [log] [blame]
Larry Fingerf7c92d22014-03-28 21:37:39 -05001/******************************************************************************
2 *
3 * Copyright(c) 2007 - 2012 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 ******************************************************************************/
15#define _HAL_INIT_C_
16
17#include <linux/firmware.h>
18#include <drv_types.h>
19#include <rtw_efuse.h>
20
21#include <rtl8723a_hal.h>
Jes Sorensen050abc42014-05-16 10:05:08 +020022#include <usb_ops_linux.h>
Larry Fingerf7c92d22014-03-28 21:37:39 -050023
24static void _FWDownloadEnable(struct rtw_adapter *padapter, bool enable)
25{
26 u8 tmp;
27
28 if (enable) {
29 /* 8051 enable */
Jes Sorensen050abc42014-05-16 10:05:08 +020030 tmp = rtl8723au_read8(padapter, REG_SYS_FUNC_EN + 1);
Jes Sorensenedbfd672014-05-16 10:05:09 +020031 rtl8723au_write8(padapter, REG_SYS_FUNC_EN + 1, tmp | 0x04);
Larry Fingerf7c92d22014-03-28 21:37:39 -050032
33 /* MCU firmware download enable. */
Jes Sorensen050abc42014-05-16 10:05:08 +020034 tmp = rtl8723au_read8(padapter, REG_MCUFWDL);
Jes Sorensenedbfd672014-05-16 10:05:09 +020035 rtl8723au_write8(padapter, REG_MCUFWDL, tmp | 0x01);
Larry Fingerf7c92d22014-03-28 21:37:39 -050036
37 /* 8051 reset */
Jes Sorensen050abc42014-05-16 10:05:08 +020038 tmp = rtl8723au_read8(padapter, REG_MCUFWDL + 2);
Jes Sorensenedbfd672014-05-16 10:05:09 +020039 rtl8723au_write8(padapter, REG_MCUFWDL + 2, tmp & 0xf7);
Larry Fingerf7c92d22014-03-28 21:37:39 -050040 } else {
41 /* MCU firmware download disable. */
Jes Sorensen050abc42014-05-16 10:05:08 +020042 tmp = rtl8723au_read8(padapter, REG_MCUFWDL);
Jes Sorensenedbfd672014-05-16 10:05:09 +020043 rtl8723au_write8(padapter, REG_MCUFWDL, tmp & 0xfe);
Larry Fingerf7c92d22014-03-28 21:37:39 -050044
45 /* Reserved for fw extension. */
Jes Sorensenedbfd672014-05-16 10:05:09 +020046 rtl8723au_write8(padapter, REG_MCUFWDL + 1, 0x00);
Larry Fingerf7c92d22014-03-28 21:37:39 -050047 }
48}
49
Larry Fingerf7c92d22014-03-28 21:37:39 -050050static int
51_PageWrite(struct rtw_adapter *padapter, u32 page, void *buffer, u32 size)
52{
53 u8 value8;
54 u8 u8Page = (u8) (page & 0x07);
55
Jes Sorensen7b78bdc2014-11-30 16:05:12 -050056 if (size > MAX_PAGE_SIZE)
57 return _FAIL;
58
Jes Sorensen050abc42014-05-16 10:05:08 +020059 value8 = (rtl8723au_read8(padapter, REG_MCUFWDL + 2) & 0xF8) | u8Page;
Jes Sorensenedbfd672014-05-16 10:05:09 +020060 rtl8723au_write8(padapter, REG_MCUFWDL + 2, value8);
Larry Fingerf7c92d22014-03-28 21:37:39 -050061
Jes Sorensen7b78bdc2014-11-30 16:05:12 -050062 return rtl8723au_writeN(padapter, FW_8723A_START_ADDRESS, size, buffer);
Larry Fingerf7c92d22014-03-28 21:37:39 -050063}
64
65static int _WriteFW(struct rtw_adapter *padapter, void *buffer, u32 size)
66{
67 /* Since we need dynamic decide method of dwonload fw, so we
68 call this function to get chip version. */
69 /* We can remove _ReadChipVersion from ReadpadapterInfo8192C later. */
70 int ret = _SUCCESS;
71 u32 pageNums, remainSize;
72 u32 page, offset;
73 u8 *bufferPtr = (u8 *) buffer;
74
75 pageNums = size / MAX_PAGE_SIZE;
76 /* RT_ASSERT((pageNums <= 4),
77 ("Page numbers should not greater then 4 \n")); */
78 remainSize = size % MAX_PAGE_SIZE;
79
80 for (page = 0; page < pageNums; page++) {
81 offset = page * MAX_PAGE_SIZE;
82 ret = _PageWrite(padapter, page, bufferPtr + offset,
83 MAX_PAGE_SIZE);
84
85 if (ret == _FAIL)
86 goto exit;
87 }
88 if (remainSize) {
89 offset = pageNums * MAX_PAGE_SIZE;
90 page = pageNums;
91 ret = _PageWrite(padapter, page, bufferPtr + offset,
92 remainSize);
93
94 if (ret == _FAIL)
95 goto exit;
96 }
97 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -070098 "_WriteFW Done- for Normal chip.\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -050099
100exit:
101 return ret;
102}
103
Jes Sorensena5681f92014-05-16 10:04:48 +0200104static int _FWFreeToGo(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500105{
106 u32 counter = 0;
107 u32 value32;
108
109 /* polling CheckSum report */
110 do {
Jes Sorensen050abc42014-05-16 10:05:08 +0200111 value32 = rtl8723au_read32(padapter, REG_MCUFWDL);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500112 if (value32 & FWDL_ChkSum_rpt)
113 break;
114 } while (counter++ < POLLING_READY_TIMEOUT_COUNT);
115
116 if (counter >= POLLING_READY_TIMEOUT_COUNT) {
117 RT_TRACE(_module_hal_init_c_, _drv_err_,
Joe Perches90403aa2015-03-24 16:06:44 -0700118 "%s: chksum report fail! REG_MCUFWDL:0x%08x\n",
119 __func__, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500120 return _FAIL;
121 }
122 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700123 "%s: Checksum report OK! REG_MCUFWDL:0x%08x\n", __func__,
124 value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500125
Jes Sorensen050abc42014-05-16 10:05:08 +0200126 value32 = rtl8723au_read32(padapter, REG_MCUFWDL);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500127 value32 |= MCUFWDL_RDY;
128 value32 &= ~WINTINI_RDY;
Jes Sorensenedbfd672014-05-16 10:05:09 +0200129 rtl8723au_write32(padapter, REG_MCUFWDL, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500130
131 /* polling for FW ready */
132 counter = 0;
133 do {
Jes Sorensen050abc42014-05-16 10:05:08 +0200134 value32 = rtl8723au_read32(padapter, REG_MCUFWDL);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500135 if (value32 & WINTINI_RDY) {
136 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700137 "%s: Polling FW ready success!! REG_MCUFWDL:0x%08x\n",
138 __func__, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500139 return _SUCCESS;
140 }
141 udelay(5);
142 } while (counter++ < POLLING_READY_TIMEOUT_COUNT);
143
144 RT_TRACE(_module_hal_init_c_, _drv_err_,
Joe Perches90403aa2015-03-24 16:06:44 -0700145 "%s: Polling FW ready fail!! REG_MCUFWDL:0x%08x\n",
146 __func__, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500147 return _FAIL;
148}
149
150#define IS_FW_81xxC(padapter) (((GET_HAL_DATA(padapter))->FirmwareSignature & 0xFFF0) == 0x88C0)
151
152void rtl8723a_FirmwareSelfReset(struct rtw_adapter *padapter)
153{
154 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
155 u8 u1bTmp;
156 u8 Delay = 100;
157
158 if (!(IS_FW_81xxC(padapter) &&
159 ((pHalData->FirmwareVersion < 0x21) ||
160 (pHalData->FirmwareVersion == 0x21 &&
161 pHalData->FirmwareSubVersion < 0x01)))) {
162 /* after 88C Fw v33.1 */
163 /* 0x1cf = 0x20. Inform 8051 to reset. 2009.12.25. tynli_test */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200164 rtl8723au_write8(padapter, REG_HMETFR + 3, 0x20);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500165
Jes Sorensen050abc42014-05-16 10:05:08 +0200166 u1bTmp = rtl8723au_read8(padapter, REG_SYS_FUNC_EN + 1);
Jes Sorensenbc0d10c2014-05-16 10:03:49 +0200167 while (u1bTmp & BIT(2)) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500168 Delay--;
169 if (Delay == 0)
170 break;
171 udelay(50);
Jes Sorensen050abc42014-05-16 10:05:08 +0200172 u1bTmp = rtl8723au_read8(padapter, REG_SYS_FUNC_EN + 1);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500173 }
174 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700175 "-%s: 8051 reset success (%d)\n", __func__,
176 Delay);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500177
178 if ((Delay == 0)) {
179 /* force firmware reset */
Jes Sorensen050abc42014-05-16 10:05:08 +0200180 u1bTmp = rtl8723au_read8(padapter, REG_SYS_FUNC_EN + 1);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200181 rtl8723au_write8(padapter, REG_SYS_FUNC_EN + 1,
182 u1bTmp & ~BIT(2));
Larry Fingerf7c92d22014-03-28 21:37:39 -0500183 }
184 }
185}
186
187/* */
188/* Description: */
189/* Download 8192C firmware code. */
190/* */
191/* */
Jes Sorensena5681f92014-05-16 10:04:48 +0200192int rtl8723a_FirmwareDownload(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500193{
Jes Sorensena5681f92014-05-16 10:04:48 +0200194 int rtStatus = _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500195 u8 writeFW_retry = 0;
196 unsigned long fwdl_start_time;
197 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
198 struct dvobj_priv *dvobj = adapter_to_dvobj(padapter);
199 struct device *device = dvobj_to_dev(dvobj);
200 struct rt_8723a_firmware_hdr *pFwHdr = NULL;
201 const struct firmware *fw;
202 char *fw_name;
203 u8 *firmware_buf = NULL;
204 u8 *buf;
205 int fw_size;
206 static int log_version;
207
Joe Perches90403aa2015-03-24 16:06:44 -0700208 RT_TRACE(_module_hal_init_c_, _drv_info_, "+%s\n", __func__);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500209
210 if (IS_8723A_A_CUT(pHalData->VersionID)) {
Jes Sorensen6601e312014-06-09 15:16:11 +0200211 fw_name = "rtlwifi/rtl8723aufw_A.bin";
Larry Fingerf7c92d22014-03-28 21:37:39 -0500212 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700213 "rtl8723a_FirmwareDownload: R8723FwImageArray_UMC for RTL8723A A CUT\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500214 } else if (IS_8723A_B_CUT(pHalData->VersionID)) {
215 /* WLAN Fw. */
216 if (padapter->registrypriv.wifi_spec == 1) {
217 fw_name = "rtlwifi/rtl8723aufw_B_NoBT.bin";
218 DBG_8723A(" Rtl8723_FwUMCBCutImageArrayWithoutBT for "
219 "RTL8723A B CUT\n");
220 } else {
Jes Sorensen900a3262014-05-25 22:43:41 +0200221 if (rtl8723a_BT_coexist(padapter)) {
222 fw_name = "rtlwifi/rtl8723aufw_B.bin";
223 DBG_8723A(" Rtl8723_FwUMCBCutImageArrayWithBT "
224 "for RTL8723A B CUT\n");
225 } else {
226 fw_name = "rtlwifi/rtl8723aufw_B_NoBT.bin";
227 DBG_8723A(" Rtl8723_FwUMCBCutImageArrayWithout "
228 "BT for RTL8723A B CUT\n");
229 }
Larry Fingerf7c92d22014-03-28 21:37:39 -0500230 }
231 } else {
232 /* <Roger_TODO> We should download proper RAM Code here
233 to match the ROM code. */
234 RT_TRACE(_module_hal_init_c_, _drv_err_,
Joe Perches90403aa2015-03-24 16:06:44 -0700235 "%s: unknown version!\n", __func__);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500236 rtStatus = _FAIL;
237 goto Exit;
238 }
239
240 pr_info("rtl8723au: Loading firmware %s\n", fw_name);
241 if (request_firmware(&fw, fw_name, device)) {
242 pr_err("rtl8723au: request_firmware load failed\n");
243 rtStatus = _FAIL;
244 goto Exit;
245 }
246 if (!fw) {
247 pr_err("rtl8723au: Firmware %s not available\n", fw_name);
248 rtStatus = _FAIL;
249 goto Exit;
250 }
Benoit Taine4a6eea42014-05-22 15:08:33 +0200251 firmware_buf = kmemdup(fw->data, fw->size, GFP_KERNEL);
Mateusz Kulikowski7d761732015-09-27 21:48:19 +0200252 fw_size = fw->size;
253 release_firmware(fw);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500254 if (!firmware_buf) {
255 rtStatus = _FAIL;
256 goto Exit;
257 }
Larry Fingerf7c92d22014-03-28 21:37:39 -0500258 buf = firmware_buf;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500259
260 /* To Check Fw header. Added by tynli. 2009.12.04. */
261 pFwHdr = (struct rt_8723a_firmware_hdr *)firmware_buf;
262
263 pHalData->FirmwareVersion = le16_to_cpu(pFwHdr->Version);
264 pHalData->FirmwareSubVersion = pFwHdr->Subversion;
265 pHalData->FirmwareSignature = le16_to_cpu(pFwHdr->Signature);
266
267 DBG_8723A("%s: fw_ver =%d fw_subver =%d sig = 0x%x\n",
268 __func__, pHalData->FirmwareVersion,
269 pHalData->FirmwareSubVersion, pHalData->FirmwareSignature);
270
271 if (!log_version++)
272 pr_info("%sFirmware Version %d, SubVersion %d, Signature "
273 "0x%x\n", DRIVER_PREFIX, pHalData->FirmwareVersion,
274 pHalData->FirmwareSubVersion,
275 pHalData->FirmwareSignature);
276
277 if (IS_FW_HEADER_EXIST(pFwHdr)) {
278 /* Shift 32 bytes for FW header */
279 buf = buf + 32;
280 fw_size = fw_size - 32;
281 }
282
283 /* Suggested by Filen. If 8051 is running in RAM code, driver should
284 inform Fw to reset by itself, */
285 /* or it will cause download Fw fail. 2010.02.01. by tynli. */
Jes Sorensen050abc42014-05-16 10:05:08 +0200286 if (rtl8723au_read8(padapter, REG_MCUFWDL) & RAM_DL_SEL) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500287 /* 8051 RAM code */
288 rtl8723a_FirmwareSelfReset(padapter);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200289 rtl8723au_write8(padapter, REG_MCUFWDL, 0x00);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500290 }
291
292 _FWDownloadEnable(padapter, true);
293 fwdl_start_time = jiffies;
294 while (1) {
295 /* reset the FWDL chksum */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200296 rtl8723au_write8(padapter, REG_MCUFWDL,
297 rtl8723au_read8(padapter, REG_MCUFWDL) |
298 FWDL_ChkSum_rpt);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500299
300 rtStatus = _WriteFW(padapter, buf, fw_size);
301
302 if (rtStatus == _SUCCESS ||
Larry Fingerc17416e2014-03-28 21:37:42 -0500303 (jiffies_to_msecs(jiffies - fwdl_start_time) > 500 &&
Larry Fingerf7c92d22014-03-28 21:37:39 -0500304 writeFW_retry++ >= 3))
305 break;
306
307 DBG_8723A("%s writeFW_retry:%u, time after fwdl_start_time:"
308 "%ums\n", __func__, writeFW_retry,
309 jiffies_to_msecs(jiffies - fwdl_start_time));
310 }
311 _FWDownloadEnable(padapter, false);
312 if (_SUCCESS != rtStatus) {
313 DBG_8723A("DL Firmware failed!\n");
314 goto Exit;
315 }
316
317 rtStatus = _FWFreeToGo(padapter);
318 if (_SUCCESS != rtStatus) {
319 RT_TRACE(_module_hal_init_c_, _drv_err_,
Joe Perches90403aa2015-03-24 16:06:44 -0700320 "DL Firmware failed!\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500321 goto Exit;
322 }
323 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700324 "Firmware is ready to run!\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500325
326Exit:
327 kfree(firmware_buf);
328 return rtStatus;
329}
330
331void rtl8723a_InitializeFirmwareVars(struct rtw_adapter *padapter)
332{
333 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
334
335 /* Init Fw LPS related. */
336 padapter->pwrctrlpriv.bFwCurrentInPSMode = false;
337
338 /* Init H2C counter. by tynli. 2009.12.09. */
339 pHalData->LastHMEBoxNum = 0;
340}
341
Larry Fingerf7c92d22014-03-28 21:37:39 -0500342/* */
343/* Efuse related code */
344/* */
345static u8
346hal_EfuseSwitchToBank(struct rtw_adapter *padapter, u8 bank)
347{
348 u8 bRet = false;
349 u32 value32 = 0;
350
351 DBG_8723A("%s: Efuse switch bank to %d\n", __func__, bank);
Jes Sorensen050abc42014-05-16 10:05:08 +0200352 value32 = rtl8723au_read32(padapter, EFUSE_TEST);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500353 bRet = true;
354 switch (bank) {
355 case 0:
356 value32 = (value32 & ~EFUSE_SEL_MASK) |
357 EFUSE_SEL(EFUSE_WIFI_SEL_0);
358 break;
359 case 1:
360 value32 = (value32 & ~EFUSE_SEL_MASK) |
361 EFUSE_SEL(EFUSE_BT_SEL_0);
362 break;
363 case 2:
364 value32 = (value32 & ~EFUSE_SEL_MASK) |
365 EFUSE_SEL(EFUSE_BT_SEL_1);
366 break;
367 case 3:
368 value32 = (value32 & ~EFUSE_SEL_MASK) |
369 EFUSE_SEL(EFUSE_BT_SEL_2);
370 break;
371 default:
372 value32 = (value32 & ~EFUSE_SEL_MASK) |
373 EFUSE_SEL(EFUSE_WIFI_SEL_0);
374 bRet = false;
375 break;
376 }
Jes Sorensenedbfd672014-05-16 10:05:09 +0200377 rtl8723au_write32(padapter, EFUSE_TEST, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500378
379 return bRet;
380}
381
382static void
Larry Fingerf7c92d22014-03-28 21:37:39 -0500383hal_ReadEFuse_WiFi(struct rtw_adapter *padapter,
384 u16 _offset, u16 _size_byte, u8 *pbuf)
385{
386 u8 *efuseTbl = NULL;
387 u16 eFuse_Addr = 0;
388 u8 offset, wden;
389 u8 efuseHeader, efuseExtHdr, efuseData;
390 u16 i, total, used;
Jes Sorensena2342292014-04-09 23:20:46 +0200391 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500392
393 /* Do NOT excess total size of EFuse table.
394 Added by Roger, 2008.11.10. */
395 if ((_offset + _size_byte) > EFUSE_MAP_LEN_8723A) {
396 DBG_8723A("%s: Invalid offset(%#x) with read bytes(%#x)!!\n",
397 __func__, _offset, _size_byte);
398 return;
399 }
400
Sachin Kamat2e697592014-05-28 17:36:29 +0530401 efuseTbl = kmalloc(EFUSE_MAP_LEN_8723A, GFP_KERNEL);
Bhumika Goyal28f5caa2016-02-24 15:53:57 +0530402 if (!efuseTbl) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500403 DBG_8723A("%s: alloc efuseTbl fail!\n", __func__);
404 return;
405 }
406 /* 0xff will be efuse default value instead of 0x00. */
407 memset(efuseTbl, 0xFF, EFUSE_MAP_LEN_8723A);
408
409 /* switch bank back to bank 0 for later BT and wifi use. */
410 hal_EfuseSwitchToBank(padapter, 0);
411
412 while (AVAILABLE_EFUSE_ADDR(eFuse_Addr)) {
413 ReadEFuseByte23a(padapter, eFuse_Addr++, &efuseHeader);
414 if (efuseHeader == 0xFF) {
415 DBG_8723A("%s: data end at address =%#x\n", __func__,
416 eFuse_Addr);
417 break;
418 }
419
420 /* Check PG header for section num. */
421 if (EXT_HEADER(efuseHeader)) { /* extended header */
422 offset = GET_HDR_OFFSET_2_0(efuseHeader);
423
424 ReadEFuseByte23a(padapter, eFuse_Addr++, &efuseExtHdr);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +0100425 if (ALL_WORDS_DISABLED(efuseExtHdr))
Larry Fingerf7c92d22014-03-28 21:37:39 -0500426 continue;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500427
428 offset |= ((efuseExtHdr & 0xF0) >> 1);
Haneen Mohammedacc4b972015-03-13 20:45:25 +0300429 wden = efuseExtHdr & 0x0F;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500430 } else {
Haneen Mohammedacc4b972015-03-13 20:45:25 +0300431 offset = (efuseHeader >> 4) & 0x0f;
432 wden = efuseHeader & 0x0f;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500433 }
434
435 if (offset < EFUSE_MAX_SECTION_8723A) {
436 u16 addr;
437 /* Get word enable value from PG header */
438
439 addr = offset * PGPKT_DATA_SIZE;
440 for (i = 0; i < EFUSE_MAX_WORD_UNIT; i++) {
441 /* Check word enable condition in the section */
442 if (!(wden & (0x01 << i))) {
443 ReadEFuseByte23a(padapter, eFuse_Addr++,
444 &efuseData);
445 efuseTbl[addr] = efuseData;
446
447 ReadEFuseByte23a(padapter, eFuse_Addr++,
448 &efuseData);
449 efuseTbl[addr + 1] = efuseData;
450 }
451 addr += 2;
452 }
453 } else {
454 DBG_8723A(KERN_ERR "%s: offset(%d) is illegal!!\n",
455 __func__, offset);
456 eFuse_Addr += Efuse_CalculateWordCnts23a(wden) * 2;
457 }
458 }
459
460 /* Copy from Efuse map to output pointer memory!!! */
461 for (i = 0; i < _size_byte; i++)
462 pbuf[i] = efuseTbl[_offset + i];
463
464 /* Calculate Efuse utilization */
465 EFUSE_GetEfuseDefinition23a(padapter, EFUSE_WIFI,
466 TYPE_AVAILABLE_EFUSE_BYTES_TOTAL, &total);
467 used = eFuse_Addr - 1;
Jes Sorensena2342292014-04-09 23:20:46 +0200468 pHalData->EfuseUsedBytes = used;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500469
470 kfree(efuseTbl);
471}
472
473static void
474hal_ReadEFuse_BT(struct rtw_adapter *padapter,
475 u16 _offset, u16 _size_byte, u8 *pbuf)
476{
477 u8 *efuseTbl;
478 u8 bank;
479 u16 eFuse_Addr;
480 u8 efuseHeader, efuseExtHdr, efuseData;
481 u8 offset, wden;
482 u16 i, total, used;
Jes Sorensena2342292014-04-09 23:20:46 +0200483 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500484
485 /* Do NOT excess total size of EFuse table.
486 Added by Roger, 2008.11.10. */
487 if ((_offset + _size_byte) > EFUSE_BT_MAP_LEN) {
488 DBG_8723A("%s: Invalid offset(%#x) with read bytes(%#x)!!\n",
489 __func__, _offset, _size_byte);
490 return;
491 }
492
493 efuseTbl = kmalloc(EFUSE_BT_MAP_LEN, GFP_KERNEL);
Bhumika Goyal28f5caa2016-02-24 15:53:57 +0530494 if (!efuseTbl) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500495 DBG_8723A("%s: efuseTbl malloc fail!\n", __func__);
496 return;
497 }
498 /* 0xff will be efuse default value instead of 0x00. */
499 memset(efuseTbl, 0xFF, EFUSE_BT_MAP_LEN);
500
501 EFUSE_GetEfuseDefinition23a(padapter, EFUSE_BT,
502 TYPE_AVAILABLE_EFUSE_BYTES_BANK, &total);
503
504 for (bank = 1; bank < EFUSE_MAX_BANK; bank++) {
505 if (hal_EfuseSwitchToBank(padapter, bank) == false) {
506 DBG_8723A("%s: hal_EfuseSwitchToBank Fail!!\n",
507 __func__);
508 goto exit;
509 }
510
511 eFuse_Addr = 0;
512
513 while (AVAILABLE_EFUSE_ADDR(eFuse_Addr)) {
514 ReadEFuseByte23a(padapter, eFuse_Addr++, &efuseHeader);
515 if (efuseHeader == 0xFF)
516 break;
517
518 /* Check PG header for section num. */
519 if (EXT_HEADER(efuseHeader)) { /* extended header */
520 offset = GET_HDR_OFFSET_2_0(efuseHeader);
521
522 ReadEFuseByte23a(padapter, eFuse_Addr++,
523 &efuseExtHdr);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +0100524 if (ALL_WORDS_DISABLED(efuseExtHdr))
Larry Fingerf7c92d22014-03-28 21:37:39 -0500525 continue;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500526
527 offset |= ((efuseExtHdr & 0xF0) >> 1);
Haneen Mohammedacc4b972015-03-13 20:45:25 +0300528 wden = efuseExtHdr & 0x0F;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500529 } else {
Haneen Mohammedacc4b972015-03-13 20:45:25 +0300530 offset = (efuseHeader >> 4) & 0x0f;
531 wden = efuseHeader & 0x0f;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500532 }
533
534 if (offset < EFUSE_BT_MAX_SECTION) {
535 u16 addr;
536
537 /* Get word enable value from PG header */
538
539 addr = offset * PGPKT_DATA_SIZE;
540 for (i = 0; i < EFUSE_MAX_WORD_UNIT; i++) {
541 /* Check word enable condition in
542 the section */
543 if (!(wden & (0x01 << i))) {
544 ReadEFuseByte23a(padapter,
545 eFuse_Addr++,
546 &efuseData);
547 efuseTbl[addr] = efuseData;
548
549 ReadEFuseByte23a(padapter,
550 eFuse_Addr++,
551 &efuseData);
552 efuseTbl[addr + 1] = efuseData;
553 }
554 addr += 2;
555 }
556 } else {
557 DBG_8723A(KERN_ERR
558 "%s: offset(%d) is illegal!!\n",
559 __func__, offset);
560 eFuse_Addr += Efuse_CalculateWordCnts23a(wden) * 2;
561 }
562 }
563
564 if ((eFuse_Addr - 1) < total) {
565 DBG_8723A("%s: bank(%d) data end at %#x\n",
566 __func__, bank, eFuse_Addr - 1);
567 break;
568 }
569 }
570
571 /* switch bank back to bank 0 for later BT and wifi use. */
572 hal_EfuseSwitchToBank(padapter, 0);
573
574 /* Copy from Efuse map to output pointer memory!!! */
575 for (i = 0; i < _size_byte; i++)
576 pbuf[i] = efuseTbl[_offset + i];
577
578 /* */
579 /* Calculate Efuse utilization. */
580 /* */
581 EFUSE_GetEfuseDefinition23a(padapter, EFUSE_BT,
582 TYPE_AVAILABLE_EFUSE_BYTES_TOTAL, &total);
583 used = (EFUSE_BT_REAL_BANK_CONTENT_LEN * (bank - 1)) + eFuse_Addr - 1;
Jes Sorensena2342292014-04-09 23:20:46 +0200584 pHalData->BTEfuseUsedBytes = used;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500585
586exit:
587 kfree(efuseTbl);
588}
589
Jes Sorensenb2e68042014-05-16 10:04:14 +0200590void
591rtl8723a_readefuse(struct rtw_adapter *padapter,
592 u8 efuseType, u16 _offset, u16 _size_byte, u8 *pbuf)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500593{
594 if (efuseType == EFUSE_WIFI)
595 hal_ReadEFuse_WiFi(padapter, _offset, _size_byte, pbuf);
596 else
597 hal_ReadEFuse_BT(padapter, _offset, _size_byte, pbuf);
598}
599
Jes Sorensenb2e68042014-05-16 10:04:14 +0200600u16 rtl8723a_EfuseGetCurrentSize_WiFi(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500601{
602 u16 efuse_addr = 0;
603 u8 hoffset = 0, hworden = 0;
604 u8 efuse_data, word_cnts = 0;
Jes Sorensena2342292014-04-09 23:20:46 +0200605 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500606
Jes Sorensena2342292014-04-09 23:20:46 +0200607 efuse_addr = pHalData->EfuseUsedBytes;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500608
609 DBG_8723A("%s: start_efuse_addr = 0x%X\n", __func__, efuse_addr);
610
611 /* switch bank back to bank 0 for later BT and wifi use. */
612 hal_EfuseSwitchToBank(padapter, 0);
613
614 while (AVAILABLE_EFUSE_ADDR(efuse_addr)) {
615 if (efuse_OneByteRead23a(padapter, efuse_addr, &efuse_data) ==
Jes Sorensen44f3f162014-05-16 10:04:29 +0200616 _FAIL) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500617 DBG_8723A(KERN_ERR "%s: efuse_OneByteRead23a Fail! "
618 "addr = 0x%X !!\n", __func__, efuse_addr);
619 break;
620 }
621
622 if (efuse_data == 0xFF)
623 break;
624
625 if (EXT_HEADER(efuse_data)) {
626 hoffset = GET_HDR_OFFSET_2_0(efuse_data);
627 efuse_addr++;
628 efuse_OneByteRead23a(padapter, efuse_addr, &efuse_data);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +0100629 if (ALL_WORDS_DISABLED(efuse_data))
Larry Fingerf7c92d22014-03-28 21:37:39 -0500630 continue;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500631
632 hoffset |= ((efuse_data & 0xF0) >> 1);
633 hworden = efuse_data & 0x0F;
634 } else {
635 hoffset = (efuse_data >> 4) & 0x0F;
636 hworden = efuse_data & 0x0F;
637 }
638
639 word_cnts = Efuse_CalculateWordCnts23a(hworden);
640 efuse_addr += (word_cnts * 2) + 1;
641 }
642
Jes Sorensena2342292014-04-09 23:20:46 +0200643 pHalData->EfuseUsedBytes = efuse_addr;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500644
645 DBG_8723A("%s: CurrentSize =%d\n", __func__, efuse_addr);
646
647 return efuse_addr;
648}
649
Jes Sorensenb2e68042014-05-16 10:04:14 +0200650u16 rtl8723a_EfuseGetCurrentSize_BT(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500651{
652 u16 btusedbytes;
653 u16 efuse_addr;
654 u8 bank, startBank;
655 u8 hoffset = 0, hworden = 0;
656 u8 efuse_data, word_cnts = 0;
657 u16 retU2 = 0;
Jes Sorensena2342292014-04-09 23:20:46 +0200658 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500659
Jes Sorensena2342292014-04-09 23:20:46 +0200660 btusedbytes = pHalData->BTEfuseUsedBytes;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500661
662 efuse_addr = (u16) ((btusedbytes % EFUSE_BT_REAL_BANK_CONTENT_LEN));
663 startBank = (u8) (1 + (btusedbytes / EFUSE_BT_REAL_BANK_CONTENT_LEN));
664
665 DBG_8723A("%s: start from bank =%d addr = 0x%X\n", __func__, startBank,
666 efuse_addr);
667
668 EFUSE_GetEfuseDefinition23a(padapter, EFUSE_BT,
669 TYPE_AVAILABLE_EFUSE_BYTES_BANK, &retU2);
670
671 for (bank = startBank; bank < EFUSE_MAX_BANK; bank++) {
672 if (hal_EfuseSwitchToBank(padapter, bank) == false) {
673 DBG_8723A(KERN_ERR "%s: switch bank(%d) Fail!!\n",
674 __func__, bank);
675 bank = EFUSE_MAX_BANK;
676 break;
677 }
678
679 /* only when bank is switched we have to reset
680 the efuse_addr. */
681 if (bank != startBank)
682 efuse_addr = 0;
683
684 while (AVAILABLE_EFUSE_ADDR(efuse_addr)) {
685 if (efuse_OneByteRead23a(padapter, efuse_addr,
Jes Sorensen44f3f162014-05-16 10:04:29 +0200686 &efuse_data) == _FAIL) {
Larry Fingerf7c92d22014-03-28 21:37:39 -0500687 DBG_8723A(KERN_ERR "%s: efuse_OneByteRead23a Fail!"
688 " addr = 0x%X !!\n",
689 __func__, efuse_addr);
690 bank = EFUSE_MAX_BANK;
691 break;
692 }
693
694 if (efuse_data == 0xFF)
695 break;
696
697 if (EXT_HEADER(efuse_data)) {
698 hoffset = GET_HDR_OFFSET_2_0(efuse_data);
699 efuse_addr++;
700 efuse_OneByteRead23a(padapter, efuse_addr,
701 &efuse_data);
702 if (ALL_WORDS_DISABLED(efuse_data)) {
703 efuse_addr++;
704 continue;
705 }
706
707 hoffset |= ((efuse_data & 0xF0) >> 1);
708 hworden = efuse_data & 0x0F;
709 } else {
710 hoffset = (efuse_data >> 4) & 0x0F;
711 hworden = efuse_data & 0x0F;
712 }
713 word_cnts = Efuse_CalculateWordCnts23a(hworden);
714 /* read next header */
715 efuse_addr += (word_cnts * 2) + 1;
716 }
717
718 /* Check if we need to check next bank efuse */
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +0100719 if (efuse_addr < retU2)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500720 break; /* don't need to check next bank. */
Larry Fingerf7c92d22014-03-28 21:37:39 -0500721 }
722
723 retU2 = ((bank - 1) * EFUSE_BT_REAL_BANK_CONTENT_LEN) + efuse_addr;
Jes Sorensena2342292014-04-09 23:20:46 +0200724 pHalData->BTEfuseUsedBytes = retU2;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500725
726 DBG_8723A("%s: CurrentSize =%d\n", __func__, retU2);
727 return retU2;
728}
729
Jes Sorensen44e621c2014-05-16 10:04:04 +0200730void rtl8723a_read_chip_version(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500731{
732 u32 value32;
733 struct hal_version ChipVersion;
734 struct hal_data_8723a *pHalData;
735
736 pHalData = GET_HAL_DATA(padapter);
737
Jes Sorensen050abc42014-05-16 10:05:08 +0200738 value32 = rtl8723au_read32(padapter, REG_SYS_CFG);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500739 ChipVersion.ICType = CHIP_8723A;
740 ChipVersion.ChipType = ((value32 & RTL_ID) ? TEST_CHIP : NORMAL_CHIP);
Jes Sorensen277c7222015-02-27 15:45:32 -0500741 pHalData->rf_type = RF_1T1R;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500742 ChipVersion.VendorType =
743 ((value32 & VENDOR_ID) ? CHIP_VENDOR_UMC : CHIP_VENDOR_TSMC);
744 ChipVersion.CUTVersion = (value32 & CHIP_VER_RTL_MASK) >> CHIP_VER_RTL_SHIFT; /* IC version (CUT) */
745
746 /* For regulator mode. by tynli. 2011.01.14 */
747 pHalData->RegulatorMode = ((value32 & SPS_SEL) ?
748 RT_LDO_REGULATOR : RT_SWITCHING_REGULATOR);
749
Jes Sorensen050abc42014-05-16 10:05:08 +0200750 value32 = rtl8723au_read32(padapter, REG_GPIO_OUTSTS);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500751 /* ROM code version. */
Aya Mahfouze1bc88f2015-03-04 07:34:07 +0200752 ChipVersion.ROMVer = (value32 & RF_RL_ID) >> 20;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500753
754 /* For multi-function consideration. Added by Roger, 2010.10.06. */
755 pHalData->MultiFunc = RT_MULTI_FUNC_NONE;
Jes Sorensen050abc42014-05-16 10:05:08 +0200756 value32 = rtl8723au_read32(padapter, REG_MULTI_FUNC_CTRL);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500757 pHalData->MultiFunc |=
758 ((value32 & WL_FUNC_EN) ? RT_MULTI_FUNC_WIFI : 0);
759 pHalData->MultiFunc |= ((value32 & BT_FUNC_EN) ? RT_MULTI_FUNC_BT : 0);
760 pHalData->MultiFunc |=
761 ((value32 & GPS_FUNC_EN) ? RT_MULTI_FUNC_GPS : 0);
762 pHalData->PolarityCtl =
763 ((value32 & WL_HWPDN_SL) ? RT_POLARITY_HIGH_ACT :
764 RT_POLARITY_LOW_ACT);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500765 pHalData->VersionID = ChipVersion;
766
Larry Fingerf7c92d22014-03-28 21:37:39 -0500767 MSG_8723A("RF_Type is %x!!\n", pHalData->rf_type);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500768}
769
770/* */
771/* */
772/* 20100209 Joseph: */
773/* This function is used only for 92C to set REG_BCN_CTRL(0x550) register. */
774/* We just reserve the value of the register in variable
775 pHalData->RegBcnCtrlVal and then operate */
776/* the value of the register via atomic operation. */
777/* This prevents from race condition when setting this register. */
778/* The value of pHalData->RegBcnCtrlVal is initialized in
779 HwConfigureRTL8192CE() function. */
780/* */
781void SetBcnCtrlReg23a(struct rtw_adapter *padapter, u8 SetBits, u8 ClearBits)
782{
Jes Sorensene4d480b2014-07-17 22:59:52 +0200783 u8 val8;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500784
Jes Sorensene4d480b2014-07-17 22:59:52 +0200785 val8 = rtl8723au_read8(padapter, REG_BCN_CTRL);
786 val8 |= SetBits;
787 val8 &= ~ClearBits;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500788
Jes Sorensene4d480b2014-07-17 22:59:52 +0200789 rtl8723au_write8(padapter, REG_BCN_CTRL, val8);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500790}
791
792void rtl8723a_InitBeaconParameters(struct rtw_adapter *padapter)
793{
Jes Sorensenedbfd672014-05-16 10:05:09 +0200794 rtl8723au_write16(padapter, REG_BCN_CTRL, 0x1010);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500795
796 /* TODO: Remove these magic number */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200797 rtl8723au_write16(padapter, REG_TBTT_PROHIBIT, 0x6404); /* ms */
Larry Fingerf7c92d22014-03-28 21:37:39 -0500798 /* Firmware will control REG_DRVERLYINT when power saving is enable, */
799 /* so don't set this register on STA mode. */
800 if (check_fwstate(&padapter->mlmepriv, WIFI_STATION_STATE) == false)
Jes Sorensenedbfd672014-05-16 10:05:09 +0200801 rtl8723au_write8(padapter, REG_DRVERLYINT,
802 DRIVER_EARLY_INT_TIME);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500803 /* 2ms */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200804 rtl8723au_write8(padapter, REG_BCNDMATIM, BCN_DMA_ATIME_INT_TIME);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500805
806 /* Suggested by designer timchen. Change beacon AIFS to the
807 largest number beacause test chip does not contension before
808 sending beacon. by tynli. 2009.11.03 */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200809 rtl8723au_write16(padapter, REG_BCNTCFG, 0x660F);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500810}
811
812static void ResumeTxBeacon(struct rtw_adapter *padapter)
813{
814 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
815
816 /* 2010.03.01. Marked by tynli. No need to call workitem beacause
817 we record the value */
818 /* which should be read from register to a global variable. */
819
Joe Perches90403aa2015-03-24 16:06:44 -0700820 RT_TRACE(_module_hci_hal_init_c_, _drv_info_, "+ResumeTxBeacon\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500821
822 pHalData->RegFwHwTxQCtrl |= BIT(6);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200823 rtl8723au_write8(padapter, REG_FWHW_TXQ_CTRL + 2,
824 pHalData->RegFwHwTxQCtrl);
825 rtl8723au_write8(padapter, REG_TBTT_PROHIBIT + 1, 0xff);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500826 pHalData->RegReg542 |= BIT(0);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200827 rtl8723au_write8(padapter, REG_TBTT_PROHIBIT + 2, pHalData->RegReg542);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500828}
829
830static void StopTxBeacon(struct rtw_adapter *padapter)
831{
832 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
833
834 /* 2010.03.01. Marked by tynli. No need to call workitem beacause
835 we record the value */
836 /* which should be read from register to a global variable. */
837
Joe Perches90403aa2015-03-24 16:06:44 -0700838 RT_TRACE(_module_hci_hal_init_c_, _drv_info_, "+StopTxBeacon\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500839
840 pHalData->RegFwHwTxQCtrl &= ~BIT(6);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200841 rtl8723au_write8(padapter, REG_FWHW_TXQ_CTRL + 2,
842 pHalData->RegFwHwTxQCtrl);
843 rtl8723au_write8(padapter, REG_TBTT_PROHIBIT + 1, 0x64);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500844 pHalData->RegReg542 &= ~BIT(0);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200845 rtl8723au_write8(padapter, REG_TBTT_PROHIBIT + 2, pHalData->RegReg542);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500846}
847
848static void _BeaconFunctionEnable(struct rtw_adapter *padapter, u8 Enable,
849 u8 Linked)
850{
851 SetBcnCtrlReg23a(padapter, DIS_TSF_UDT | EN_BCN_FUNCTION | DIS_BCNQ_SUB,
852 0);
Jes Sorensenedbfd672014-05-16 10:05:09 +0200853 rtl8723au_write8(padapter, REG_RD_CTRL + 1, 0x6F);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500854}
855
Jes Sorensen88715bc2014-05-16 10:04:10 +0200856void rtl8723a_SetBeaconRelatedRegisters(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500857{
858 u32 value32;
859 struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
860 struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
861
862 /* reset TSF, enable update TSF, correcting TSF On Beacon */
863
864 /* REG_BCN_INTERVAL */
865 /* REG_BCNDMATIM */
866 /* REG_ATIMWND */
867 /* REG_TBTT_PROHIBIT */
868 /* REG_DRVERLYINT */
869 /* REG_BCN_MAX_ERR */
870 /* REG_BCNTCFG (0x510) */
871 /* REG_DUAL_TSF_RST */
872 /* REG_BCN_CTRL (0x550) */
873
874 /* */
875 /* ATIM window */
876 /* */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200877 rtl8723au_write16(padapter, REG_ATIMWND, 2);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500878
879 /* */
880 /* Beacon interval (in unit of TU). */
881 /* */
Jes Sorensenedbfd672014-05-16 10:05:09 +0200882 rtl8723au_write16(padapter, REG_BCN_INTERVAL, pmlmeinfo->bcn_interval);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500883
884 rtl8723a_InitBeaconParameters(padapter);
885
Jes Sorensenedbfd672014-05-16 10:05:09 +0200886 rtl8723au_write8(padapter, REG_SLOT, 0x09);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500887
888 /* */
889 /* Reset TSF Timer to zero, added by Roger. 2008.06.24 */
890 /* */
Jes Sorensen050abc42014-05-16 10:05:08 +0200891 value32 = rtl8723au_read32(padapter, REG_TCR);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500892 value32 &= ~TSFRST;
Jes Sorensenedbfd672014-05-16 10:05:09 +0200893 rtl8723au_write32(padapter, REG_TCR, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500894
895 value32 |= TSFRST;
Jes Sorensenedbfd672014-05-16 10:05:09 +0200896 rtl8723au_write32(padapter, REG_TCR, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500897
898 /* NOTE: Fix test chip's bug (about contention windows's randomness) */
899 if (check_fwstate(&padapter->mlmepriv, WIFI_ADHOC_STATE |
900 WIFI_ADHOC_MASTER_STATE | WIFI_AP_STATE) == true) {
Jes Sorensenedbfd672014-05-16 10:05:09 +0200901 rtl8723au_write8(padapter, REG_RXTSF_OFFSET_CCK, 0x50);
902 rtl8723au_write8(padapter, REG_RXTSF_OFFSET_OFDM, 0x50);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500903 }
904
905 _BeaconFunctionEnable(padapter, true, true);
906
907 ResumeTxBeacon(padapter);
908 SetBcnCtrlReg23a(padapter, DIS_BCNQ_SUB, 0);
909}
910
Jes Sorensendbe98492014-05-16 10:04:13 +0200911void rtl8723a_SetHalODMVar(struct rtw_adapter *Adapter,
912 enum hal_odm_variable eVariable,
913 void *pValue1, bool bSet)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500914{
915 struct hal_data_8723a *pHalData = GET_HAL_DATA(Adapter);
916 struct dm_odm_t *podmpriv = &pHalData->odmpriv;
917 switch (eVariable) {
918 case HAL_ODM_STA_INFO:
919 {
920 struct sta_info *psta = (struct sta_info *)pValue1;
921
922 if (bSet) {
923 DBG_8723A("Set STA_(%d) info\n", psta->mac_id);
924 ODM_CmnInfoPtrArrayHook23a(podmpriv,
925 ODM_CMNINFO_STA_STATUS,
926 psta->mac_id, psta);
927 } else {
928 DBG_8723A("Clean STA_(%d) info\n", psta->mac_id);
929 ODM_CmnInfoPtrArrayHook23a(podmpriv,
930 ODM_CMNINFO_STA_STATUS,
931 psta->mac_id, NULL);
932 }
933 }
934 break;
935 case HAL_ODM_P2P_STATE:
936 ODM_CmnInfoUpdate23a(podmpriv, ODM_CMNINFO_WIFI_DIRECT, bSet);
937 break;
938 case HAL_ODM_WIFI_DISPLAY_STATE:
939 ODM_CmnInfoUpdate23a(podmpriv, ODM_CMNINFO_WIFI_DISPLAY, bSet);
940 break;
941 default:
942 break;
943 }
944}
945
Jes Sorensen369074f2014-05-16 10:04:02 +0200946void rtl8723a_notch_filter(struct rtw_adapter *adapter, bool enable)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500947{
948 if (enable) {
949 DBG_8723A("Enable notch filter\n");
Jes Sorensenedbfd672014-05-16 10:05:09 +0200950 rtl8723au_write8(adapter, rOFDM0_RxDSP + 1,
951 rtl8723au_read8(adapter, rOFDM0_RxDSP + 1) |
952 BIT(1));
Larry Fingerf7c92d22014-03-28 21:37:39 -0500953 } else {
954 DBG_8723A("Disable notch filter\n");
Jes Sorensenedbfd672014-05-16 10:05:09 +0200955 rtl8723au_write8(adapter, rOFDM0_RxDSP + 1,
956 rtl8723au_read8(adapter, rOFDM0_RxDSP + 1) &
957 ~BIT(1));
Larry Fingerf7c92d22014-03-28 21:37:39 -0500958 }
959}
960
Jes Sorensena5681f92014-05-16 10:04:48 +0200961bool c2h_id_filter_ccx_8723a(u8 id)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500962{
Jes Sorensena5681f92014-05-16 10:04:48 +0200963 bool ret = false;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500964 if (id == C2H_CCX_TX_RPT)
965 ret = true;
966
967 return ret;
968}
969
Jes Sorensena5681f92014-05-16 10:04:48 +0200970int c2h_handler_8723a(struct rtw_adapter *padapter, struct c2h_evt_hdr *c2h_evt)
Larry Fingerf7c92d22014-03-28 21:37:39 -0500971{
Jes Sorensena5681f92014-05-16 10:04:48 +0200972 int ret = _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -0500973 u8 i = 0;
974
975 if (c2h_evt == NULL) {
976 DBG_8723A("%s c2h_evt is NULL\n", __func__);
977 ret = _FAIL;
978 goto exit;
979 }
980
981 switch (c2h_evt->id) {
982 case C2H_DBG:
983 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700984 "C2HCommandHandler: %s\n", c2h_evt->payload);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500985 break;
986
987 case C2H_CCX_TX_RPT:
988 handle_txrpt_ccx_8723a(padapter, c2h_evt->payload);
989 break;
990 case C2H_EXT_RA_RPT:
991 break;
992 case C2H_HW_INFO_EXCH:
993 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700994 "[BT], C2H_HW_INFO_EXCH\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -0500995 for (i = 0; i < c2h_evt->plen; i++) {
996 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -0700997 "[BT], tmpBuf[%d]= 0x%x\n", i,
998 c2h_evt->payload[i]);
Larry Fingerf7c92d22014-03-28 21:37:39 -0500999 }
1000 break;
1001
1002 case C2H_C2H_H2C_TEST:
1003 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001004 "[BT], C2H_H2C_TEST\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -05001005 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001006 "[BT], tmpBuf[0]/[1]/[2]/[3]/[4]= 0x%x/ 0x%x/ 0x%x/ 0x%x/ 0x%x\n",
1007 c2h_evt->payload[0],
1008 c2h_evt->payload[1], c2h_evt->payload[2],
1009 c2h_evt->payload[3], c2h_evt->payload[4]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001010 break;
1011
Larry Fingerf7c92d22014-03-28 21:37:39 -05001012 case C2H_BT_INFO:
1013 DBG_8723A("%s , Got C2H_BT_INFO \n", __func__);
Jes Sorensen1d599de2014-05-25 22:43:46 +02001014 rtl8723a_fw_c2h_BT_info(padapter,
1015 c2h_evt->payload, c2h_evt->plen);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001016 break;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001017
1018 default:
1019 ret = _FAIL;
1020 break;
1021 }
1022
1023exit:
1024 return ret;
1025}
1026
Jes Sorensen116bf142014-11-30 16:04:56 -05001027void handle_txrpt_ccx_8723a(struct rtw_adapter *adapter, void *buf)
1028{
1029 struct txrpt_ccx_8723a *txrpt_ccx = buf;
1030 struct submit_ctx *pack_tx_ops = &adapter->xmitpriv.ack_tx_ops;
1031
1032 if (txrpt_ccx->int_ccx && adapter->xmitpriv.ack_tx) {
1033 if (txrpt_ccx->pkt_ok)
1034 rtw23a_sctx_done_err(&pack_tx_ops,
1035 RTW_SCTX_DONE_SUCCESS);
1036 else
1037 rtw23a_sctx_done_err(&pack_tx_ops,
1038 RTW_SCTX_DONE_CCX_PKT_FAIL);
1039 }
1040}
1041
Larry Fingerf7c92d22014-03-28 21:37:39 -05001042void rtl8723a_InitAntenna_Selection(struct rtw_adapter *padapter)
1043{
Larry Fingerf7c92d22014-03-28 21:37:39 -05001044 u8 val;
1045
Jes Sorensen050abc42014-05-16 10:05:08 +02001046 val = rtl8723au_read8(padapter, REG_LEDCFG2);
Jiading Gai3d7a8472015-12-16 08:24:58 -05001047 /* Let 8051 take control antenna setting */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001048 val |= BIT(7); /* DPDT_SEL_EN, 0x4C[23] */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001049 rtl8723au_write8(padapter, REG_LEDCFG2, val);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001050}
1051
1052void rtl8723a_CheckAntenna_Selection(struct rtw_adapter *padapter)
1053{
Larry Fingerf7c92d22014-03-28 21:37:39 -05001054 u8 val;
1055
Jes Sorensen050abc42014-05-16 10:05:08 +02001056 val = rtl8723au_read8(padapter, REG_LEDCFG2);
Jiading Gai3d7a8472015-12-16 08:24:58 -05001057 /* Let 8051 take control antenna setting */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001058 if (!(val & BIT(7))) {
1059 val |= BIT(7); /* DPDT_SEL_EN, 0x4C[23] */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001060 rtl8723au_write8(padapter, REG_LEDCFG2, val);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001061 }
1062}
1063
1064void rtl8723a_DeinitAntenna_Selection(struct rtw_adapter *padapter)
1065{
Larry Fingerf7c92d22014-03-28 21:37:39 -05001066 u8 val;
1067
Jes Sorensen050abc42014-05-16 10:05:08 +02001068 val = rtl8723au_read8(padapter, REG_LEDCFG2);
Jiading Gai3d7a8472015-12-16 08:24:58 -05001069 /* Let 8051 take control antenna setting */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001070 val &= ~BIT(7); /* DPDT_SEL_EN, clear 0x4C[23] */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001071 rtl8723au_write8(padapter, REG_LEDCFG2, val);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001072}
1073
1074void rtl8723a_init_default_value(struct rtw_adapter *padapter)
1075{
1076 struct hal_data_8723a *pHalData;
1077 struct dm_priv *pdmpriv;
1078 u8 i;
1079
1080 pHalData = GET_HAL_DATA(padapter);
1081 pdmpriv = &pHalData->dmpriv;
1082
1083 /* init default value */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001084 pHalData->bIQKInitialized = false;
1085 if (!padapter->pwrctrlpriv.bkeepfwalive)
1086 pHalData->LastHMEBoxNum = 0;
1087
1088 pHalData->bIQKInitialized = false;
1089
1090 /* init dm default value */
1091 pdmpriv->TM_Trigger = 0; /* for IQK */
1092/* pdmpriv->binitialized = false; */
1093/* pdmpriv->prv_traffic_idx = 3; */
1094/* pdmpriv->initialize = 0; */
1095
1096 pdmpriv->ThermalValue_HP_index = 0;
1097 for (i = 0; i < HP_THERMAL_NUM; i++)
1098 pdmpriv->ThermalValue_HP[i] = 0;
1099
1100 /* init Efuse variables */
1101 pHalData->EfuseUsedBytes = 0;
1102 pHalData->BTEfuseUsedBytes = 0;
1103}
1104
1105u8 GetEEPROMSize8723A(struct rtw_adapter *padapter)
1106{
1107 u8 size = 0;
1108 u32 cr;
1109
Jes Sorensen050abc42014-05-16 10:05:08 +02001110 cr = rtl8723au_read16(padapter, REG_9346CR);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001111 /* 6: EEPROM used is 93C46, 4: boot from E-Fuse. */
1112 size = (cr & BOOT_FROM_EEPROM) ? 6 : 4;
1113
1114 MSG_8723A("EEPROM type is %s\n", size == 4 ? "E-FUSE" : "93C46");
1115
1116 return size;
1117}
1118
1119/* */
1120/* */
1121/* LLT R/W/Init function */
1122/* */
1123/* */
Jes Sorensena5681f92014-05-16 10:04:48 +02001124static int _LLTWrite(struct rtw_adapter *padapter, u32 address, u32 data)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001125{
Jes Sorensena5681f92014-05-16 10:04:48 +02001126 int status = _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001127 s32 count = 0;
1128 u32 value = _LLT_INIT_ADDR(address) | _LLT_INIT_DATA(data) |
1129 _LLT_OP(_LLT_WRITE_ACCESS);
1130 u16 LLTReg = REG_LLT_INIT;
1131
Jes Sorensenedbfd672014-05-16 10:05:09 +02001132 rtl8723au_write32(padapter, LLTReg, value);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001133
1134 /* polling */
1135 do {
Jes Sorensen050abc42014-05-16 10:05:08 +02001136 value = rtl8723au_read32(padapter, LLTReg);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001137 if (_LLT_NO_ACTIVE == _LLT_OP_VALUE(value))
Larry Fingerf7c92d22014-03-28 21:37:39 -05001138 break;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001139
1140 if (count > POLLING_LLT_THRESHOLD) {
1141 RT_TRACE(_module_hal_init_c_, _drv_err_,
Joe Perches90403aa2015-03-24 16:06:44 -07001142 "Failed to polling write LLT done at address %d!\n",
1143 address);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001144 status = _FAIL;
1145 break;
1146 }
1147 } while (count++);
1148
1149 return status;
1150}
1151
Jes Sorensena5681f92014-05-16 10:04:48 +02001152int InitLLTTable23a(struct rtw_adapter *padapter, u32 boundary)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001153{
Jes Sorensena5681f92014-05-16 10:04:48 +02001154 int status = _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001155 u32 i;
1156 u32 txpktbuf_bndy = boundary;
1157 u32 Last_Entry_Of_TxPktBuf = LAST_ENTRY_OF_TX_PKT_BUFFER;
1158
1159 for (i = 0; i < (txpktbuf_bndy - 1); i++) {
1160 status = _LLTWrite(padapter, i, i + 1);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001161 if (status != _SUCCESS)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001162 return status;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001163 }
1164
1165 /* end of list */
1166 status = _LLTWrite(padapter, (txpktbuf_bndy - 1), 0xFF);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001167 if (status != _SUCCESS)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001168 return status;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001169
1170 /* Make the other pages as ring buffer */
1171 /* This ring buffer is used as beacon buffer if we config this
1172 MAC as two MAC transfer. */
1173 /* Otherwise used as local loopback buffer. */
1174 for (i = txpktbuf_bndy; i < Last_Entry_Of_TxPktBuf; i++) {
1175 status = _LLTWrite(padapter, i, (i + 1));
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001176 if (_SUCCESS != status)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001177 return status;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001178 }
1179
1180 /* Let last entry point to the start entry of ring buffer */
1181 status = _LLTWrite(padapter, Last_Entry_Of_TxPktBuf, txpktbuf_bndy);
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001182 if (status != _SUCCESS)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001183 return status;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001184
1185 return status;
1186}
1187
1188static void _DisableGPIO(struct rtw_adapter *padapter)
1189{
1190/***************************************
1191j. GPIO_PIN_CTRL 0x44[31:0]= 0x000
1192k.Value = GPIO_PIN_CTRL[7:0]
1193l. GPIO_PIN_CTRL 0x44[31:0] = 0x00FF0000 | (value <<8); write external PIN level
1194m. GPIO_MUXCFG 0x42 [15:0] = 0x0780
1195n. LEDCFG 0x4C[15:0] = 0x8080
1196***************************************/
1197 u32 value32;
1198 u32 u4bTmp;
1199
1200 /* 1. Disable GPIO[7:0] */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001201 rtl8723au_write16(padapter, REG_GPIO_PIN_CTRL + 2, 0x0000);
Jes Sorensen050abc42014-05-16 10:05:08 +02001202 value32 = rtl8723au_read32(padapter, REG_GPIO_PIN_CTRL) & 0xFFFF00FF;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001203 u4bTmp = value32 & 0x000000FF;
1204 value32 |= ((u4bTmp << 8) | 0x00FF0000);
Jes Sorensenedbfd672014-05-16 10:05:09 +02001205 rtl8723au_write32(padapter, REG_GPIO_PIN_CTRL, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001206
1207 /* */
1208 /* <Roger_Notes> For RTL8723u multi-function configuration which
1209 was autoload from Efuse offset 0x0a and 0x0b, */
1210 /* WLAN HW GPIO[9], GPS HW GPIO[10] and BT HW GPIO[11]. */
1211 /* Added by Roger, 2010.10.07. */
1212 /* */
1213 /* 2. Disable GPIO[8] and GPIO[12] */
1214
1215 /* Configure all pins as input mode. */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001216 rtl8723au_write16(padapter, REG_GPIO_IO_SEL_2, 0x0000);
Jes Sorensen050abc42014-05-16 10:05:08 +02001217 value32 = rtl8723au_read32(padapter, REG_GPIO_PIN_CTRL_2) & 0xFFFF001F;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001218 u4bTmp = value32 & 0x0000001F;
1219 /* Set pin 8, 10, 11 and pin 12 to output mode. */
1220 value32 |= ((u4bTmp << 8) | 0x001D0000);
Jes Sorensenedbfd672014-05-16 10:05:09 +02001221 rtl8723au_write32(padapter, REG_GPIO_PIN_CTRL_2, value32);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001222
1223 /* 3. Disable LED0 & 1 */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001224 rtl8723au_write16(padapter, REG_LEDCFG0, 0x8080);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001225} /* end of _DisableGPIO() */
1226
1227static void _DisableRFAFEAndResetBB8192C(struct rtw_adapter *padapter)
1228{
1229/**************************************
1230a. TXPAUSE 0x522[7:0] = 0xFF Pause MAC TX queue
1231b. RF path 0 offset 0x00 = 0x00 disable RF
1232c. APSD_CTRL 0x600[7:0] = 0x40
1233d. SYS_FUNC_EN 0x02[7:0] = 0x16 reset BB state machine
1234e. SYS_FUNC_EN 0x02[7:0] = 0x14 reset BB state machine
1235***************************************/
Jes Sorensenaba3eb12014-11-30 16:05:14 -05001236 u8 value8;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001237
Jes Sorensenedbfd672014-05-16 10:05:09 +02001238 rtl8723au_write8(padapter, REG_TXPAUSE, 0xFF);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001239
Jes Sorensenb1e111d2014-11-30 16:05:13 -05001240 PHY_SetRFReg(padapter, RF_PATH_A, 0x0, bMaskByte0, 0x0);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001241
Jes Sorensenaba3eb12014-11-30 16:05:14 -05001242 value8 = APSDOFF;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001243 rtl8723au_write8(padapter, REG_APSD_CTRL, value8); /* 0x40 */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001244
1245 /* Set BB reset at first */
Jes Sorensenaba3eb12014-11-30 16:05:14 -05001246 value8 = FEN_USBD | FEN_USBA | FEN_BB_GLB_RSTn;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001247 rtl8723au_write8(padapter, REG_SYS_FUNC_EN, value8); /* 0x16 */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001248
1249 /* Set global reset. */
1250 value8 &= ~FEN_BB_GLB_RSTn;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001251 rtl8723au_write8(padapter, REG_SYS_FUNC_EN, value8); /* 0x14 */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001252
1253 /* 2010/08/12 MH We need to set BB/GLBAL reset to save power
1254 for SS mode. */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001255}
1256
Larry Fingerf7c92d22014-03-28 21:37:39 -05001257static void _ResetDigitalProcedure1_92C(struct rtw_adapter *padapter,
1258 bool bWithoutHWSM)
1259{
1260 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1261
1262 if (IS_FW_81xxC(padapter) && (pHalData->FirmwareVersion <= 0x20)) {
1263 /*****************************
1264 f. MCUFWDL 0x80[7:0]= 0 reset MCU ready status
1265 g. SYS_FUNC_EN 0x02[10]= 0 reset MCU register, (8051 reset)
1266 h. SYS_FUNC_EN 0x02[15-12]= 5 reset MAC register, DCORE
1267 i. SYS_FUNC_EN 0x02[10]= 1 enable MCU register,
1268 (8051 enable)
1269 ******************************/
Jes Sorensen31b7e842014-11-30 16:05:16 -05001270 u16 valu16;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001271 rtl8723au_write8(padapter, REG_MCUFWDL, 0);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001272
Jes Sorensen050abc42014-05-16 10:05:08 +02001273 valu16 = rtl8723au_read16(padapter, REG_SYS_FUNC_EN);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001274 /* reset MCU , 8051 */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001275 rtl8723au_write16(padapter, REG_SYS_FUNC_EN,
Jes Sorensen31b7e842014-11-30 16:05:16 -05001276 valu16 & ~FEN_CPUEN);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001277
Jes Sorensen050abc42014-05-16 10:05:08 +02001278 valu16 = rtl8723au_read16(padapter, REG_SYS_FUNC_EN) & 0x0FFF;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001279 /* reset MAC */
1280 rtl8723au_write16(padapter, REG_SYS_FUNC_EN,
Jes Sorensen31b7e842014-11-30 16:05:16 -05001281 valu16 | FEN_HWPDN | FEN_ELDR);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001282
Jes Sorensen050abc42014-05-16 10:05:08 +02001283 valu16 = rtl8723au_read16(padapter, REG_SYS_FUNC_EN);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001284 /* enable MCU , 8051 */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001285 rtl8723au_write16(padapter, REG_SYS_FUNC_EN,
1286 valu16 | FEN_CPUEN);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001287 } else {
1288 u8 retry_cnts = 0;
Jes Sorensen31b7e842014-11-30 16:05:16 -05001289 u8 val8;
1290
1291 val8 = rtl8723au_read8(padapter, REG_MCUFWDL);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001292
1293 /* 2010/08/12 MH For USB SS, we can not stop 8051 when we
1294 are trying to enter IPS/HW&SW radio off. For
1295 S3/S4/S5/Disable, we can stop 8051 because */
1296 /* we will init FW when power on again. */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001297 /* If we want to SS mode, we can not reset 8051. */
Jes Sorensen31b7e842014-11-30 16:05:16 -05001298 if ((val8 & BIT(1)) && padapter->bFWReady) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001299 /* IF fw in RAM code, do reset */
Jiading Gai3d7a8472015-12-16 08:24:58 -05001300 /* 2010/08/25 MH According to RD alfred's
Jes Sorensen31b7e842014-11-30 16:05:16 -05001301 suggestion, we need to disable other */
1302 /* HRCV INT to influence 8051 reset. */
1303 rtl8723au_write8(padapter, REG_FWIMR, 0x20);
1304 /* 2011/02/15 MH According to Alex's
1305 suggestion, close mask to prevent
1306 incorrect FW write operation. */
1307 rtl8723au_write8(padapter, REG_FTIMR, 0x00);
1308 rtl8723au_write8(padapter, REG_FSIMR, 0x00);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001309
Jes Sorensen31b7e842014-11-30 16:05:16 -05001310 /* 8051 reset by self */
1311 rtl8723au_write8(padapter, REG_HMETFR + 3, 0x20);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001312
Jes Sorensen31b7e842014-11-30 16:05:16 -05001313 while ((retry_cnts++ < 100) &&
1314 (rtl8723au_read16(padapter, REG_SYS_FUNC_EN) &
1315 FEN_CPUEN)) {
1316 udelay(50); /* us */
1317 }
Larry Fingerf7c92d22014-03-28 21:37:39 -05001318
Jes Sorensen31b7e842014-11-30 16:05:16 -05001319 if (retry_cnts >= 100) {
1320 /* Reset MAC and Enable 8051 */
1321 rtl8723au_write8(padapter,
1322 REG_SYS_FUNC_EN + 1, 0x50);
1323 mdelay(10);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001324 }
1325 }
1326 /* Reset MAC and Enable 8051 */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001327 rtl8723au_write8(padapter, REG_SYS_FUNC_EN + 1, 0x54);
1328 rtl8723au_write8(padapter, REG_MCUFWDL, 0);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001329 }
1330
1331 if (bWithoutHWSM) {
1332 /*****************************
1333 Without HW auto state machine
1334 g. SYS_CLKR 0x08[15:0] = 0x30A3 disable MAC clock
1335 h. AFE_PLL_CTRL 0x28[7:0] = 0x80 disable AFE PLL
1336 i. AFE_XTAL_CTRL 0x24[15:0] = 0x880F gated AFE DIG_CLOCK
1337 j. SYS_ISO_CTRL 0x00[7:0] = 0xF9 isolated digital to PON
1338 ******************************/
1339 /* modify to 0x70A3 by Scott. */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001340 rtl8723au_write16(padapter, REG_SYS_CLKR, 0x70A3);
1341 rtl8723au_write8(padapter, REG_AFE_PLL_CTRL, 0x80);
1342 rtl8723au_write16(padapter, REG_AFE_XTAL_CTRL, 0x880F);
1343 rtl8723au_write8(padapter, REG_SYS_ISO_CTRL, 0xF9);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001344 } else {
1345 /* Disable all RF/BB power */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001346 rtl8723au_write8(padapter, REG_RF_CTRL, 0x00);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001347 }
1348}
1349
Larry Fingerf7c92d22014-03-28 21:37:39 -05001350static void _ResetDigitalProcedure2(struct rtw_adapter *padapter)
1351{
1352/*****************************
1353k. SYS_FUNC_EN 0x03[7:0] = 0x44 disable ELDR runction
1354l. SYS_CLKR 0x08[15:0] = 0x3083 disable ELDR clock
1355m. SYS_ISO_CTRL 0x01[7:0] = 0x83 isolated ELDR to PON
1356******************************/
1357 /* modify to 0x70a3 by Scott. */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001358 rtl8723au_write16(padapter, REG_SYS_CLKR, 0x70a3);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001359 /* modify to 0x82 by Scott. */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001360 rtl8723au_write8(padapter, REG_SYS_ISO_CTRL + 1, 0x82);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001361}
1362
1363static void _DisableAnalog(struct rtw_adapter *padapter, bool bWithoutHWSM)
1364{
1365 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
Jes Sorensencffca682014-11-30 16:05:18 -05001366 u16 value16;
1367 u8 value8;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001368
1369 if (bWithoutHWSM) {
1370 /*****************************
1371 n. LDOA15_CTRL 0x20[7:0] = 0x04 disable A15 power
1372 o. LDOV12D_CTRL 0x21[7:0] = 0x54 disable digital core power
1373 r. When driver call disable, the ASIC will turn off remaining
1374 clock automatically
1375 ******************************/
1376
Jes Sorensenedbfd672014-05-16 10:05:09 +02001377 rtl8723au_write8(padapter, REG_LDOA15_CTRL, 0x04);
1378 /* rtl8723au_write8(padapter, REG_LDOV12D_CTRL, 0x54); */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001379
Jes Sorensen050abc42014-05-16 10:05:08 +02001380 value8 = rtl8723au_read8(padapter, REG_LDOV12D_CTRL);
Jes Sorensencffca682014-11-30 16:05:18 -05001381 value8 &= ~LDV12_EN;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001382 rtl8723au_write8(padapter, REG_LDOV12D_CTRL, value8);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001383 }
1384
1385 /*****************************
1386 h. SPS0_CTRL 0x11[7:0] = 0x23 enter PFM mode
1387 i. APS_FSMCO 0x04[15:0] = 0x4802 set USB suspend
1388 ******************************/
1389 value8 = 0x23;
1390 if (IS_81xxC_VENDOR_UMC_B_CUT(pHalData->VersionID))
Jes Sorensenbc0d10c2014-05-16 10:03:49 +02001391 value8 |= BIT(3);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001392
Jes Sorensenedbfd672014-05-16 10:05:09 +02001393 rtl8723au_write8(padapter, REG_SPS0_CTRL, value8);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001394
1395 if (bWithoutHWSM) {
1396 /* value16 |= (APDM_HOST | FSM_HSUS |/PFM_ALDN); */
1397 /* 2010/08/31 According to Filen description, we need to
1398 use HW to shut down 8051 automatically. */
Carlos E. Garcia69e98df2015-04-24 09:40:42 -04001399 /* Because suspend operation need the asistance of 8051
Larry Fingerf7c92d22014-03-28 21:37:39 -05001400 to wait for 3ms. */
Jes Sorensencffca682014-11-30 16:05:18 -05001401 value16 = APDM_HOST | AFSM_HSUS | PFM_ALDN;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001402 } else {
Jes Sorensencffca682014-11-30 16:05:18 -05001403 value16 = APDM_HOST | AFSM_HSUS | PFM_ALDN;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001404 }
1405
Jes Sorensenedbfd672014-05-16 10:05:09 +02001406 rtl8723au_write16(padapter, REG_APS_FSMCO, value16); /* 0x4802 */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001407
Jes Sorensenedbfd672014-05-16 10:05:09 +02001408 rtl8723au_write8(padapter, REG_RSV_CTRL, 0x0e);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001409}
1410
1411/* HW Auto state machine */
Jes Sorensena5681f92014-05-16 10:04:48 +02001412int CardDisableHWSM(struct rtw_adapter *padapter, u8 resetMCU)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001413{
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001414 if (padapter->bSurpriseRemoved)
Roberta Dobrescu147e45d2014-10-26 23:30:07 +02001415 return _SUCCESS;
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001416
Larry Fingerf7c92d22014-03-28 21:37:39 -05001417 /* RF Off Sequence ==== */
Jes Sorensen38f36d82014-11-30 16:05:15 -05001418 _DisableRFAFEAndResetBB8192C(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001419
1420 /* ==== Reset digital sequence ====== */
Jes Sorensen441272e2014-11-30 16:05:17 -05001421 _ResetDigitalProcedure1_92C(padapter, false);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001422
1423 /* ==== Pull GPIO PIN to balance level and LED control ====== */
1424 _DisableGPIO(padapter);
1425
1426 /* ==== Disable analog sequence === */
1427 _DisableAnalog(padapter, false);
1428
1429 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001430 "======> Card disable finished.\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -05001431
Roberta Dobrescu147e45d2014-10-26 23:30:07 +02001432 return _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001433}
1434
1435/* without HW Auto state machine */
Jes Sorensena5681f92014-05-16 10:04:48 +02001436int CardDisableWithoutHWSM(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001437{
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001438 if (padapter->bSurpriseRemoved)
Roberta Dobrescu147e45d2014-10-26 23:30:07 +02001439 return _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001440
1441 /* RF Off Sequence ==== */
Jes Sorensen38f36d82014-11-30 16:05:15 -05001442 _DisableRFAFEAndResetBB8192C(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001443
1444 /* ==== Reset digital sequence ====== */
Jes Sorensen441272e2014-11-30 16:05:17 -05001445 _ResetDigitalProcedure1_92C(padapter, true);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001446
1447 /* ==== Pull GPIO PIN to balance level and LED control ====== */
1448 _DisableGPIO(padapter);
1449
1450 /* ==== Reset digital sequence ====== */
1451 _ResetDigitalProcedure2(padapter);
1452
1453 /* ==== Disable analog sequence === */
1454 _DisableAnalog(padapter, true);
1455
Roberta Dobrescu147e45d2014-10-26 23:30:07 +02001456 return _SUCCESS;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001457}
1458
1459void Hal_InitPGData(struct rtw_adapter *padapter, u8 *PROMContent)
1460{
1461 struct eeprom_priv *pEEPROM = GET_EEPROM_EFUSE_PRIV(padapter);
1462
Jes Sorensen315f9a22014-11-30 16:05:19 -05001463 if (!pEEPROM->bautoload_fail_flag) { /* autoload OK. */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001464 if (!pEEPROM->EepromOrEfuse) {
1465 /* Read EFUSE real map to shadow. */
1466 EFUSE_ShadowMapUpdate23a(padapter, EFUSE_WIFI);
Jes Sorensen315f9a22014-11-30 16:05:19 -05001467 memcpy(PROMContent, pEEPROM->efuse_eeprom_data,
Larry Fingerf7c92d22014-03-28 21:37:39 -05001468 HWSET_MAX_SIZE);
1469 }
Jes Sorensen315f9a22014-11-30 16:05:19 -05001470 } else {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001471 RT_TRACE(_module_hci_hal_init_c_, _drv_notice_,
Joe Perches90403aa2015-03-24 16:06:44 -07001472 "AutoLoad Fail reported from CR9346!!\n");
Larry Fingerf7c92d22014-03-28 21:37:39 -05001473 /* update to default value 0xFF */
Jes Sorensen315f9a22014-11-30 16:05:19 -05001474 if (!pEEPROM->EepromOrEfuse)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001475 EFUSE_ShadowMapUpdate23a(padapter, EFUSE_WIFI);
Jes Sorensen315f9a22014-11-30 16:05:19 -05001476 memcpy(PROMContent, pEEPROM->efuse_eeprom_data,
Larry Fingerf7c92d22014-03-28 21:37:39 -05001477 HWSET_MAX_SIZE);
1478 }
1479}
1480
1481void Hal_EfuseParseIDCode(struct rtw_adapter *padapter, u8 *hwinfo)
1482{
1483 struct eeprom_priv *pEEPROM = GET_EEPROM_EFUSE_PRIV(padapter);
1484/* struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter); */
1485 u16 EEPROMId;
1486
1487 /* Checl 0x8129 again for making sure autoload status!! */
Juston Li711c9b32015-05-22 22:03:04 -07001488 EEPROMId = le16_to_cpu(*((__le16 *) hwinfo));
Larry Fingerf7c92d22014-03-28 21:37:39 -05001489 if (EEPROMId != RTL_EEPROM_ID) {
1490 DBG_8723A("EEPROM ID(%#x) is invalid!!\n", EEPROMId);
1491 pEEPROM->bautoload_fail_flag = true;
1492 } else {
1493 pEEPROM->bautoload_fail_flag = false;
1494 }
1495
1496 RT_TRACE(_module_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001497 "EEPROM ID = 0x%04x\n", EEPROMId);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001498}
1499
Larry Fingerf7c92d22014-03-28 21:37:39 -05001500static void
1501Hal_ReadPowerValueFromPROM_8723A(struct txpowerinfo *pwrInfo,
1502 u8 *PROMContent, bool AutoLoadFail)
1503{
1504 u32 rfPath, eeAddr, group, rfPathMax = 1;
1505
1506 memset(pwrInfo, 0, sizeof(*pwrInfo));
1507
1508 if (AutoLoadFail) {
1509 for (group = 0; group < MAX_CHNL_GROUP; group++) {
1510 for (rfPath = 0; rfPath < rfPathMax; rfPath++) {
1511 pwrInfo->CCKIndex[rfPath][group] =
1512 EEPROM_Default_TxPowerLevel;
1513 pwrInfo->HT40_1SIndex[rfPath][group] =
1514 EEPROM_Default_TxPowerLevel;
1515 pwrInfo->HT40_2SIndexDiff[rfPath][group] =
1516 EEPROM_Default_HT40_2SDiff;
1517 pwrInfo->HT20IndexDiff[rfPath][group] =
1518 EEPROM_Default_HT20_Diff;
1519 pwrInfo->OFDMIndexDiff[rfPath][group] =
1520 EEPROM_Default_LegacyHTTxPowerDiff;
1521 pwrInfo->HT40MaxOffset[rfPath][group] =
1522 EEPROM_Default_HT40_PwrMaxOffset;
1523 pwrInfo->HT20MaxOffset[rfPath][group] =
1524 EEPROM_Default_HT20_PwrMaxOffset;
1525 }
1526 }
1527 pwrInfo->TSSI_A[0] = EEPROM_Default_TSSI;
1528 return;
1529 }
1530
1531 for (rfPath = 0; rfPath < rfPathMax; rfPath++) {
1532 for (group = 0; group < MAX_CHNL_GROUP; group++) {
1533 eeAddr =
1534 EEPROM_CCK_TX_PWR_INX_8723A + (rfPath * 3) + group;
Jes Sorensen96a98aa2015-09-15 16:32:44 -04001535
1536 pwrInfo->CCKIndex[rfPath][group] = PROMContent[eeAddr];
1537 if (pwrInfo->CCKIndex[rfPath][group] > 63)
1538 pwrInfo->CCKIndex[rfPath][group] =
1539 EEPROM_Default_TxPowerLevel;
1540
Larry Fingerf7c92d22014-03-28 21:37:39 -05001541 eeAddr = EEPROM_HT40_1S_TX_PWR_INX_8723A +
1542 (rfPath * 3) + group;
Jes Sorensen96a98aa2015-09-15 16:32:44 -04001543 pwrInfo->HT40_1SIndex[rfPath][group] =
1544 PROMContent[eeAddr];
1545 if (pwrInfo->HT40_1SIndex[rfPath][group] > 63)
1546 pwrInfo->HT40_1SIndex[rfPath][group] =
1547 EEPROM_Default_TxPowerLevel;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001548 }
1549 }
1550
1551 for (group = 0; group < MAX_CHNL_GROUP; group++) {
1552 for (rfPath = 0; rfPath < rfPathMax; rfPath++) {
1553 pwrInfo->HT40_2SIndexDiff[rfPath][group] = 0;
1554 pwrInfo->HT20IndexDiff[rfPath][group] =
1555 (PROMContent
1556 [EEPROM_HT20_TX_PWR_INX_DIFF_8723A +
1557 group] >> (rfPath * 4)) & 0xF;
1558 /* 4bit sign number to 8 bit sign number */
Jes Sorensenbc0d10c2014-05-16 10:03:49 +02001559 if (pwrInfo->HT20IndexDiff[rfPath][group] & BIT(3))
Larry Fingerf7c92d22014-03-28 21:37:39 -05001560 pwrInfo->HT20IndexDiff[rfPath][group] |= 0xF0;
1561
1562 pwrInfo->OFDMIndexDiff[rfPath][group] =
1563 (PROMContent[EEPROM_OFDM_TX_PWR_INX_DIFF_8723A +
1564 group] >> (rfPath * 4)) & 0xF;
1565
1566 pwrInfo->HT40MaxOffset[rfPath][group] =
1567 (PROMContent[EEPROM_HT40_MAX_PWR_OFFSET_8723A +
1568 group] >> (rfPath * 4)) & 0xF;
1569
1570 pwrInfo->HT20MaxOffset[rfPath][group] =
1571 (PROMContent[EEPROM_HT20_MAX_PWR_OFFSET_8723A +
1572 group] >> (rfPath * 4)) & 0xF;
1573 }
1574 }
1575
1576 pwrInfo->TSSI_A[0] = PROMContent[EEPROM_TSSI_A_8723A];
1577}
1578
1579static u8 Hal_GetChnlGroup(u8 chnl)
1580{
1581 u8 group = 0;
1582
1583 if (chnl < 3) /* Cjanel 1-3 */
1584 group = 0;
1585 else if (chnl < 9) /* Channel 4-9 */
1586 group = 1;
1587 else /* Channel 10-14 */
1588 group = 2;
1589
1590 return group;
1591}
1592
1593void
1594Hal_EfuseParsetxpowerinfo_8723A(struct rtw_adapter *padapter,
1595 u8 *PROMContent, bool AutoLoadFail)
1596{
1597 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1598 struct txpowerinfo pwrInfo;
1599 u8 rfPath, ch, group, rfPathMax = 1;
1600 u8 pwr, diff;
1601
1602 Hal_ReadPowerValueFromPROM_8723A(&pwrInfo, PROMContent, AutoLoadFail);
1603 for (rfPath = 0; rfPath < rfPathMax; rfPath++) {
1604 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++) {
1605 group = Hal_GetChnlGroup(ch);
1606
1607 pHalData->TxPwrLevelCck[rfPath][ch] =
1608 pwrInfo.CCKIndex[rfPath][group];
1609 pHalData->TxPwrLevelHT40_1S[rfPath][ch] =
1610 pwrInfo.HT40_1SIndex[rfPath][group];
1611
1612 pHalData->TxPwrHt20Diff[rfPath][ch] =
1613 pwrInfo.HT20IndexDiff[rfPath][group];
1614 pHalData->TxPwrLegacyHtDiff[rfPath][ch] =
1615 pwrInfo.OFDMIndexDiff[rfPath][group];
1616 pHalData->PwrGroupHT20[rfPath][ch] =
1617 pwrInfo.HT20MaxOffset[rfPath][group];
1618 pHalData->PwrGroupHT40[rfPath][ch] =
1619 pwrInfo.HT40MaxOffset[rfPath][group];
1620
1621 pwr = pwrInfo.HT40_1SIndex[rfPath][group];
1622 diff = pwrInfo.HT40_2SIndexDiff[rfPath][group];
1623
1624 pHalData->TxPwrLevelHT40_2S[rfPath][ch] =
1625 (pwr > diff) ? (pwr - diff) : 0;
1626 }
1627 }
1628 for (rfPath = 0; rfPath < RF_PATH_MAX; rfPath++) {
1629 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++) {
1630 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001631 "RF(%u)-Ch(%u) [CCK / HT40_1S / HT40_2S] = [0x%x / 0x%x / 0x%x]\n",
1632 rfPath, ch,
1633 pHalData->TxPwrLevelCck[rfPath][ch],
1634 pHalData->TxPwrLevelHT40_1S[rfPath][ch],
1635 pHalData->TxPwrLevelHT40_2S[rfPath][ch]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001636
1637 }
1638 }
1639 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++) {
1640 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001641 "RF-A Ht20 to HT40 Diff[%u] = 0x%x(%d)\n", ch,
1642 pHalData->TxPwrHt20Diff[RF_PATH_A][ch],
1643 pHalData->TxPwrHt20Diff[RF_PATH_A][ch]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001644 }
1645 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++)
1646 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001647 "RF-A Legacy to Ht40 Diff[%u] = 0x%x\n", ch,
1648 pHalData->TxPwrLegacyHtDiff[RF_PATH_A][ch]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001649 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++) {
1650 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001651 "RF-B Ht20 to HT40 Diff[%u] = 0x%x(%d)\n", ch,
1652 pHalData->TxPwrHt20Diff[RF_PATH_B][ch],
1653 pHalData->TxPwrHt20Diff[RF_PATH_B][ch]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001654 }
1655 for (ch = 0; ch < CHANNEL_MAX_NUMBER; ch++)
1656 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001657 "RF-B Legacy to HT40 Diff[%u] = 0x%x\n", ch,
1658 pHalData->TxPwrLegacyHtDiff[RF_PATH_B][ch]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001659 if (!AutoLoadFail) {
1660 struct registry_priv *registry_par = &padapter->registrypriv;
1661 if (registry_par->regulatory_tid == 0xff) {
1662 if (PROMContent[RF_OPTION1_8723A] == 0xff)
1663 pHalData->EEPROMRegulatory = 0;
1664 else
1665 pHalData->EEPROMRegulatory =
1666 PROMContent[RF_OPTION1_8723A] & 0x7;
1667 } else {
1668 pHalData->EEPROMRegulatory =
1669 registry_par->regulatory_tid;
1670 }
1671 } else {
1672 pHalData->EEPROMRegulatory = 0;
1673 }
1674 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001675 "EEPROMRegulatory = 0x%x\n", pHalData->EEPROMRegulatory);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001676
1677 if (!AutoLoadFail)
1678 pHalData->bTXPowerDataReadFromEEPORM = true;
1679}
1680
1681void
1682Hal_EfuseParseBTCoexistInfo_8723A(struct rtw_adapter *padapter,
1683 u8 *hwinfo, bool AutoLoadFail)
1684{
1685 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1686 u8 tempval;
1687 u32 tmpu4;
1688
1689 if (!AutoLoadFail) {
Jes Sorensen050abc42014-05-16 10:05:08 +02001690 tmpu4 = rtl8723au_read32(padapter, REG_MULTI_FUNC_CTRL);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001691 if (tmpu4 & BT_FUNC_EN)
1692 pHalData->EEPROMBluetoothCoexist = 1;
1693 else
1694 pHalData->EEPROMBluetoothCoexist = 0;
1695 pHalData->EEPROMBluetoothType = BT_RTL8723A;
1696
1697 /* The following need to be checked with newer version of */
1698 /* eeprom spec */
1699 tempval = hwinfo[RF_OPTION4_8723A];
1700 pHalData->EEPROMBluetoothAntNum = (tempval & 0x1);
Aya Mahfouze1bc88f2015-03-04 07:34:07 +02001701 pHalData->EEPROMBluetoothAntIsolation = (tempval & 0x10) >> 4;
1702 pHalData->EEPROMBluetoothRadioShared = (tempval & 0x20) >> 5;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001703 } else {
1704 pHalData->EEPROMBluetoothCoexist = 0;
1705 pHalData->EEPROMBluetoothType = BT_RTL8723A;
1706 pHalData->EEPROMBluetoothAntNum = Ant_x2;
1707 pHalData->EEPROMBluetoothAntIsolation = 0;
1708 pHalData->EEPROMBluetoothRadioShared = BT_Radio_Shared;
1709 }
Jes Sorensen9b2d4ae2014-05-25 22:43:44 +02001710
1711 rtl8723a_BT_init_hal_vars(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001712}
1713
1714void
1715Hal_EfuseParseEEPROMVer(struct rtw_adapter *padapter,
1716 u8 *hwinfo, bool AutoLoadFail)
1717{
1718 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1719
1720 if (!AutoLoadFail)
1721 pHalData->EEPROMVersion = hwinfo[EEPROM_VERSION_8723A];
1722 else
1723 pHalData->EEPROMVersion = 1;
1724 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001725 "Hal_EfuseParseEEPROMVer(), EEVer = %d\n",
1726 pHalData->EEPROMVersion);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001727}
1728
1729void
1730rtl8723a_EfuseParseChnlPlan(struct rtw_adapter *padapter,
1731 u8 *hwinfo, bool AutoLoadFail)
1732{
1733 padapter->mlmepriv.ChannelPlan =
1734 hal_com_get_channel_plan23a(padapter, hwinfo ?
1735 hwinfo[EEPROM_ChannelPlan_8723A]:0xFF,
1736 padapter->registrypriv.channel_plan,
1737 RT_CHANNEL_DOMAIN_WORLD_WIDE_13,
1738 AutoLoadFail);
1739
1740 DBG_8723A("mlmepriv.ChannelPlan = 0x%02x\n",
1741 padapter->mlmepriv.ChannelPlan);
1742}
1743
1744void
1745Hal_EfuseParseCustomerID(struct rtw_adapter *padapter,
1746 u8 *hwinfo, bool AutoLoadFail)
1747{
1748 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1749
1750 if (!AutoLoadFail) {
1751 pHalData->EEPROMCustomerID = hwinfo[EEPROM_CustomID_8723A];
1752 pHalData->EEPROMSubCustomerID =
1753 hwinfo[EEPROM_SubCustomID_8723A];
1754 } else {
1755 pHalData->EEPROMCustomerID = 0;
1756 pHalData->EEPROMSubCustomerID = 0;
1757 }
1758 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001759 "EEPROM Customer ID: 0x%2x\n", pHalData->EEPROMCustomerID);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001760 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001761 "EEPROM SubCustomer ID: 0x%02x\n",
1762 pHalData->EEPROMSubCustomerID);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001763}
1764
1765void
1766Hal_EfuseParseAntennaDiversity(struct rtw_adapter *padapter,
1767 u8 *hwinfo, bool AutoLoadFail)
1768{
1769}
1770
1771void
1772Hal_EfuseParseRateIndicationOption(struct rtw_adapter *padapter,
1773 u8 *hwinfo, bool AutoLoadFail)
1774{
1775}
1776
1777void
1778Hal_EfuseParseXtal_8723A(struct rtw_adapter *pAdapter,
1779 u8 *hwinfo, u8 AutoLoadFail)
1780{
1781 struct hal_data_8723a *pHalData = GET_HAL_DATA(pAdapter);
1782
1783 if (!AutoLoadFail) {
1784 pHalData->CrystalCap = hwinfo[EEPROM_XTAL_K_8723A];
1785 if (pHalData->CrystalCap == 0xFF)
1786 pHalData->CrystalCap = EEPROM_Default_CrystalCap_8723A;
1787 } else {
1788 pHalData->CrystalCap = EEPROM_Default_CrystalCap_8723A;
1789 }
1790 RT_TRACE(_module_hci_hal_init_c_, _drv_info_,
Joe Perches90403aa2015-03-24 16:06:44 -07001791 "%s: CrystalCap = 0x%2x\n", __func__,
1792 pHalData->CrystalCap);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001793}
1794
1795void
1796Hal_EfuseParseThermalMeter_8723A(struct rtw_adapter *padapter,
Jes Sorensena5681f92014-05-16 10:04:48 +02001797 u8 *PROMContent, bool AutoloadFail)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001798{
1799 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
1800
1801 /* */
1802 /* ThermalMeter from EEPROM */
1803 /* */
Roberta Dobrescu08551cb2014-10-26 23:30:05 +02001804 if (!AutoloadFail)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001805 pHalData->EEPROMThermalMeter =
1806 PROMContent[EEPROM_THERMAL_METER_8723A];
1807 else
1808 pHalData->EEPROMThermalMeter = EEPROM_Default_ThermalMeter;
1809
Roberta Dobrescu08551cb2014-10-26 23:30:05 +02001810 if ((pHalData->EEPROMThermalMeter == 0xff) || AutoloadFail) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001811 pHalData->bAPKThermalMeterIgnore = true;
1812 pHalData->EEPROMThermalMeter = EEPROM_Default_ThermalMeter;
1813 }
1814
1815 DBG_8723A("%s: ThermalMeter = 0x%x\n", __func__,
1816 pHalData->EEPROMThermalMeter);
1817}
1818
Larry Fingerf7c92d22014-03-28 21:37:39 -05001819static void rtl8723a_cal_txdesc_chksum(struct tx_desc *ptxdesc)
1820{
Johannes Postma8fc8cf42015-08-06 12:21:52 +01001821 __le16 *usPtr = (__le16 *)ptxdesc;
Larry Fingerf7c92d22014-03-28 21:37:39 -05001822 u32 count = 16; /* (32 bytes / 2 bytes per XOR) => 16 times */
1823 u32 index;
1824 u16 checksum = 0;
1825
1826 /* Clear first */
1827 ptxdesc->txdw7 &= cpu_to_le32(0xffff0000);
1828
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001829 for (index = 0; index < count; index++)
Johannes Postma25e66322015-08-07 14:04:34 +01001830 checksum ^= le16_to_cpu(usPtr[index]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001831
1832 ptxdesc->txdw7 |= cpu_to_le32(checksum & 0x0000ffff);
1833}
1834
Larry Fingerf7c92d22014-03-28 21:37:39 -05001835/*
1836 * Description: In normal chip, we should send some packet to Hw which
1837 * will be used by Fw in FW LPS mode. The function is to fill the Tx
1838 * descriptor of this packets, then
1839 */
1840/* Fw can tell Hw to send these packet derectly. */
1841/* Added by tynli. 2009.10.15. */
1842/* */
1843void rtl8723a_fill_fake_txdesc(struct rtw_adapter *padapter, u8 *pDesc,
1844 u32 BufferLen, u8 IsPsPoll, u8 IsBTQosNull)
1845{
1846 struct tx_desc *ptxdesc;
1847
1848 /* Clear all status */
1849 ptxdesc = (struct tx_desc *)pDesc;
1850 memset(pDesc, 0, TXDESC_SIZE);
1851
1852 /* offset 0 */
1853 /* own, bFirstSeg, bLastSeg; */
1854 ptxdesc->txdw0 |= cpu_to_le32(OWN | FSG | LSG);
1855
1856 /* 32 bytes for TX Desc */
1857 ptxdesc->txdw0 |= cpu_to_le32(((TXDESC_SIZE + OFFSET_SZ) <<
1858 OFFSET_SHT) & 0x00ff0000);
1859
1860 /* Buffer size + command header */
1861 ptxdesc->txdw0 |= cpu_to_le32(BufferLen & 0x0000ffff);
1862
1863 /* offset 4 */
1864 /* Fixed queue of Mgnt queue */
1865 ptxdesc->txdw1 |= cpu_to_le32((QSLT_MGNT << QSEL_SHT) & 0x00001f00);
1866
1867 /* Set NAVUSEHDR to prevent Ps-poll AId filed to be changed
1868 to error vlaue by Hw. */
1869 if (IsPsPoll) {
1870 ptxdesc->txdw1 |= cpu_to_le32(NAVUSEHDR);
1871 } else {
1872 /* Hw set sequence number */
1873 ptxdesc->txdw4 |= cpu_to_le32(BIT(7));
1874 /* set bit3 to 1. Suugested by TimChen. 2009.12.29. */
1875 ptxdesc->txdw3 |= cpu_to_le32((8 << 28));
1876 }
1877
Daniele Alessandrellidd03eed2015-02-22 20:28:44 +01001878 if (true == IsBTQosNull)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001879 ptxdesc->txdw2 |= cpu_to_le32(BIT(23)); /* BT NULL */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001880
1881 /* offset 16 */
1882 ptxdesc->txdw4 |= cpu_to_le32(BIT(8)); /* driver uses rate */
1883
1884 /* USB interface drop packet if the checksum of descriptor isn't
1885 correct. */
1886 /* Using this checksum can let hardware recovery from packet bulk
1887 out error (e.g. Cancel URC, Bulk out error.). */
1888 rtl8723a_cal_txdesc_chksum(ptxdesc);
1889}
1890
Jes Sorensendbdcd362014-04-09 23:20:17 +02001891void hw_var_set_opmode(struct rtw_adapter *padapter, u8 mode)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001892{
1893 u8 val8;
1894
Jes Sorensen6ec26272014-07-17 22:59:46 +02001895 if (mode == MSR_INFRA || mode == MSR_NOLINK) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001896 StopTxBeacon(padapter);
1897
1898 /* disable atim wnd */
1899 val8 = DIS_TSF_UDT | EN_BCN_FUNCTION | DIS_ATIM;
1900 SetBcnCtrlReg23a(padapter, val8, ~val8);
Jes Sorensen6ec26272014-07-17 22:59:46 +02001901 } else if (mode == MSR_ADHOC) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001902 ResumeTxBeacon(padapter);
1903
1904 val8 = DIS_TSF_UDT | EN_BCN_FUNCTION | DIS_BCNQ_SUB;
1905 SetBcnCtrlReg23a(padapter, val8, ~val8);
Jes Sorensen6ec26272014-07-17 22:59:46 +02001906 } else if (mode == MSR_AP) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001907 /* add NULL Data and BT NULL Data Packets to FW RSVD Page */
1908 rtl8723a_set_BTCoex_AP_mode_FwRsvdPkt_cmd(padapter);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001909
1910 ResumeTxBeacon(padapter);
1911
1912 val8 = DIS_TSF_UDT | DIS_BCNQ_SUB;
1913 SetBcnCtrlReg23a(padapter, val8, ~val8);
1914
1915 /* Set RCR */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001916 /* rtl8723au_write32(padapter, REG_RCR, 0x70002a8e);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001917 CBSSID_DATA must set to 0 */
1918 /* CBSSID_DATA must set to 0 */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001919 rtl8723au_write32(padapter, REG_RCR, 0x7000228e);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001920 /* enable to rx data frame */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001921 rtl8723au_write16(padapter, REG_RXFLTMAP2, 0xFFFF);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001922 /* enable to rx ps-poll */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001923 rtl8723au_write16(padapter, REG_RXFLTMAP1, 0x0400);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001924
1925 /* Beacon Control related register for first time */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001926 /* 2ms */
1927 rtl8723au_write8(padapter, REG_BCNDMATIM, 0x02);
1928 /* 5ms */
1929 rtl8723au_write8(padapter, REG_DRVERLYINT, 0x05);
1930 /* 10ms for port0 */
1931 rtl8723au_write8(padapter, REG_ATIMWND, 0x0a);
1932 rtl8723au_write16(padapter, REG_BCNTCFG, 0x00);
1933 rtl8723au_write16(padapter, REG_TBTT_PROHIBIT, 0xff04);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001934 /* +32767 (~32ms) */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001935 rtl8723au_write16(padapter, REG_TSFTR_SYN_OFFSET, 0x7fff);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001936
1937 /* reset TSF */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001938 rtl8723au_write8(padapter, REG_DUAL_TSF_RST, BIT(0));
Larry Fingerf7c92d22014-03-28 21:37:39 -05001939
1940 /* enable BCN Function */
1941 /* don't enable update TSF (due to TSF update when
1942 beacon/probe rsp are received) */
1943 val8 = DIS_TSF_UDT | EN_BCN_FUNCTION |
1944 EN_TXBCN_RPT | DIS_BCNQ_SUB;
1945 SetBcnCtrlReg23a(padapter, val8, ~val8);
1946 }
1947
Jes Sorensen050abc42014-05-16 10:05:08 +02001948 val8 = rtl8723au_read8(padapter, MSR);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001949 val8 = (val8 & 0xC) | mode;
Jes Sorensenedbfd672014-05-16 10:05:09 +02001950 rtl8723au_write8(padapter, MSR, val8);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001951}
1952
Jes Sorensen3c5660e2014-04-09 23:20:19 +02001953void hw_var_set_macaddr(struct rtw_adapter *padapter, u8 *val)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001954{
1955 u8 idx = 0;
1956 u32 reg_macid;
1957
1958 reg_macid = REG_MACID;
1959
1960 for (idx = 0; idx < 6; idx++)
Jes Sorensenedbfd672014-05-16 10:05:09 +02001961 rtl8723au_write8(padapter, (reg_macid + idx), val[idx]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001962}
1963
Jes Sorensen38dd10b2014-04-09 23:20:20 +02001964void hw_var_set_bssid(struct rtw_adapter *padapter, u8 *val)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001965{
1966 u8 idx = 0;
1967 u32 reg_bssid;
1968
1969 reg_bssid = REG_BSSID;
1970
1971 for (idx = 0; idx < 6; idx++)
Jes Sorensenedbfd672014-05-16 10:05:09 +02001972 rtl8723au_write8(padapter, (reg_bssid + idx), val[idx]);
Larry Fingerf7c92d22014-03-28 21:37:39 -05001973}
1974
Jes Sorensen763b4242014-04-09 23:20:14 +02001975void hw_var_set_correct_tsf(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -05001976{
1977 u64 tsf;
1978 u32 reg_tsftr;
1979 struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
1980 struct mlme_ext_info *pmlmeinfo = &pmlmeext->mlmext_info;
1981
1982 /* tsf = pmlmeext->TSFValue - ((u32)pmlmeext->TSFValue %
1983 (pmlmeinfo->bcn_interval*1024)) - 1024; us */
1984 tsf = pmlmeext->TSFValue -
Jes Sorensenf4337aa2014-04-09 23:21:19 +02001985 do_div(pmlmeext->TSFValue,
1986 (pmlmeinfo->bcn_interval * 1024)) - 1024; /* us */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001987
Jes Sorensen6ec26272014-07-17 22:59:46 +02001988 if (((pmlmeinfo->state & 0x03) == MSR_ADHOC) ||
1989 ((pmlmeinfo->state & 0x03) == MSR_AP)) {
Larry Fingerf7c92d22014-03-28 21:37:39 -05001990 /* pHalData->RegTxPause |= STOP_BCNQ;BIT(6) */
Jes Sorensenedbfd672014-05-16 10:05:09 +02001991 /* rtl8723au_write8(padapter, REG_TXPAUSE,
Jes Sorensen050abc42014-05-16 10:05:08 +02001992 (rtl8723au_read8(Adapter, REG_TXPAUSE)|BIT(6))); */
Larry Fingerf7c92d22014-03-28 21:37:39 -05001993 StopTxBeacon(padapter);
1994 }
1995
1996 reg_tsftr = REG_TSFTR;
1997
1998 /* disable related TSF function */
1999 SetBcnCtrlReg23a(padapter, 0, EN_BCN_FUNCTION);
2000
Jes Sorensenedbfd672014-05-16 10:05:09 +02002001 rtl8723au_write32(padapter, reg_tsftr, tsf);
2002 rtl8723au_write32(padapter, reg_tsftr + 4, tsf >> 32);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002003
2004 /* enable related TSF function */
2005 SetBcnCtrlReg23a(padapter, EN_BCN_FUNCTION, 0);
2006
Jes Sorensen6ec26272014-07-17 22:59:46 +02002007 if (((pmlmeinfo->state & 0x03) == MSR_ADHOC) ||
2008 ((pmlmeinfo->state & 0x03) == MSR_AP))
Larry Fingerf7c92d22014-03-28 21:37:39 -05002009 ResumeTxBeacon(padapter);
2010}
2011
Jes Sorensen763b4242014-04-09 23:20:14 +02002012void hw_var_set_mlme_disconnect(struct rtw_adapter *padapter)
Larry Fingerf7c92d22014-03-28 21:37:39 -05002013{
2014 /* reject all data frames */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002015 rtl8723au_write16(padapter, REG_RXFLTMAP2, 0);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002016
2017 /* reset TSF */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002018 rtl8723au_write8(padapter, REG_DUAL_TSF_RST, BIT(0));
Larry Fingerf7c92d22014-03-28 21:37:39 -05002019
2020 /* disable update TSF */
2021 SetBcnCtrlReg23a(padapter, DIS_TSF_UDT, 0);
2022}
2023
Jes Sorensenea0cd732014-04-09 23:20:22 +02002024void hw_var_set_mlme_join(struct rtw_adapter *padapter, u8 type)
Larry Fingerf7c92d22014-03-28 21:37:39 -05002025{
2026 u8 RetryLimit = 0x30;
2027
2028 struct hal_data_8723a *pHalData = GET_HAL_DATA(padapter);
2029 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
2030
2031 if (type == 0) { /* prepare to join */
2032 u32 v32;
2033
2034 /* enable to rx data frame.Accept all data frame */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002035 /* rtl8723au_write32(padapter, REG_RCR,
Jes Sorensen050abc42014-05-16 10:05:08 +02002036 rtl8723au_read32(padapter, REG_RCR)|RCR_ADF); */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002037 rtl8723au_write16(padapter, REG_RXFLTMAP2, 0xFFFF);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002038
Jes Sorensen050abc42014-05-16 10:05:08 +02002039 v32 = rtl8723au_read32(padapter, REG_RCR);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002040 v32 |= RCR_CBSSID_DATA | RCR_CBSSID_BCN;
Jes Sorensenedbfd672014-05-16 10:05:09 +02002041 rtl8723au_write32(padapter, REG_RCR, v32);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002042
2043 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE) == true)
2044 RetryLimit =
2045 (pHalData->CustomerID == RT_CID_CCX) ? 7 : 48;
2046 else /* Ad-hoc Mode */
2047 RetryLimit = 0x7;
2048 } else if (type == 1) { /* joinbss_event callback when join res < 0 */
2049 /* config RCR to receive different BSSID & not to
2050 receive data frame during linking */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002051 rtl8723au_write16(padapter, REG_RXFLTMAP2, 0);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002052 } else if (type == 2) { /* sta add event callback */
2053 /* enable update TSF */
2054 SetBcnCtrlReg23a(padapter, 0, DIS_TSF_UDT);
2055
2056 if (check_fwstate(pmlmepriv,
2057 WIFI_ADHOC_STATE | WIFI_ADHOC_MASTER_STATE)) {
2058 /* fixed beacon issue for 8191su........... */
Jes Sorensenedbfd672014-05-16 10:05:09 +02002059 rtl8723au_write8(padapter, 0x542, 0x02);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002060 RetryLimit = 0x7;
2061 }
2062 }
2063
Jes Sorensenedbfd672014-05-16 10:05:09 +02002064 rtl8723au_write16(padapter, REG_RL,
2065 RetryLimit << RETRY_LIMIT_SHORT_SHIFT | RetryLimit <<
2066 RETRY_LIMIT_LONG_SHIFT);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002067
Larry Fingerf7c92d22014-03-28 21:37:39 -05002068 switch (type) {
2069 case 0:
2070 /* prepare to join */
Jes Sorensenc814d4d2014-05-25 22:43:42 +02002071 rtl8723a_BT_wifiassociate_notify(padapter, true);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002072 break;
2073 case 1:
2074 /* joinbss_event callback when join res < 0 */
Jes Sorensenc814d4d2014-05-25 22:43:42 +02002075 rtl8723a_BT_wifiassociate_notify(padapter, false);
Larry Fingerf7c92d22014-03-28 21:37:39 -05002076 break;
2077 case 2:
2078 /* sta add event callback */
2079/* BT_WifiMediaStatusNotify(padapter, RT_MEDIA_CONNECT); */
2080 break;
2081 }
Larry Fingerf7c92d22014-03-28 21:37:39 -05002082}