1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright(c) 2009-2014 Realtek Corporation.*/
11 #include "../rtl8723com/fw_common.h"
13 static bool _rtl8723be_check_fw_read_last_h2c(struct ieee80211_hw
*hw
,
16 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
20 val_hmetfr
= rtl_read_byte(rtlpriv
, REG_HMETFR
);
21 if (((val_hmetfr
>> boxnum
) & BIT(0)) == 0)
26 static void _rtl8723be_fill_h2c_command(struct ieee80211_hw
*hw
, u8 element_id
,
27 u32 cmd_len
, u8
*p_cmdbuffer
)
29 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
30 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
32 u16 box_reg
= 0, box_extreg
= 0;
34 bool isfw_read
= false;
36 bool bwrite_sucess
= false;
37 u8 wait_h2c_limmit
= 100;
38 u8 wait_writeh2c_limmit
= 100;
39 u8 boxcontent
[4], boxextcontent
[4];
40 u32 h2c_waitcounter
= 0;
44 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
, "come in\n");
47 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
48 if (rtlhal
->h2c_setinprogress
) {
49 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
50 "H2C set in progress! Wait to set..element_id(%d).\n",
53 while (rtlhal
->h2c_setinprogress
) {
54 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
,
57 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
58 "Wait 100 us (%d times)...\n",
62 if (h2c_waitcounter
> 1000)
64 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
,
67 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
69 rtlhal
->h2c_setinprogress
= true;
70 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
75 while (!bwrite_sucess
) {
76 wait_writeh2c_limmit
--;
77 if (wait_writeh2c_limmit
== 0) {
78 pr_err("Write H2C fail because no trigger for FW INT!\n");
82 boxnum
= rtlhal
->last_hmeboxnum
;
85 box_reg
= REG_HMEBOX_0
;
86 box_extreg
= REG_HMEBOX_EXT_0
;
89 box_reg
= REG_HMEBOX_1
;
90 box_extreg
= REG_HMEBOX_EXT_1
;
93 box_reg
= REG_HMEBOX_2
;
94 box_extreg
= REG_HMEBOX_EXT_2
;
97 box_reg
= REG_HMEBOX_3
;
98 box_extreg
= REG_HMEBOX_EXT_3
;
101 pr_err("switch case %#x not processed\n",
106 isfw_read
= _rtl8723be_check_fw_read_last_h2c(hw
, boxnum
);
109 if (wait_h2c_limmit
== 0) {
110 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
111 "Waiting too long for FW read clear HMEBox(%d)!\n",
118 isfw_read
= _rtl8723be_check_fw_read_last_h2c(hw
,
120 u1b_tmp
= rtl_read_byte(rtlpriv
, 0x130);
121 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
122 "Waiting for FW read clear HMEBox(%d)!!! 0x130 = %2x\n",
127 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
128 "Write H2C register BOX[%d] fail!!!!! Fw do not read.\n",
133 memset(boxcontent
, 0, sizeof(boxcontent
));
134 memset(boxextcontent
, 0, sizeof(boxextcontent
));
135 boxcontent
[0] = element_id
;
136 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
137 "Write element_id box_reg(%4x) = %2x\n",
138 box_reg
, element_id
);
144 /*boxcontent[0] &= ~(BIT(7));*/
145 memcpy((u8
*)(boxcontent
) + 1,
146 p_cmdbuffer
+ buf_index
, cmd_len
);
148 for (idx
= 0; idx
< 4; idx
++) {
149 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
157 /*boxcontent[0] |= (BIT(7));*/
158 memcpy((u8
*)(boxextcontent
),
159 p_cmdbuffer
+ buf_index
+3, cmd_len
-3);
160 memcpy((u8
*)(boxcontent
) + 1,
161 p_cmdbuffer
+ buf_index
, 3);
163 for (idx
= 0; idx
< 4; idx
++) {
164 rtl_write_byte(rtlpriv
, box_extreg
+ idx
,
168 for (idx
= 0; idx
< 4; idx
++) {
169 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
174 pr_err("switch case %#x not processed\n",
179 bwrite_sucess
= true;
181 rtlhal
->last_hmeboxnum
= boxnum
+ 1;
182 if (rtlhal
->last_hmeboxnum
== 4)
183 rtlhal
->last_hmeboxnum
= 0;
185 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
186 "pHalData->last_hmeboxnum = %d\n",
187 rtlhal
->last_hmeboxnum
);
190 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
191 rtlhal
->h2c_setinprogress
= false;
192 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
194 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
, "go out\n");
197 void rtl8723be_fill_h2c_cmd(struct ieee80211_hw
*hw
, u8 element_id
,
198 u32 cmd_len
, u8
*p_cmdbuffer
)
200 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
203 if (!rtlhal
->fw_ready
) {
205 "rtl8723be: error H2C cmd because of Fw download fail!!!\n");
209 memset(tmp_cmdbuf
, 0, 8);
210 memcpy(tmp_cmdbuf
, p_cmdbuffer
, cmd_len
);
211 _rtl8723be_fill_h2c_command(hw
, element_id
, cmd_len
,
216 void rtl8723be_set_fw_pwrmode_cmd(struct ieee80211_hw
*hw
, u8 mode
)
218 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
219 u8 u1_h2c_set_pwrmode
[H2C_PWEMODE_LENGTH
] = { 0 };
220 struct rtl_ps_ctl
*ppsc
= rtl_psc(rtl_priv(hw
));
221 u8 rlbm
, power_state
= 0, byte5
= 0;
222 u8 awake_intvl
; /* DTIM = (awake_intvl - 1) */
223 struct rtl_btc_ops
*btc_ops
= rtlpriv
->btcoexist
.btc_ops
;
224 bool bt_ctrl_lps
= (rtlpriv
->cfg
->ops
->get_btc_status() ?
225 btc_ops
->btc_is_bt_ctrl_lps(rtlpriv
) : false);
226 bool bt_lps_on
= (rtlpriv
->cfg
->ops
->get_btc_status() ?
227 btc_ops
->btc_is_bt_lps_on(rtlpriv
) : false);
230 mode
= (bt_lps_on
? FW_PS_MIN_MODE
: FW_PS_ACTIVE_MODE
);
232 RT_TRACE(rtlpriv
, COMP_POWER
, DBG_DMESG
, "FW LPS mode = %d (coex:%d)\n",
244 case FW_PS_DTIM_MODE
:
246 awake_intvl
= ppsc
->reg_max_lps_awakeintvl
;
247 /* hw->conf.ps_dtim_period or mac->vif->bss_conf.dtim_period
248 * is only used in swlps.
257 if (rtlpriv
->mac80211
.p2p
) {
262 if (mode
== FW_PS_ACTIVE_MODE
) {
264 power_state
= FW_PWR_STATE_ACTIVE
;
267 byte5
= btc_ops
->btc_get_lps_val(rtlpriv
);
268 power_state
= btc_ops
->btc_get_rpwm_val(rtlpriv
);
270 if ((rlbm
== 2) && (byte5
& BIT(4))) {
271 /* Keep awake interval to 1 to prevent from
272 * decreasing coex performance
279 power_state
= FW_PWR_STATE_RF_OFF
;
283 SET_H2CCMD_PWRMODE_PARM_MODE(u1_h2c_set_pwrmode
, ((mode
) ? 1 : 0));
284 SET_H2CCMD_PWRMODE_PARM_RLBM(u1_h2c_set_pwrmode
, rlbm
);
285 SET_H2CCMD_PWRMODE_PARM_SMART_PS(u1_h2c_set_pwrmode
,
286 bt_ctrl_lps
? 0 : ppsc
->smart_ps
);
287 SET_H2CCMD_PWRMODE_PARM_AWAKE_INTERVAL(u1_h2c_set_pwrmode
,
289 SET_H2CCMD_PWRMODE_PARM_ALL_QUEUE_UAPSD(u1_h2c_set_pwrmode
, 0);
290 SET_H2CCMD_PWRMODE_PARM_PWR_STATE(u1_h2c_set_pwrmode
, power_state
);
291 SET_H2CCMD_PWRMODE_PARM_BYTE5(u1_h2c_set_pwrmode
, byte5
);
293 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
294 "rtl92c_set_fw_pwrmode(): u1_h2c_set_pwrmode\n",
295 u1_h2c_set_pwrmode
, H2C_PWEMODE_LENGTH
);
296 if (rtlpriv
->cfg
->ops
->get_btc_status())
297 btc_ops
->btc_record_pwr_mode(rtlpriv
, u1_h2c_set_pwrmode
,
299 rtl8723be_fill_h2c_cmd(hw
, H2C_8723B_SETPWRMODE
, H2C_PWEMODE_LENGTH
,
303 void rtl8723be_set_fw_media_status_rpt_cmd(struct ieee80211_hw
*hw
, u8 mstatus
)
305 u8 parm
[3] = { 0, 0, 0 };
306 /* parm[0]: bit0=0-->Disconnect, bit0=1-->Connect
307 * bit1=0-->update Media Status to MACID
308 * bit1=1-->update Media Status from MACID to MACID_End
309 * parm[1]: MACID, if this is INFRA_STA, MacID = 0
312 SET_H2CCMD_MSRRPT_PARM_OPMODE(parm
, mstatus
);
313 SET_H2CCMD_MSRRPT_PARM_MACID_IND(parm
, 0);
315 rtl8723be_fill_h2c_cmd(hw
, H2C_8723B_MSRRPT
, 3, parm
);
318 #define BEACON_PG 0 /* ->1 */
321 #define PROBERSP_PG 4 /* ->5 */
322 #define QOS_NULL_PG 6
323 #define BT_QOS_NULL_PG 7
325 #define TOTAL_RESERVED_PKT_LEN 1024 /* can be up to 1280 (tx_bndy=245) */
327 static u8 reserved_page_packet
[TOTAL_RESERVED_PKT_LEN
] = {
329 0x80, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF,
330 0xFF, 0xFF, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
331 0xEC, 0x1A, 0x59, 0x0B, 0xAD, 0xD4, 0x20, 0x00,
332 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
333 0x64, 0x00, 0x10, 0x04, 0x00, 0x05, 0x54, 0x65,
334 0x73, 0x74, 0x32, 0x01, 0x08, 0x82, 0x84, 0x0B,
335 0x16, 0x24, 0x30, 0x48, 0x6C, 0x03, 0x01, 0x06,
336 0x06, 0x02, 0x00, 0x00, 0x2A, 0x01, 0x02, 0x32,
337 0x04, 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C,
338 0x09, 0x03, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00,
339 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
340 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
341 0x00, 0x3D, 0x00, 0xDD, 0x07, 0x00, 0xE0, 0x4C,
342 0x02, 0x02, 0x00, 0x00, 0xDD, 0x18, 0x00, 0x50,
343 0xF2, 0x01, 0x01, 0x00, 0x00, 0x50, 0xF2, 0x04,
344 0x01, 0x00, 0x00, 0x50, 0xF2, 0x04, 0x01, 0x00,
347 0x00, 0x50, 0xF2, 0x02, 0x00, 0x00, 0x00, 0x00,
348 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
349 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
350 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
351 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
352 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
353 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
354 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
355 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
356 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
357 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
358 0x10, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
359 0x00, 0x00, 0x00, 0x00, 0x00, 0x81, 0x00, 0x00,
360 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
361 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
362 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
365 0xA4, 0x10, 0x01, 0xC0, 0xEC, 0x1A, 0x59, 0x0B,
366 0xAD, 0xD4, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
367 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
368 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
369 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
370 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
371 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
372 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
373 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
374 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
375 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
376 0x18, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
377 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
378 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
379 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
380 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
383 0x48, 0x01, 0x00, 0x00, 0xEC, 0x1A, 0x59, 0x0B,
384 0xAD, 0xD4, 0x00, 0xE0, 0x4C, 0x02, 0xB1, 0x78,
385 0xEC, 0x1A, 0x59, 0x0B, 0xAD, 0xD4, 0x00, 0x00,
386 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
387 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
388 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
389 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
390 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
391 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
392 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
393 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
394 0x72, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
395 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
396 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
397 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
398 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
400 /* page 4 probe_resp */
401 0x50, 0x00, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10,
402 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
403 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00,
404 0x9E, 0x46, 0x15, 0x32, 0x27, 0xF2, 0x2D, 0x00,
405 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69,
406 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C,
407 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96,
408 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A,
409 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C,
410 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18,
411 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
412 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
413 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
414 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02,
415 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
416 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
418 /* page 5 probe_resp */
419 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
420 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
421 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
422 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
423 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
424 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
425 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
426 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
427 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
428 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
429 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
430 0x1A, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
431 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
432 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
433 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
434 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
436 /* page 6 qos null data */
437 0xC8, 0x01, 0x00, 0x00, 0x84, 0xC9, 0xB2, 0xA7,
438 0xB3, 0x6E, 0x00, 0xE0, 0x4C, 0x02, 0x51, 0x02,
439 0x84, 0xC9, 0xB2, 0xA7, 0xB3, 0x6E, 0x00, 0x00,
440 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
441 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
442 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
443 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
444 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
445 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
446 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
447 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
448 0x1A, 0x00, 0x28, 0x8C, 0x00, 0x12, 0x00, 0x00,
449 0x00, 0x00, 0x80, 0x00, 0x00, 0x01, 0x00, 0x00,
450 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
451 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
452 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
454 /* page 7 BT-qos null data */
455 0xC8, 0x01, 0x00, 0x00, 0x84, 0xC9, 0xB2, 0xA7,
456 0xB3, 0x6E, 0x00, 0xE0, 0x4C, 0x02, 0x51, 0x02,
457 0x84, 0xC9, 0xB2, 0xA7, 0xB3, 0x6E, 0x00, 0x00,
458 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
459 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
460 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
461 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
462 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
463 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
464 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
465 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
466 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
467 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
468 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
469 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
470 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
474 void rtl8723be_set_fw_rsvdpagepkt(struct ieee80211_hw
*hw
,
477 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
478 struct rtl_mac
*mac
= rtl_mac(rtl_priv(hw
));
479 struct sk_buff
*skb
= NULL
;
483 u8 u1rsvdpageloc
[5] = { 0 };
492 /*---------------------------------------------------------
494 *---------------------------------------------------------
496 beacon
= &reserved_page_packet
[BEACON_PG
* 128];
497 SET_80211_HDR_ADDRESS2(beacon
, mac
->mac_addr
);
498 SET_80211_HDR_ADDRESS3(beacon
, mac
->bssid
);
500 /*-------------------------------------------------------
502 *-------------------------------------------------------
504 p_pspoll
= &reserved_page_packet
[PSPOLL_PG
* 128];
505 SET_80211_PS_POLL_AID(p_pspoll
, (mac
->assoc_id
| 0xc000));
506 SET_80211_PS_POLL_BSSID(p_pspoll
, mac
->bssid
);
507 SET_80211_PS_POLL_TA(p_pspoll
, mac
->mac_addr
);
509 SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(u1rsvdpageloc
, PSPOLL_PG
);
511 /*--------------------------------------------------------
513 *--------------------------------------------------------
515 nullfunc
= &reserved_page_packet
[NULL_PG
* 128];
516 SET_80211_HDR_ADDRESS1(nullfunc
, mac
->bssid
);
517 SET_80211_HDR_ADDRESS2(nullfunc
, mac
->mac_addr
);
518 SET_80211_HDR_ADDRESS3(nullfunc
, mac
->bssid
);
520 SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(u1rsvdpageloc
, NULL_PG
);
522 /*---------------------------------------------------------
524 *---------------------------------------------------------
526 p_probersp
= &reserved_page_packet
[PROBERSP_PG
* 128];
527 SET_80211_HDR_ADDRESS1(p_probersp
, mac
->bssid
);
528 SET_80211_HDR_ADDRESS2(p_probersp
, mac
->mac_addr
);
529 SET_80211_HDR_ADDRESS3(p_probersp
, mac
->bssid
);
531 SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(u1rsvdpageloc
, PROBERSP_PG
);
533 /*---------------------------------------------------------
535 *---------------------------------------------------------
537 qosnull
= &reserved_page_packet
[QOS_NULL_PG
* 128];
538 SET_80211_HDR_ADDRESS1(qosnull
, mac
->bssid
);
539 SET_80211_HDR_ADDRESS2(qosnull
, mac
->mac_addr
);
540 SET_80211_HDR_ADDRESS3(qosnull
, mac
->bssid
);
542 SET_H2CCMD_RSVDPAGE_LOC_QOS_NULL_DATA(u1rsvdpageloc
, QOS_NULL_PG
);
544 /*---------------------------------------------------------
546 *---------------------------------------------------------
548 btqosnull
= &reserved_page_packet
[BT_QOS_NULL_PG
* 128];
549 SET_80211_HDR_ADDRESS1(btqosnull
, mac
->bssid
);
550 SET_80211_HDR_ADDRESS2(btqosnull
, mac
->mac_addr
);
551 SET_80211_HDR_ADDRESS3(btqosnull
, mac
->bssid
);
553 SET_H2CCMD_RSVDPAGE_LOC_BT_QOS_NULL_DATA(u1rsvdpageloc
, BT_QOS_NULL_PG
);
555 totalpacketlen
= TOTAL_RESERVED_PKT_LEN
;
557 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_LOUD
,
558 "rtl8723be_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
559 &reserved_page_packet
[0], totalpacketlen
);
560 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
561 "rtl8723be_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
562 u1rsvdpageloc
, sizeof(u1rsvdpageloc
));
564 skb
= dev_alloc_skb(totalpacketlen
);
567 skb_put_data(skb
, &reserved_page_packet
, totalpacketlen
);
569 rtstatus
= rtl_cmd_send_packet(hw
, skb
);
575 RT_TRACE(rtlpriv
, COMP_POWER
, DBG_LOUD
,
576 "Set RSVD page location to Fw.\n");
577 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
, "H2C_RSVDPAGE:\n",
578 u1rsvdpageloc
, sizeof(u1rsvdpageloc
));
579 rtl8723be_fill_h2c_cmd(hw
, H2C_8723B_RSVDPAGE
,
580 sizeof(u1rsvdpageloc
), u1rsvdpageloc
);
582 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_WARNING
,
583 "Set RSVD page location to Fw FAIL!!!!!!.\n");
586 /*Should check FW support p2p or not.*/
587 static void rtl8723be_set_p2p_ctw_period_cmd(struct ieee80211_hw
*hw
,
590 u8 u1_ctwindow_period
[1] = { ctwindow
};
592 rtl8723be_fill_h2c_cmd(hw
, H2C_8723B_P2P_PS_CTW_CMD
, 1,
596 void rtl8723be_set_p2p_ps_offload_cmd(struct ieee80211_hw
*hw
,
599 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
600 struct rtl_ps_ctl
*rtlps
= rtl_psc(rtl_priv(hw
));
601 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
602 struct rtl_p2p_ps_info
*p2pinfo
= &(rtlps
->p2p_ps_info
);
603 struct p2p_ps_offload_t
*p2p_ps_offload
= &rtlhal
->p2p_ps_offload
;
606 u32 start_time
, tsf_low
;
608 switch (p2p_ps_state
) {
610 RT_TRACE(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_DISABLE\n");
611 memset(p2p_ps_offload
, 0, sizeof(*p2p_ps_offload
));
614 RT_TRACE(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_ENABLE\n");
615 /* update CTWindow value. */
616 if (p2pinfo
->ctwindow
> 0) {
617 p2p_ps_offload
->ctwindow_en
= 1;
618 ctwindow
= p2pinfo
->ctwindow
;
619 rtl8723be_set_p2p_ctw_period_cmd(hw
, ctwindow
);
621 /* hw only support 2 set of NoA */
622 for (i
= 0 ; i
< p2pinfo
->noa_num
; i
++) {
623 /* To control the register setting
626 rtl_write_byte(rtlpriv
, 0x5cf, (i
<< 4));
628 p2p_ps_offload
->noa0_en
= 1;
630 p2p_ps_offload
->noa1_en
= 1;
632 /* config P2P NoA Descriptor Register */
633 rtl_write_dword(rtlpriv
, 0x5E0,
634 p2pinfo
->noa_duration
[i
]);
635 rtl_write_dword(rtlpriv
, 0x5E4,
636 p2pinfo
->noa_interval
[i
]);
638 /*Get Current TSF value */
639 tsf_low
= rtl_read_dword(rtlpriv
, REG_TSFTR
);
641 start_time
= p2pinfo
->noa_start_time
[i
];
642 if (p2pinfo
->noa_count_type
[i
] != 1) {
643 while (start_time
<= (tsf_low
+ (50 * 1024))) {
644 start_time
+= p2pinfo
->noa_interval
[i
];
645 if (p2pinfo
->noa_count_type
[i
] != 255)
646 p2pinfo
->noa_count_type
[i
]--;
649 rtl_write_dword(rtlpriv
, 0x5E8, start_time
);
650 rtl_write_dword(rtlpriv
, 0x5EC,
651 p2pinfo
->noa_count_type
[i
]);
654 if ((p2pinfo
->opp_ps
== 1) ||
655 (p2pinfo
->noa_num
> 0)) {
656 /* rst p2p circuit */
657 rtl_write_byte(rtlpriv
, REG_DUAL_TSF_RST
, BIT(4));
659 p2p_ps_offload
->offload_en
= 1;
661 if (P2P_ROLE_GO
== rtlpriv
->mac80211
.p2p
) {
662 p2p_ps_offload
->role
= 1;
663 p2p_ps_offload
->allstasleep
= 0;
665 p2p_ps_offload
->role
= 0;
667 p2p_ps_offload
->discovery
= 0;
671 RT_TRACE(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_SCAN\n");
672 p2p_ps_offload
->discovery
= 1;
674 case P2P_PS_SCAN_DONE
:
675 RT_TRACE(rtlpriv
, COMP_FW
, DBG_LOUD
, "P2P_PS_SCAN_DONE\n");
676 p2p_ps_offload
->discovery
= 0;
677 p2pinfo
->p2p_ps_state
= P2P_PS_ENABLE
;
683 rtl8723be_fill_h2c_cmd(hw
, H2C_8723B_P2P_PS_OFFLOAD
, 1,
684 (u8
*)p2p_ps_offload
);