1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright(c) 2009-2014 Realtek Corporation.*/
14 static void _rtl92ee_enable_fw_download(struct ieee80211_hw
*hw
, bool enable
)
16 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
20 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, 0x05);
22 tmp
= rtl_read_byte(rtlpriv
, REG_MCUFWDL
+ 2);
23 rtl_write_byte(rtlpriv
, REG_MCUFWDL
+ 2, tmp
& 0xf7);
25 tmp
= rtl_read_byte(rtlpriv
, REG_MCUFWDL
);
26 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, tmp
& 0xfe);
30 static void _rtl92ee_write_fw(struct ieee80211_hw
*hw
,
31 enum version_8192e version
,
34 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
35 u8
*bufferptr
= (u8
*)buffer
;
36 u32 pagenums
, remainsize
;
39 rtl_dbg(rtlpriv
, COMP_FW
, DBG_LOUD
, "FW size is %d bytes,\n", size
);
41 rtl_fill_dummy(bufferptr
, &size
);
43 pagenums
= size
/ FW_8192C_PAGE_SIZE
;
44 remainsize
= size
% FW_8192C_PAGE_SIZE
;
47 pr_err("Page numbers should not greater then 8\n");
49 for (page
= 0; page
< pagenums
; page
++) {
50 offset
= page
* FW_8192C_PAGE_SIZE
;
51 rtl_fw_page_write(hw
, page
, (bufferptr
+ offset
),
57 offset
= pagenums
* FW_8192C_PAGE_SIZE
;
59 rtl_fw_page_write(hw
, page
, (bufferptr
+ offset
), remainsize
);
63 static int _rtl92ee_fw_free_to_go(struct ieee80211_hw
*hw
)
65 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
71 value32
= rtl_read_dword(rtlpriv
, REG_MCUFWDL
);
72 } while ((counter
++ < FW_8192C_POLLING_TIMEOUT_COUNT
) &&
73 (!(value32
& FWDL_CHKSUM_RPT
)));
75 if (counter
>= FW_8192C_POLLING_TIMEOUT_COUNT
) {
76 pr_err("chksum report fail! REG_MCUFWDL:0x%08x\n",
80 value32
= rtl_read_dword(rtlpriv
, REG_MCUFWDL
);
81 value32
|= MCUFWDL_RDY
;
82 value32
&= ~WINTINI_RDY
;
83 rtl_write_dword(rtlpriv
, REG_MCUFWDL
, value32
);
85 rtl92ee_firmware_selfreset(hw
);
89 value32
= rtl_read_dword(rtlpriv
, REG_MCUFWDL
);
90 if (value32
& WINTINI_RDY
)
93 udelay(FW_8192C_POLLING_DELAY
*10);
95 } while (counter
++ < FW_8192C_POLLING_TIMEOUT_COUNT
);
97 pr_err("Polling FW ready fail!! REG_MCUFWDL:0x%08x. count = %d\n",
104 int rtl92ee_download_fw(struct ieee80211_hw
*hw
, bool buse_wake_on_wlan_fw
)
106 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
107 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
108 struct rtlwifi_firmware_header
*pfwheader
;
111 enum version_8192e version
= rtlhal
->version
;
113 if (!rtlhal
->pfirmware
)
116 pfwheader
= (struct rtlwifi_firmware_header
*)rtlhal
->pfirmware
;
117 rtlhal
->fw_version
= le16_to_cpu(pfwheader
->version
);
118 rtlhal
->fw_subversion
= pfwheader
->subversion
;
119 pfwdata
= (u8
*)rtlhal
->pfirmware
;
120 fwsize
= rtlhal
->fwsize
;
121 rtl_dbg(rtlpriv
, COMP_FW
, DBG_DMESG
,
122 "normal Firmware SIZE %d\n", fwsize
);
124 if (IS_FW_HEADER_EXIST(pfwheader
)) {
125 rtl_dbg(rtlpriv
, COMP_FW
, DBG_DMESG
,
126 "Firmware Version(%d), Signature(%#x),Size(%d)\n",
127 pfwheader
->version
, pfwheader
->signature
,
128 (int)sizeof(struct rtlwifi_firmware_header
));
130 pfwdata
= pfwdata
+ sizeof(struct rtlwifi_firmware_header
);
131 fwsize
= fwsize
- sizeof(struct rtlwifi_firmware_header
);
133 rtl_dbg(rtlpriv
, COMP_FW
, DBG_DMESG
,
134 "Firmware no Header, Signature(%#x)\n",
135 pfwheader
->signature
);
138 if (rtlhal
->mac_func_enable
) {
139 if (rtl_read_byte(rtlpriv
, REG_MCUFWDL
) & BIT(7)) {
140 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, 0);
141 rtl92ee_firmware_selfreset(hw
);
144 _rtl92ee_enable_fw_download(hw
, true);
145 _rtl92ee_write_fw(hw
, version
, pfwdata
, fwsize
);
146 _rtl92ee_enable_fw_download(hw
, false);
148 return _rtl92ee_fw_free_to_go(hw
);
151 static bool _rtl92ee_check_fw_read_last_h2c(struct ieee80211_hw
*hw
, u8 boxnum
)
153 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
157 val_hmetfr
= rtl_read_byte(rtlpriv
, REG_HMETFR
);
158 if (((val_hmetfr
>> boxnum
) & BIT(0)) == 0)
163 static void _rtl92ee_fill_h2c_command(struct ieee80211_hw
*hw
, u8 element_id
,
164 u32 cmd_len
, u8
*cmdbuffer
)
166 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
167 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
168 struct rtl_ps_ctl
*ppsc
= rtl_psc(rtl_priv(hw
));
170 u16 box_reg
= 0, box_extreg
= 0;
172 bool isfw_read
= false;
174 bool bwrite_sucess
= false;
175 u8 wait_h2c_limmit
= 100;
176 u8 boxcontent
[4], boxextcontent
[4];
177 u32 h2c_waitcounter
= 0;
181 if (ppsc
->dot11_psmode
!= EACTIVE
||
182 ppsc
->inactive_pwrstate
== ERFOFF
) {
183 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
184 "FillH2CCommand8192E(): Return because RF is off!!!\n");
188 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
, "come in\n");
190 /* 1. Prevent race condition in setting H2C cmd.
191 * (copy from MgntActSet_RF_State().)
194 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
195 if (rtlhal
->h2c_setinprogress
) {
196 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
197 "H2C set in progress! Wait to set..element_id(%d).\n",
200 while (rtlhal
->h2c_setinprogress
) {
201 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
,
204 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
205 "Wait 100 us (%d times)...\n",
209 if (h2c_waitcounter
> 1000)
211 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
,
214 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
216 rtlhal
->h2c_setinprogress
= true;
217 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
222 while (!bwrite_sucess
) {
223 /* 2. Find the last BOX number which has been writen. */
224 boxnum
= rtlhal
->last_hmeboxnum
;
227 box_reg
= REG_HMEBOX_0
;
228 box_extreg
= REG_HMEBOX_EXT_0
;
231 box_reg
= REG_HMEBOX_1
;
232 box_extreg
= REG_HMEBOX_EXT_1
;
235 box_reg
= REG_HMEBOX_2
;
236 box_extreg
= REG_HMEBOX_EXT_2
;
239 box_reg
= REG_HMEBOX_3
;
240 box_extreg
= REG_HMEBOX_EXT_3
;
243 rtl_dbg(rtlpriv
, COMP_ERR
, DBG_LOUD
,
244 "switch case %#x not processed\n", boxnum
);
248 /* 3. Check if the box content is empty. */
250 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_CR
);
252 if (u1b_tmp
!= 0xea) {
255 if (rtl_read_byte(rtlpriv
, REG_TXDMA_STATUS
) == 0xea ||
256 rtl_read_byte(rtlpriv
, REG_TXPKT_EMPTY
) == 0xea)
257 rtl_write_byte(rtlpriv
, REG_SYS_CFG1
+ 3, 0xff);
261 wait_h2c_limmit
= 100;
262 isfw_read
= _rtl92ee_check_fw_read_last_h2c(hw
, boxnum
);
265 if (wait_h2c_limmit
== 0) {
266 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
267 "Waiting too long for FW read clear HMEBox(%d)!!!\n",
273 _rtl92ee_check_fw_read_last_h2c(hw
, boxnum
);
274 u1b_tmp
= rtl_read_byte(rtlpriv
, 0x130);
275 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
276 "Waiting for FW read clear HMEBox(%d)!!! 0x130 = %2x\n",
281 /* If Fw has not read the last
282 * H2C cmd, break and give up this H2C.
285 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
286 "Write H2C reg BOX[%d] fail,Fw don't read.\n",
290 /* 4. Fill the H2C cmd into box */
291 memset(boxcontent
, 0, sizeof(boxcontent
));
292 memset(boxextcontent
, 0, sizeof(boxextcontent
));
293 boxcontent
[0] = element_id
;
294 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
295 "Write element_id box_reg(%4x) = %2x\n",
296 box_reg
, element_id
);
302 /*boxcontent[0] &= ~(BIT(7));*/
303 memcpy((u8
*)(boxcontent
) + 1,
304 cmdbuffer
+ buf_index
, cmd_len
);
306 for (idx
= 0; idx
< 4; idx
++) {
307 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
315 /*boxcontent[0] |= (BIT(7));*/
316 memcpy((u8
*)(boxextcontent
),
317 cmdbuffer
+ buf_index
+3, cmd_len
-3);
318 memcpy((u8
*)(boxcontent
) + 1,
319 cmdbuffer
+ buf_index
, 3);
321 for (idx
= 0; idx
< 4; idx
++) {
322 rtl_write_byte(rtlpriv
, box_extreg
+ idx
,
326 for (idx
= 0; idx
< 4; idx
++) {
327 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
332 rtl_dbg(rtlpriv
, COMP_ERR
, DBG_LOUD
,
333 "switch case %#x not processed\n", cmd_len
);
337 bwrite_sucess
= true;
339 rtlhal
->last_hmeboxnum
= boxnum
+ 1;
340 if (rtlhal
->last_hmeboxnum
== 4)
341 rtlhal
->last_hmeboxnum
= 0;
343 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
,
344 "pHalData->last_hmeboxnum = %d\n",
345 rtlhal
->last_hmeboxnum
);
348 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
349 rtlhal
->h2c_setinprogress
= false;
350 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
352 rtl_dbg(rtlpriv
, COMP_CMD
, DBG_LOUD
, "go out\n");
355 void rtl92ee_fill_h2c_cmd(struct ieee80211_hw
*hw
,
356 u8 element_id
, u32 cmd_len
, u8
*cmdbuffer
)
358 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
361 if (!rtlhal
->fw_ready
) {
363 "rtl8192ee: error H2C cmd because of Fw download fail!!!\n");
367 memset(tmp_cmdbuf
, 0, 8);
368 memcpy(tmp_cmdbuf
, cmdbuffer
, cmd_len
);
369 _rtl92ee_fill_h2c_command(hw
, element_id
, cmd_len
, (u8
*)&tmp_cmdbuf
);
372 void rtl92ee_firmware_selfreset(struct ieee80211_hw
*hw
)
375 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
377 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_RSV_CTRL
+ 1);
378 rtl_write_byte(rtlpriv
, REG_RSV_CTRL
+ 1, (u1b_tmp
& (~BIT(0))));
380 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1);
381 rtl_write_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1, (u1b_tmp
& (~BIT(2))));
385 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_RSV_CTRL
+ 1);
386 rtl_write_byte(rtlpriv
, REG_RSV_CTRL
+ 1, (u1b_tmp
| BIT(0)));
388 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1);
389 rtl_write_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1, (u1b_tmp
| BIT(2)));
391 rtl_dbg(rtlpriv
, COMP_INIT
, DBG_LOUD
,
392 " _8051Reset92E(): 8051 reset success .\n");
395 void rtl92ee_set_fw_pwrmode_cmd(struct ieee80211_hw
*hw
, u8 mode
)
397 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
398 u8 u1_h2c_set_pwrmode
[H2C_92E_PWEMODE_LENGTH
] = { 0 };
399 struct rtl_ps_ctl
*ppsc
= rtl_psc(rtl_priv(hw
));
400 u8 rlbm
, power_state
= 0, byte5
= 0;
401 u8 awake_intvl
; /* DTIM = (awake_intvl - 1) */
402 struct rtl_btc_ops
*btc_ops
= rtlpriv
->btcoexist
.btc_ops
;
403 bool bt_ctrl_lps
= (rtlpriv
->cfg
->ops
->get_btc_status() ?
404 btc_ops
->btc_is_bt_ctrl_lps(rtlpriv
) : false);
405 bool bt_lps_on
= (rtlpriv
->cfg
->ops
->get_btc_status() ?
406 btc_ops
->btc_is_bt_lps_on(rtlpriv
) : false);
409 mode
= (bt_lps_on
? FW_PS_MIN_MODE
: FW_PS_ACTIVE_MODE
);
411 rtl_dbg(rtlpriv
, COMP_POWER
, DBG_DMESG
, "FW LPS mode = %d (coex:%d)\n",
423 case FW_PS_DTIM_MODE
:
425 awake_intvl
= ppsc
->reg_max_lps_awakeintvl
;
426 /* hw->conf.ps_dtim_period or mac->vif->bss_conf.dtim_period
427 * is only used in swlps.
436 if (rtlpriv
->mac80211
.p2p
) {
441 if (mode
== FW_PS_ACTIVE_MODE
) {
443 power_state
= FW_PWR_STATE_ACTIVE
;
446 byte5
= btc_ops
->btc_get_lps_val(rtlpriv
);
447 power_state
= btc_ops
->btc_get_rpwm_val(rtlpriv
);
449 if ((rlbm
== 2) && (byte5
& BIT(4))) {
450 /* Keep awake interval to 1 to prevent from
451 * decreasing coex performance
458 power_state
= FW_PWR_STATE_RF_OFF
;
462 SET_H2CCMD_PWRMODE_PARM_MODE(u1_h2c_set_pwrmode
, ((mode
) ? 1 : 0));
463 SET_H2CCMD_PWRMODE_PARM_RLBM(u1_h2c_set_pwrmode
, rlbm
);
464 SET_H2CCMD_PWRMODE_PARM_SMART_PS(u1_h2c_set_pwrmode
,
466 ((rtlpriv
->mac80211
.p2p
) ?
467 ppsc
->smart_ps
: 1));
468 SET_H2CCMD_PWRMODE_PARM_AWAKE_INTERVAL(u1_h2c_set_pwrmode
,
470 SET_H2CCMD_PWRMODE_PARM_ALL_QUEUE_UAPSD(u1_h2c_set_pwrmode
, 0);
471 SET_H2CCMD_PWRMODE_PARM_PWR_STATE(u1_h2c_set_pwrmode
, power_state
);
472 SET_H2CCMD_PWRMODE_PARM_BYTE5(u1_h2c_set_pwrmode
, byte5
);
474 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
475 "rtl92c_set_fw_pwrmode(): u1_h2c_set_pwrmode\n",
476 u1_h2c_set_pwrmode
, H2C_92E_PWEMODE_LENGTH
);
477 if (rtlpriv
->cfg
->ops
->get_btc_status())
478 btc_ops
->btc_record_pwr_mode(rtlpriv
, u1_h2c_set_pwrmode
,
479 H2C_92E_PWEMODE_LENGTH
);
480 rtl92ee_fill_h2c_cmd(hw
, H2C_92E_SETPWRMODE
, H2C_92E_PWEMODE_LENGTH
,
484 void rtl92ee_set_fw_media_status_rpt_cmd(struct ieee80211_hw
*hw
, u8 mstatus
)
486 u8 parm
[3] = { 0 , 0 , 0 };
487 /* parm[0]: bit0=0-->Disconnect, bit0=1-->Connect
488 * bit1=0-->update Media Status to MACID
489 * bit1=1-->update Media Status from MACID to MACID_End
490 * parm[1]: MACID, if this is INFRA_STA, MacID = 0
494 SET_H2CCMD_MSRRPT_PARM_OPMODE(parm
, mstatus
);
495 SET_H2CCMD_MSRRPT_PARM_MACID_IND(parm
, 0);
497 rtl92ee_fill_h2c_cmd(hw
, H2C_92E_MSRRPT
, 3, parm
);
500 #define BEACON_PG 0 /* ->1 */
503 #define PROBERSP_PG 4 /* ->5 */
504 #define QOS_NULL_PG 6
505 #define BT_QOS_NULL_PG 7
507 #define TOTAL_RESERVED_PKT_LEN 1024
509 static u8 reserved_page_packet
[TOTAL_RESERVED_PKT_LEN
] = {
511 0x80, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF,
512 0xFF, 0xFF, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
513 0xEC, 0x1A, 0x59, 0x0B, 0xAD, 0xD4, 0x20, 0x00,
514 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
515 0x64, 0x00, 0x10, 0x04, 0x00, 0x05, 0x54, 0x65,
516 0x73, 0x74, 0x32, 0x01, 0x08, 0x82, 0x84, 0x0B,
517 0x16, 0x24, 0x30, 0x48, 0x6C, 0x03, 0x01, 0x06,
518 0x06, 0x02, 0x00, 0x00, 0x2A, 0x01, 0x02, 0x32,
519 0x04, 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C,
520 0x09, 0x03, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
521 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
522 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
523 0x00, 0x3D, 0x00, 0xDD, 0x07, 0x00, 0xE0, 0x4C,
524 0x02, 0x02, 0x00, 0x00, 0xDD, 0x18, 0x00, 0x50,
525 0xF2, 0x01, 0x01, 0x00, 0x00, 0x50, 0xF2, 0x04,
526 0x01, 0x00, 0x00, 0x50, 0xF2, 0x04, 0x01, 0x00,
529 0x00, 0x50, 0xF2, 0x02, 0x00, 0x00, 0x00, 0x00,
530 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
531 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
532 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
533 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
534 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
535 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
536 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
537 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
538 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
539 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
540 0x10, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
541 0x00, 0x00, 0x00, 0x00, 0x00, 0x81, 0x00, 0x00,
542 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
543 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
544 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
547 0xA4, 0x10, 0x01, 0xC0, 0xEC, 0x1A, 0x59, 0x0B,
548 0xAD, 0xD4, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
549 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
550 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
551 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
552 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
553 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
554 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
555 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
556 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
557 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
558 0x18, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
559 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
560 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
561 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
562 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
565 0x48, 0x01, 0x00, 0x00, 0xEC, 0x1A, 0x59, 0x0B,
566 0xAD, 0xD4, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
567 0xEC, 0x1A, 0x59, 0x0B, 0xAD, 0xD4, 0x00, 0x00,
568 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
569 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
570 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
571 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
572 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
573 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
574 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
575 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
576 0x72, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
577 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
578 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
579 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
580 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
582 /* page 4 probe_resp */
583 0x50, 0x00, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10,
584 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
585 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00,
586 0x9E, 0x46, 0x15, 0x32, 0x27, 0xF2, 0x2D, 0x00,
587 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69,
588 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C,
589 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96,
590 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A,
591 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C,
592 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18,
593 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
594 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
595 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
596 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02,
597 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
598 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
600 /* page 5 probe_resp */
601 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
602 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
603 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
604 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
605 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
606 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
607 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
608 0x1A, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
609 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
610 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
611 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
612 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
614 /* page 6 qos null data */
615 0xC8, 0x01, 0x00, 0x00, 0x84, 0xC9, 0xB2, 0xA7,
616 0xB3, 0x6E, 0x00, 0xE0, 0x4C, 0x02, 0x51, 0x02,
617 0x84, 0xC9, 0xB2, 0xA7, 0xB3, 0x6E, 0x00, 0x00,
618 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
619 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
620 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
621 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
622 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
623 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
624 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
625 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
626 0x1A, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
627 0x00, 0x00, 0x80, 0x00, 0x00, 0x01, 0x00, 0x00,
628 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
629 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
630 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
632 /* page 7 BT-qos null data */
633 0xC8, 0x01, 0x00, 0x00, 0x84, 0xC9, 0xB2, 0xA7,
634 0xB3, 0x6E, 0x00, 0xE0, 0x4C, 0x02, 0x51, 0x02,
635 0x84, 0xC9, 0xB2, 0xA7, 0xB3, 0x6E, 0x00, 0x00,
636 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
637 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
638 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
639 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
640 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
641 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
642 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
643 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
644 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
645 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
646 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
647 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
648 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
649 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
650 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
651 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
652 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
655 void rtl92ee_set_fw_rsvdpagepkt(struct ieee80211_hw
*hw
, bool b_dl_finished
)
657 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
658 struct rtl_mac
*mac
= rtl_mac(rtl_priv(hw
));
659 struct sk_buff
*skb
= NULL
;
662 u8 u1rsvdpageloc
[5] = { 0 };
671 /*---------------------------------------------------------
673 *---------------------------------------------------------
675 beacon
= &reserved_page_packet
[BEACON_PG
* 128];
676 SET_80211_HDR_ADDRESS2(beacon
, mac
->mac_addr
);
677 SET_80211_HDR_ADDRESS3(beacon
, mac
->bssid
);
679 /*-------------------------------------------------------
681 *--------------------------------------------------------
683 p_pspoll
= &reserved_page_packet
[PSPOLL_PG
* 128];
684 SET_80211_PS_POLL_AID(p_pspoll
, (mac
->assoc_id
| 0xc000));
685 SET_80211_PS_POLL_BSSID(p_pspoll
, mac
->bssid
);
686 SET_80211_PS_POLL_TA(p_pspoll
, mac
->mac_addr
);
688 SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(u1rsvdpageloc
, PSPOLL_PG
);
690 /*--------------------------------------------------------
692 *---------------------------------------------------------
694 nullfunc
= &reserved_page_packet
[NULL_PG
* 128];
695 SET_80211_HDR_ADDRESS1(nullfunc
, mac
->bssid
);
696 SET_80211_HDR_ADDRESS2(nullfunc
, mac
->mac_addr
);
697 SET_80211_HDR_ADDRESS3(nullfunc
, mac
->bssid
);
699 SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(u1rsvdpageloc
, NULL_PG
);
701 /*---------------------------------------------------------
703 *----------------------------------------------------------
705 p_probersp
= &reserved_page_packet
[PROBERSP_PG
* 128];
706 SET_80211_HDR_ADDRESS1(p_probersp
, mac
->bssid
);
707 SET_80211_HDR_ADDRESS2(p_probersp
, mac
->mac_addr
);
708 SET_80211_HDR_ADDRESS3(p_probersp
, mac
->bssid
);
710 SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(u1rsvdpageloc
, PROBERSP_PG
);
712 /*---------------------------------------------------------
714 *----------------------------------------------------------
716 qosnull
= &reserved_page_packet
[QOS_NULL_PG
* 128];
717 SET_80211_HDR_ADDRESS1(qosnull
, mac
->bssid
);
718 SET_80211_HDR_ADDRESS2(qosnull
, mac
->mac_addr
);
719 SET_80211_HDR_ADDRESS3(qosnull
, mac
->bssid
);
721 SET_H2CCMD_RSVDPAGE_LOC_QOS_NULL_DATA(u1rsvdpageloc
, QOS_NULL_PG
);
723 /*---------------------------------------------------------
724 * (6) BT QoS null data
725 *----------------------------------------------------------
727 btqosnull
= &reserved_page_packet
[BT_QOS_NULL_PG
* 128];
728 SET_80211_HDR_ADDRESS1(btqosnull
, mac
->bssid
);
729 SET_80211_HDR_ADDRESS2(btqosnull
, mac
->mac_addr
);
730 SET_80211_HDR_ADDRESS3(btqosnull
, mac
->bssid
);
732 SET_H2CCMD_RSVDPAGE_LOC_BT_QOS_NULL_DATA(u1rsvdpageloc
, BT_QOS_NULL_PG
);
734 totalpacketlen
= TOTAL_RESERVED_PKT_LEN
;
736 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_LOUD
,
737 "rtl92ee_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
738 &reserved_page_packet
[0], totalpacketlen
);
739 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_LOUD
,
740 "rtl92ee_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
743 skb
= dev_alloc_skb(totalpacketlen
);
746 skb_put_data(skb
, &reserved_page_packet
, totalpacketlen
);
748 rtstatus
= rtl_cmd_send_packet(hw
, skb
);
753 rtl_dbg(rtlpriv
, COMP_POWER
, DBG_LOUD
,
754 "Set RSVD page location to Fw.\n");
755 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_LOUD
,
756 "H2C_RSVDPAGE:\n", u1rsvdpageloc
, 3);
757 rtl92ee_fill_h2c_cmd(hw
, H2C_92E_RSVDPAGE
,
758 sizeof(u1rsvdpageloc
), u1rsvdpageloc
);
760 rtl_dbg(rtlpriv
, COMP_ERR
, DBG_WARNING
,
761 "Set RSVD page location to Fw FAIL!!!!!!.\n");
765 /*Shoud check FW support p2p or not.*/
766 static void rtl92ee_set_p2p_ctw_period_cmd(struct ieee80211_hw
*hw
, u8 ctwindow
)
768 u8 u1_ctwindow_period
[1] = {ctwindow
};
770 rtl92ee_fill_h2c_cmd(hw
, H2C_92E_P2P_PS_CTW_CMD
, 1, u1_ctwindow_period
);
773 void rtl92ee_set_p2p_ps_offload_cmd(struct ieee80211_hw
*hw
, u8 p2p_ps_state
)
775 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
776 struct rtl_ps_ctl
*rtlps
= rtl_psc(rtl_priv(hw
));
777 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
778 struct rtl_p2p_ps_info
*p2pinfo
= &rtlps
->p2p_ps_info
;
779 struct p2p_ps_offload_t
*p2p_ps_offload
= &rtlhal
->p2p_ps_offload
;
782 u32 start_time
, tsf_low
;
784 switch (p2p_ps_state
) {
786 rtl_dbg(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_DISABLE\n");
787 memset(p2p_ps_offload
, 0, sizeof(*p2p_ps_offload
));
790 rtl_dbg(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_ENABLE\n");
791 /* update CTWindow value. */
792 if (p2pinfo
->ctwindow
> 0) {
793 p2p_ps_offload
->ctwindow_en
= 1;
794 ctwindow
= p2pinfo
->ctwindow
;
795 rtl92ee_set_p2p_ctw_period_cmd(hw
, ctwindow
);
797 /* hw only support 2 set of NoA */
798 for (i
= 0 ; i
< p2pinfo
->noa_num
; i
++) {
799 /* To control the register setting for which NOA*/
800 rtl_write_byte(rtlpriv
, 0x5cf, (i
<< 4));
802 p2p_ps_offload
->noa0_en
= 1;
804 p2p_ps_offload
->noa1_en
= 1;
805 /* config P2P NoA Descriptor Register */
806 rtl_write_dword(rtlpriv
, 0x5E0,
807 p2pinfo
->noa_duration
[i
]);
808 rtl_write_dword(rtlpriv
, 0x5E4,
809 p2pinfo
->noa_interval
[i
]);
811 /*Get Current TSF value */
812 tsf_low
= rtl_read_dword(rtlpriv
, REG_TSFTR
);
814 start_time
= p2pinfo
->noa_start_time
[i
];
815 if (p2pinfo
->noa_count_type
[i
] != 1) {
816 while (start_time
<= (tsf_low
+ (50 * 1024))) {
817 start_time
+= p2pinfo
->noa_interval
[i
];
818 if (p2pinfo
->noa_count_type
[i
] != 255)
819 p2pinfo
->noa_count_type
[i
]--;
822 rtl_write_dword(rtlpriv
, 0x5E8, start_time
);
823 rtl_write_dword(rtlpriv
, 0x5EC,
824 p2pinfo
->noa_count_type
[i
]);
826 if ((p2pinfo
->opp_ps
== 1) || (p2pinfo
->noa_num
> 0)) {
827 /* rst p2p circuit */
828 rtl_write_byte(rtlpriv
, REG_DUAL_TSF_RST
, BIT(4));
829 p2p_ps_offload
->offload_en
= 1;
831 if (P2P_ROLE_GO
== rtlpriv
->mac80211
.p2p
) {
832 p2p_ps_offload
->role
= 1;
833 p2p_ps_offload
->allstasleep
= 0;
835 p2p_ps_offload
->role
= 0;
837 p2p_ps_offload
->discovery
= 0;
841 rtl_dbg(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_SCAN\n");
842 p2p_ps_offload
->discovery
= 1;
844 case P2P_PS_SCAN_DONE
:
845 rtl_dbg(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_SCAN_DONE\n");
846 p2p_ps_offload
->discovery
= 0;
847 p2pinfo
->p2p_ps_state
= P2P_PS_ENABLE
;
852 rtl92ee_fill_h2c_cmd(hw
, H2C_92E_P2P_PS_OFFLOAD
, 1,
853 (u8
*)p2p_ps_offload
);
856 void rtl92ee_c2h_ra_report_handler(struct ieee80211_hw
*hw
,
857 u8
*cmd_buf
, u8 cmd_len
)
859 u8 rate
= cmd_buf
[0] & 0x3F;
860 bool collision_state
= cmd_buf
[3] & BIT(0);
862 rtl92ee_dm_dynamic_arfb_select(hw
, rate
, collision_state
);