1 /******************************************************************************
3 * Copyright(c) 2009-2010 Realtek Corporation.
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.
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
14 * You should have received a copy of the GNU General Public License along with
15 * this program; if not, write to the Free Software Foundation, Inc.,
16 * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA
18 * The full GNU General Public License is included in this distribution in the
19 * file called LICENSE.
21 * Contact Information:
22 * wlanfae <wlanfae@realtek.com>
23 * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park,
24 * Hsinchu 300, Taiwan.
26 * Larry Finger <Larry.Finger@lwfinger.net>
28 *****************************************************************************/
38 static bool _rtl92d_is_fw_downloaded(struct rtl_priv
*rtlpriv
)
40 return (rtl_read_dword(rtlpriv
, REG_MCUFWDL
) & MCUFWDL_RDY
) ?
44 static void _rtl92d_enable_fw_download(struct ieee80211_hw
*hw
, bool enable
)
46 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
50 tmp
= rtl_read_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1);
51 rtl_write_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1, tmp
| 0x04);
52 tmp
= rtl_read_byte(rtlpriv
, REG_MCUFWDL
);
53 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, tmp
| 0x01);
54 tmp
= rtl_read_byte(rtlpriv
, REG_MCUFWDL
+ 2);
55 rtl_write_byte(rtlpriv
, REG_MCUFWDL
+ 2, tmp
& 0xf7);
57 tmp
= rtl_read_byte(rtlpriv
, REG_MCUFWDL
);
58 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, tmp
& 0xfe);
59 /* Reserved for fw extension.
60 * 0x81[7] is used for mac0 status ,
61 * so don't write this reg here
62 * rtl_write_byte(rtlpriv, REG_MCUFWDL + 1, 0x00);*/
66 static void _rtl92d_fw_block_write(struct ieee80211_hw
*hw
,
67 const u8
*buffer
, u32 size
)
69 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
70 u32 blocksize
= sizeof(u32
);
71 u8
*bufferptr
= (u8
*) buffer
;
72 u32
*pu4BytePtr
= (u32
*) buffer
;
73 u32 i
, offset
, blockCount
, remainSize
;
75 blockCount
= size
/ blocksize
;
76 remainSize
= size
% blocksize
;
77 for (i
= 0; i
< blockCount
; i
++) {
78 offset
= i
* blocksize
;
79 rtl_write_dword(rtlpriv
, (FW_8192D_START_ADDRESS
+ offset
),
83 offset
= blockCount
* blocksize
;
85 for (i
= 0; i
< remainSize
; i
++) {
86 rtl_write_byte(rtlpriv
, (FW_8192D_START_ADDRESS
+
87 offset
+ i
), *(bufferptr
+ i
));
92 static void _rtl92d_fw_page_write(struct ieee80211_hw
*hw
,
93 u32 page
, const u8
*buffer
, u32 size
)
95 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
97 u8 u8page
= (u8
) (page
& 0x07);
99 value8
= (rtl_read_byte(rtlpriv
, REG_MCUFWDL
+ 2) & 0xF8) | u8page
;
100 rtl_write_byte(rtlpriv
, (REG_MCUFWDL
+ 2), value8
);
101 _rtl92d_fw_block_write(hw
, buffer
, size
);
104 static void _rtl92d_fill_dummy(u8
*pfwbuf
, u32
*pfwlen
)
107 u8 remain
= (u8
) (fwlen
% 4);
109 remain
= (remain
== 0) ? 0 : (4 - remain
);
118 static void _rtl92d_write_fw(struct ieee80211_hw
*hw
,
119 enum version_8192d version
, u8
*buffer
, u32 size
)
121 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
122 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
123 u8
*bufferPtr
= (u8
*) buffer
;
124 u32 pagenums
, remainSize
;
127 RT_TRACE(rtlpriv
, COMP_FW
, DBG_TRACE
, ("FW size is %d bytes,\n", size
));
128 if (rtlhal
->hw_type
== HARDWARE_TYPE_RTL8192DE
)
129 _rtl92d_fill_dummy(bufferPtr
, &size
);
130 pagenums
= size
/ FW_8192D_PAGE_SIZE
;
131 remainSize
= size
% FW_8192D_PAGE_SIZE
;
133 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
134 ("Page numbers should not greater then 8\n"));
136 for (page
= 0; page
< pagenums
; page
++) {
137 offset
= page
* FW_8192D_PAGE_SIZE
;
138 _rtl92d_fw_page_write(hw
, page
, (bufferPtr
+ offset
),
142 offset
= pagenums
* FW_8192D_PAGE_SIZE
;
144 _rtl92d_fw_page_write(hw
, page
, (bufferPtr
+ offset
),
149 static int _rtl92d_fw_free_to_go(struct ieee80211_hw
*hw
)
151 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
156 value32
= rtl_read_dword(rtlpriv
, REG_MCUFWDL
);
157 } while ((counter
++ < FW_8192D_POLLING_TIMEOUT_COUNT
) &&
158 (!(value32
& FWDL_ChkSum_rpt
)));
159 if (counter
>= FW_8192D_POLLING_TIMEOUT_COUNT
) {
160 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
161 ("chksum report faill ! REG_MCUFWDL:0x%08x .\n",
165 RT_TRACE(rtlpriv
, COMP_FW
, DBG_TRACE
,
166 ("Checksum report OK ! REG_MCUFWDL:0x%08x .\n", value32
));
167 value32
= rtl_read_dword(rtlpriv
, REG_MCUFWDL
);
168 value32
|= MCUFWDL_RDY
;
169 rtl_write_dword(rtlpriv
, REG_MCUFWDL
, value32
);
173 void rtl92d_firmware_selfreset(struct ieee80211_hw
*hw
)
175 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
179 /* Set (REG_HMETFR + 3) to 0x20 is reset 8051 */
180 rtl_write_byte(rtlpriv
, REG_HMETFR
+ 3, 0x20);
181 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1);
182 while (u1b_tmp
& BIT(2)) {
187 u1b_tmp
= rtl_read_byte(rtlpriv
, REG_SYS_FUNC_EN
+ 1);
189 RT_ASSERT((delay
> 0), ("8051 reset failed!\n"));
190 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
191 ("=====> 8051 reset success (%d) .\n", delay
));
194 static int _rtl92d_fw_init(struct ieee80211_hw
*hw
)
196 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
197 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
200 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
, ("FW already have download\n"));
201 /* polling for FW ready */
204 if (rtlhal
->interfaceindex
== 0) {
205 if (rtl_read_byte(rtlpriv
, FW_MAC0_READY
) &
207 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
208 ("Polling FW ready success!! "
209 "REG_MCUFWDL: 0x%x .\n",
210 rtl_read_byte(rtlpriv
,
216 if (rtl_read_byte(rtlpriv
, FW_MAC1_READY
) &
218 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
219 ("Polling FW ready success!! "
220 "REG_MCUFWDL: 0x%x .\n",
221 rtl_read_byte(rtlpriv
,
227 } while (counter
++ < POLLING_READY_TIMEOUT_COUNT
);
229 if (rtlhal
->interfaceindex
== 0) {
230 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
231 ("Polling FW ready fail!! MAC0 FW init not ready: "
233 rtl_read_byte(rtlpriv
, FW_MAC0_READY
)));
235 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
236 ("Polling FW ready fail!! MAC1 FW init not ready: "
238 rtl_read_byte(rtlpriv
, FW_MAC1_READY
)));
240 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
241 ("Polling FW ready fail!! REG_MCUFWDL:0x%08ul .\n",
242 rtl_read_dword(rtlpriv
, REG_MCUFWDL
)));
246 int rtl92d_download_fw(struct ieee80211_hw
*hw
)
248 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
249 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
254 enum version_8192d version
= rtlhal
->version
;
257 bool fw_downloaded
= false, fwdl_in_process
= false;
260 if (rtlpriv
->max_fw_size
== 0 || !rtlhal
->pfirmware
)
262 fwsize
= rtlhal
->fwsize
;
263 pfwheader
= (u8
*) rtlhal
->pfirmware
;
264 pfwdata
= (u8
*) rtlhal
->pfirmware
;
265 rtlhal
->fw_version
= (u16
) GET_FIRMWARE_HDR_VERSION(pfwheader
);
266 rtlhal
->fw_subversion
= (u16
) GET_FIRMWARE_HDR_SUB_VER(pfwheader
);
267 RT_TRACE(rtlpriv
, COMP_INIT
, DBG_LOUD
, (" FirmwareVersion(%d),"
268 "FirmwareSubVersion(%d), Signature(%#x)\n",
269 rtlhal
->fw_version
, rtlhal
->fw_subversion
,
270 GET_FIRMWARE_HDR_SIGNATURE(pfwheader
)));
271 if (IS_FW_HEADER_EXIST(pfwheader
)) {
272 RT_TRACE(rtlpriv
, COMP_INIT
, DBG_LOUD
,
273 ("Shift 32 bytes for FW header!!\n"));
274 pfwdata
= pfwdata
+ 32;
275 fwsize
= fwsize
- 32;
278 spin_lock_irqsave(&globalmutex_for_fwdownload
, flags
);
279 fw_downloaded
= _rtl92d_is_fw_downloaded(rtlpriv
);
280 if ((rtl_read_byte(rtlpriv
, 0x1f) & BIT(5)) == BIT(5))
281 fwdl_in_process
= true;
283 fwdl_in_process
= false;
285 spin_unlock_irqrestore(&globalmutex_for_fwdownload
, flags
);
287 } else if (fwdl_in_process
) {
288 spin_unlock_irqrestore(&globalmutex_for_fwdownload
, flags
);
289 for (count
= 0; count
< 5000; count
++) {
291 spin_lock_irqsave(&globalmutex_for_fwdownload
, flags
);
292 fw_downloaded
= _rtl92d_is_fw_downloaded(rtlpriv
);
293 if ((rtl_read_byte(rtlpriv
, 0x1f) & BIT(5)) == BIT(5))
294 fwdl_in_process
= true;
296 fwdl_in_process
= false;
297 spin_unlock_irqrestore(&globalmutex_for_fwdownload
,
301 else if (!fwdl_in_process
)
304 RT_TRACE(rtlpriv
, COMP_FW
, DBG_DMESG
,
305 ("Wait for another mac "
308 spin_lock_irqsave(&globalmutex_for_fwdownload
, flags
);
309 value
= rtl_read_byte(rtlpriv
, 0x1f);
311 rtl_write_byte(rtlpriv
, 0x1f, value
);
312 spin_unlock_irqrestore(&globalmutex_for_fwdownload
, flags
);
314 value
= rtl_read_byte(rtlpriv
, 0x1f);
316 rtl_write_byte(rtlpriv
, 0x1f, value
);
317 spin_unlock_irqrestore(&globalmutex_for_fwdownload
, flags
);
320 /* If 8051 is running in RAM code, driver should
321 * inform Fw to reset by itself, or it will cause
322 * download Fw fail.*/
324 if (rtl_read_byte(rtlpriv
, REG_MCUFWDL
) & BIT(7)) {
325 rtl92d_firmware_selfreset(hw
);
326 rtl_write_byte(rtlpriv
, REG_MCUFWDL
, 0x00);
328 _rtl92d_enable_fw_download(hw
, true);
329 _rtl92d_write_fw(hw
, version
, pfwdata
, fwsize
);
330 _rtl92d_enable_fw_download(hw
, false);
331 spin_lock_irqsave(&globalmutex_for_fwdownload
, flags
);
332 err
= _rtl92d_fw_free_to_go(hw
);
333 /* download fw over,clear 0x1f[5] */
334 value
= rtl_read_byte(rtlpriv
, 0x1f);
336 rtl_write_byte(rtlpriv
, 0x1f, value
);
337 spin_unlock_irqrestore(&globalmutex_for_fwdownload
, flags
);
339 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
340 ("fw is not ready to run!\n"));
343 RT_TRACE(rtlpriv
, COMP_FW
, DBG_TRACE
,
344 ("fw is ready to run!\n"));
347 err
= _rtl92d_fw_init(hw
);
351 static bool _rtl92d_check_fw_read_last_h2c(struct ieee80211_hw
*hw
, u8 boxnum
)
353 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
357 val_hmetfr
= rtl_read_byte(rtlpriv
, REG_HMETFR
);
358 if (((val_hmetfr
>> boxnum
) & BIT(0)) == 0)
363 static void _rtl92d_fill_h2c_command(struct ieee80211_hw
*hw
,
364 u8 element_id
, u32 cmd_len
, u8
*cmdbuffer
)
366 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
367 struct rtl_hal
*rtlhal
= rtl_hal(rtl_priv(hw
));
368 struct rtl_ps_ctl
*ppsc
= rtl_psc(rtl_priv(hw
));
370 u16 box_reg
= 0, box_extreg
= 0;
372 bool isfw_read
= false;
374 bool bwrite_sucess
= false;
375 u8 wait_h2c_limmit
= 100;
376 u8 wait_writeh2c_limmit
= 100;
377 u8 boxcontent
[4], boxextcontent
[2];
378 u32 h2c_waitcounter
= 0;
382 if (ppsc
->rfpwr_state
== ERFOFF
|| ppsc
->inactive_pwrstate
== ERFOFF
) {
383 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
384 ("Return as RF is off!!!\n"));
387 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
, ("come in\n"));
389 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
390 if (rtlhal
->h2c_setinprogress
) {
391 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
392 ("H2C set in progress! Wait to set.."
393 "element_id(%d).\n", element_id
));
395 while (rtlhal
->h2c_setinprogress
) {
396 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
,
399 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
400 ("Wait 100 us (%d times)...\n",
404 if (h2c_waitcounter
> 1000)
407 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
,
410 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
412 rtlhal
->h2c_setinprogress
= true;
413 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
417 while (!bwrite_sucess
) {
418 wait_writeh2c_limmit
--;
419 if (wait_writeh2c_limmit
== 0) {
420 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
421 ("Write H2C fail because no trigger "
425 boxnum
= rtlhal
->last_hmeboxnum
;
428 box_reg
= REG_HMEBOX_0
;
429 box_extreg
= REG_HMEBOX_EXT_0
;
432 box_reg
= REG_HMEBOX_1
;
433 box_extreg
= REG_HMEBOX_EXT_1
;
436 box_reg
= REG_HMEBOX_2
;
437 box_extreg
= REG_HMEBOX_EXT_2
;
440 box_reg
= REG_HMEBOX_3
;
441 box_extreg
= REG_HMEBOX_EXT_3
;
444 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
445 ("switch case not process\n"));
448 isfw_read
= _rtl92d_check_fw_read_last_h2c(hw
, boxnum
);
451 if (wait_h2c_limmit
== 0) {
452 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
453 ("Wating too long for FW read "
454 "clear HMEBox(%d)!\n", boxnum
));
458 isfw_read
= _rtl92d_check_fw_read_last_h2c(hw
, boxnum
);
459 u1b_tmp
= rtl_read_byte(rtlpriv
, 0x1BF);
460 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
461 ("Wating for FW read clear HMEBox(%d)!!! "
462 "0x1BF = %2x\n", boxnum
, u1b_tmp
));
465 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
466 ("Write H2C register BOX[%d] fail!!!!! "
467 "Fw do not read.\n", boxnum
));
470 memset(boxcontent
, 0, sizeof(boxcontent
));
471 memset(boxextcontent
, 0, sizeof(boxextcontent
));
472 boxcontent
[0] = element_id
;
473 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
474 ("Write element_id box_reg(%4x) = %2x\n",
475 box_reg
, element_id
));
478 boxcontent
[0] &= ~(BIT(7));
479 memcpy(boxcontent
+ 1, cmdbuffer
+ buf_index
, 1);
480 for (idx
= 0; idx
< 4; idx
++)
481 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
485 boxcontent
[0] &= ~(BIT(7));
486 memcpy(boxcontent
+ 1, cmdbuffer
+ buf_index
, 2);
487 for (idx
= 0; idx
< 4; idx
++)
488 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
492 boxcontent
[0] &= ~(BIT(7));
493 memcpy(boxcontent
+ 1, cmdbuffer
+ buf_index
, 3);
494 for (idx
= 0; idx
< 4; idx
++)
495 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
499 boxcontent
[0] |= (BIT(7));
500 memcpy(boxextcontent
, cmdbuffer
+ buf_index
, 2);
501 memcpy(boxcontent
+ 1, cmdbuffer
+ buf_index
+ 2, 2);
502 for (idx
= 0; idx
< 2; idx
++)
503 rtl_write_byte(rtlpriv
, box_extreg
+ idx
,
505 for (idx
= 0; idx
< 4; idx
++)
506 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
510 boxcontent
[0] |= (BIT(7));
511 memcpy(boxextcontent
, cmdbuffer
+ buf_index
, 2);
512 memcpy(boxcontent
+ 1, cmdbuffer
+ buf_index
+ 2, 3);
513 for (idx
= 0; idx
< 2; idx
++)
514 rtl_write_byte(rtlpriv
, box_extreg
+ idx
,
516 for (idx
= 0; idx
< 4; idx
++)
517 rtl_write_byte(rtlpriv
, box_reg
+ idx
,
521 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_EMERG
,
522 ("switch case not process\n"));
525 bwrite_sucess
= true;
526 rtlhal
->last_hmeboxnum
= boxnum
+ 1;
527 if (rtlhal
->last_hmeboxnum
== 4)
528 rtlhal
->last_hmeboxnum
= 0;
529 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
,
530 ("pHalData->last_hmeboxnum = %d\n",
531 rtlhal
->last_hmeboxnum
));
533 spin_lock_irqsave(&rtlpriv
->locks
.h2c_lock
, flag
);
534 rtlhal
->h2c_setinprogress
= false;
535 spin_unlock_irqrestore(&rtlpriv
->locks
.h2c_lock
, flag
);
536 RT_TRACE(rtlpriv
, COMP_CMD
, DBG_LOUD
, ("go out\n"));
539 void rtl92d_fill_h2c_cmd(struct ieee80211_hw
*hw
,
540 u8 element_id
, u32 cmd_len
, u8
*cmdbuffer
)
544 memset(tmp_cmdbuf
, 0, 8);
545 memcpy(tmp_cmdbuf
, cmdbuffer
, cmd_len
);
546 _rtl92d_fill_h2c_command(hw
, element_id
, cmd_len
, (u8
*)&tmp_cmdbuf
);
550 void rtl92d_set_fw_pwrmode_cmd(struct ieee80211_hw
*hw
, u8 mode
)
552 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
553 u8 u1_h2c_set_pwrmode
[3] = { 0 };
554 struct rtl_ps_ctl
*ppsc
= rtl_psc(rtl_priv(hw
));
556 RT_TRACE(rtlpriv
, COMP_POWER
, DBG_LOUD
, ("FW LPS mode = %d\n", mode
));
557 SET_H2CCMD_PWRMODE_PARM_MODE(u1_h2c_set_pwrmode
, mode
);
558 SET_H2CCMD_PWRMODE_PARM_SMART_PS(u1_h2c_set_pwrmode
, 1);
559 SET_H2CCMD_PWRMODE_PARM_BCN_PASS_TIME(u1_h2c_set_pwrmode
,
560 ppsc
->reg_max_lps_awakeintvl
);
561 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
562 "rtl92d_set_fw_rsvdpagepkt(): u1_h2c_set_pwrmode\n",
563 u1_h2c_set_pwrmode
, 3);
564 rtl92d_fill_h2c_cmd(hw
, H2C_SETPWRMODE
, 3, u1_h2c_set_pwrmode
);
567 static bool _rtl92d_cmd_send_packet(struct ieee80211_hw
*hw
,
570 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
571 struct rtl_pci
*rtlpci
= rtl_pcidev(rtl_pcipriv(hw
));
572 struct rtl8192_tx_ring
*ring
;
573 struct rtl_tx_desc
*pdesc
;
576 struct sk_buff
*pskb
;
578 ring
= &rtlpci
->tx_ring
[BEACON_QUEUE
];
579 pskb
= __skb_dequeue(&ring
->queue
);
582 spin_lock_irqsave(&rtlpriv
->locks
.irq_th_lock
, flags
);
583 pdesc
= &ring
->desc
[idx
];
584 /* discard output from call below */
585 rtlpriv
->cfg
->ops
->get_desc((u8
*) pdesc
, true, HW_DESC_OWN
);
586 rtlpriv
->cfg
->ops
->fill_tx_cmddesc(hw
, (u8
*) pdesc
, 1, 1, skb
);
587 __skb_queue_tail(&ring
->queue
, skb
);
588 spin_unlock_irqrestore(&rtlpriv
->locks
.irq_th_lock
, flags
);
589 rtlpriv
->cfg
->ops
->tx_polling(hw
, BEACON_QUEUE
);
593 #define BEACON_PG 0 /*->1 */
596 #define PROBERSP_PG 4 /*->5 */
597 #define TOTAL_RESERVED_PKT_LEN 768
599 static u8 reserved_page_packet
[TOTAL_RESERVED_PKT_LEN
] = {
601 0x80, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF,
602 0xFF, 0xFF, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
603 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x50, 0x08,
604 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
605 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69,
606 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C,
607 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96,
608 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A,
609 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C,
610 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18,
611 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
612 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
613 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
614 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02,
615 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
616 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 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
627 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
628 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
629 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
630 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
631 0x10, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x10, 0x00,
632 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
633 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
634 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
637 0xA4, 0x10, 0x01, 0xC0, 0x00, 0x40, 0x10, 0x10,
638 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
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 0x18, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00,
650 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80,
651 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
652 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
655 0x48, 0x01, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10,
656 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
657 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00,
658 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
659 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
660 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
661 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
662 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
663 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
664 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
665 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
666 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
667 0x72, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00,
668 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80,
669 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
670 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
672 /* page 4 probe_resp */
673 0x50, 0x00, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10,
674 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42,
675 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00,
676 0x9E, 0x46, 0x15, 0x32, 0x27, 0xF2, 0x2D, 0x00,
677 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69,
678 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C,
679 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96,
680 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A,
681 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C,
682 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18,
683 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
684 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
685 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
686 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02,
687 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
688 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
690 /* page 5 probe_resp */
691 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
692 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
693 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
694 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
695 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
696 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
697 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
698 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
699 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
700 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
701 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
702 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
703 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
704 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
705 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
706 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
709 void rtl92d_set_fw_rsvdpagepkt(struct ieee80211_hw
*hw
, bool dl_finished
)
711 struct rtl_priv
*rtlpriv
= rtl_priv(hw
);
712 struct rtl_mac
*mac
= rtl_mac(rtl_priv(hw
));
713 struct sk_buff
*skb
= NULL
;
716 u8 u1RsvdPageLoc
[3] = { 0 };
722 /*---------------------------------------------------------
724 ---------------------------------------------------------*/
725 beacon
= &reserved_page_packet
[BEACON_PG
* 128];
726 SET_80211_HDR_ADDRESS2(beacon
, mac
->mac_addr
);
727 SET_80211_HDR_ADDRESS3(beacon
, mac
->bssid
);
728 /*-------------------------------------------------------
730 --------------------------------------------------------*/
731 p_pspoll
= &reserved_page_packet
[PSPOLL_PG
* 128];
732 SET_80211_PS_POLL_AID(p_pspoll
, (mac
->assoc_id
| 0xc000));
733 SET_80211_PS_POLL_BSSID(p_pspoll
, mac
->bssid
);
734 SET_80211_PS_POLL_TA(p_pspoll
, mac
->mac_addr
);
735 SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(u1RsvdPageLoc
, PSPOLL_PG
);
736 /*--------------------------------------------------------
738 ---------------------------------------------------------*/
739 nullfunc
= &reserved_page_packet
[NULL_PG
* 128];
740 SET_80211_HDR_ADDRESS1(nullfunc
, mac
->bssid
);
741 SET_80211_HDR_ADDRESS2(nullfunc
, mac
->mac_addr
);
742 SET_80211_HDR_ADDRESS3(nullfunc
, mac
->bssid
);
743 SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(u1RsvdPageLoc
, NULL_PG
);
744 /*---------------------------------------------------------
746 ----------------------------------------------------------*/
747 p_probersp
= &reserved_page_packet
[PROBERSP_PG
* 128];
748 SET_80211_HDR_ADDRESS1(p_probersp
, mac
->bssid
);
749 SET_80211_HDR_ADDRESS2(p_probersp
, mac
->mac_addr
);
750 SET_80211_HDR_ADDRESS3(p_probersp
, mac
->bssid
);
751 SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(u1RsvdPageLoc
, PROBERSP_PG
);
752 totalpacketlen
= TOTAL_RESERVED_PKT_LEN
;
753 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_LOUD
,
754 "rtl92d_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
755 &reserved_page_packet
[0], totalpacketlen
);
756 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
757 "rtl92d_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n",
759 skb
= dev_alloc_skb(totalpacketlen
);
763 memcpy((u8
*) skb_put(skb
, totalpacketlen
),
764 &reserved_page_packet
, totalpacketlen
);
765 rtstatus
= _rtl92d_cmd_send_packet(hw
, skb
);
771 RT_TRACE(rtlpriv
, COMP_POWER
, DBG_LOUD
,
772 ("Set RSVD page location to Fw.\n"));
773 RT_PRINT_DATA(rtlpriv
, COMP_CMD
, DBG_DMESG
,
774 "H2C_RSVDPAGE:\n", u1RsvdPageLoc
, 3);
775 rtl92d_fill_h2c_cmd(hw
, H2C_RSVDPAGE
,
776 sizeof(u1RsvdPageLoc
), u1RsvdPageLoc
);
778 RT_TRACE(rtlpriv
, COMP_ERR
, DBG_WARNING
,
779 ("Set RSVD page location to Fw FAIL!!!!!!.\n"));
782 void rtl92d_set_fw_joinbss_report_cmd(struct ieee80211_hw
*hw
, u8 mstatus
)
784 u8 u1_joinbssrpt_parm
[1] = {0};
786 SET_H2CCMD_JOINBSSRPT_PARM_OPMODE(u1_joinbssrpt_parm
, mstatus
);
787 rtl92d_fill_h2c_cmd(hw
, H2C_JOINBSSRPT
, 1, u1_joinbssrpt_parm
);