1 /******************************************************************************
3 * Copyright(c) 2007 - 2012 Realtek Corporation. All rights reserved.
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 ******************************************************************************/
16 #include <osdep_service.h>
17 #include <drv_types.h>
18 #include <recv_osdep.h>
19 #include <mlme_osdep.h>
21 #include <linux/if_ether.h>
23 #include <linux/ieee80211.h>
25 #include <rtl8723a_recv.h>
26 #include <rtl8723a_xmit.h>
28 void rtw_signal_stat_timer_hdl23a(unsigned long data
);
30 void _rtw_init_sta_recv_priv23a(struct sta_recv_priv
*psta_recvpriv
)
35 spin_lock_init(&psta_recvpriv
->lock
);
37 /* for (i = 0; i<MAX_RX_NUMBLKS; i++) */
38 /* _rtw_init_queue23a(&psta_recvpriv->blk_strms[i]); */
40 _rtw_init_queue23a(&psta_recvpriv
->defrag_q
);
45 int _rtw_init_recv_priv23a(struct recv_priv
*precvpriv
,
46 struct rtw_adapter
*padapter
)
48 struct recv_frame
*precvframe
;
52 spin_lock_init(&precvpriv
->lock
);
54 _rtw_init_queue23a(&precvpriv
->free_recv_queue
);
55 _rtw_init_queue23a(&precvpriv
->recv_pending_queue
);
56 _rtw_init_queue23a(&precvpriv
->uc_swdec_pending_queue
);
58 precvpriv
->adapter
= padapter
;
60 for (i
= 0; i
< NR_RECVFRAME
; i
++) {
61 precvframe
= kzalloc(sizeof(struct recv_frame
), GFP_KERNEL
);
64 INIT_LIST_HEAD(&precvframe
->list
);
66 list_add_tail(&precvframe
->list
,
67 &precvpriv
->free_recv_queue
.queue
);
69 precvframe
->adapter
= padapter
;
73 precvpriv
->free_recvframe_cnt
= i
;
74 precvpriv
->rx_pending_cnt
= 1;
76 res
= rtl8723au_init_recv_priv(padapter
);
78 setup_timer(&precvpriv
->signal_stat_timer
, rtw_signal_stat_timer_hdl23a
,
79 (unsigned long)padapter
);
81 precvpriv
->signal_stat_sampling_interval
= 1000; /* ms */
83 rtw_set_signal_stat_timer(precvpriv
);
88 void _rtw_free_recv_priv23a (struct recv_priv
*precvpriv
)
90 struct rtw_adapter
*padapter
= precvpriv
->adapter
;
91 struct recv_frame
*precvframe
;
92 struct list_head
*plist
, *ptmp
;
94 rtw_free_uc_swdec_pending_queue23a(padapter
);
96 list_for_each_safe(plist
, ptmp
, &precvpriv
->free_recv_queue
.queue
) {
97 precvframe
= container_of(plist
, struct recv_frame
, list
);
98 list_del_init(&precvframe
->list
);
102 rtl8723au_free_recv_priv(padapter
);
105 struct recv_frame
*rtw_alloc_recvframe23a(struct rtw_queue
*pfree_recv_queue
)
107 struct recv_frame
*pframe
;
108 struct list_head
*plist
, *phead
;
109 struct rtw_adapter
*padapter
;
110 struct recv_priv
*precvpriv
;
112 spin_lock_bh(&pfree_recv_queue
->lock
);
114 if (list_empty(&pfree_recv_queue
->queue
))
117 phead
= get_list_head(pfree_recv_queue
);
121 pframe
= container_of(plist
, struct recv_frame
, list
);
123 list_del_init(&pframe
->list
);
124 padapter
= pframe
->adapter
;
126 precvpriv
= &padapter
->recvpriv
;
127 if (pfree_recv_queue
== &precvpriv
->free_recv_queue
)
128 precvpriv
->free_recvframe_cnt
--;
132 spin_unlock_bh(&pfree_recv_queue
->lock
);
137 int rtw_free_recvframe23a(struct recv_frame
*precvframe
)
139 struct rtw_adapter
*padapter
= precvframe
->adapter
;
140 struct recv_priv
*precvpriv
= &padapter
->recvpriv
;
141 struct rtw_queue
*pfree_recv_queue
;
143 if (precvframe
->pkt
) {
144 dev_kfree_skb_any(precvframe
->pkt
);/* free skb by driver */
145 precvframe
->pkt
= NULL
;
148 pfree_recv_queue
= &precvpriv
->free_recv_queue
;
149 spin_lock_bh(&pfree_recv_queue
->lock
);
151 list_del_init(&precvframe
->list
);
153 list_add_tail(&precvframe
->list
, get_list_head(pfree_recv_queue
));
156 if (pfree_recv_queue
== &precvpriv
->free_recv_queue
)
157 precvpriv
->free_recvframe_cnt
++;
160 spin_unlock_bh(&pfree_recv_queue
->lock
);
167 int rtw_enqueue_recvframe23a(struct recv_frame
*precvframe
, struct rtw_queue
*queue
)
169 struct rtw_adapter
*padapter
= precvframe
->adapter
;
170 struct recv_priv
*precvpriv
= &padapter
->recvpriv
;
172 spin_lock_bh(&queue
->lock
);
174 list_del_init(&precvframe
->list
);
176 list_add_tail(&precvframe
->list
, get_list_head(queue
));
179 if (queue
== &precvpriv
->free_recv_queue
)
180 precvpriv
->free_recvframe_cnt
++;
183 spin_unlock_bh(&queue
->lock
);
189 caller : defrag ; recvframe_chk_defrag23a in recv_thread (passive)
190 pframequeue: defrag_queue : will be accessed in recv_thread (passive)
192 using spinlock to protect
196 static void rtw_free_recvframe23a_queue(struct rtw_queue
*pframequeue
)
198 struct recv_frame
*hdr
;
199 struct list_head
*plist
, *phead
, *ptmp
;
201 spin_lock(&pframequeue
->lock
);
203 phead
= get_list_head(pframequeue
);
206 list_for_each_safe(plist
, ptmp
, phead
) {
207 hdr
= container_of(plist
, struct recv_frame
, list
);
208 rtw_free_recvframe23a(hdr
);
211 spin_unlock(&pframequeue
->lock
);
214 u32
rtw_free_uc_swdec_pending_queue23a(struct rtw_adapter
*adapter
)
217 struct recv_frame
*pending_frame
;
218 while ((pending_frame
= rtw_alloc_recvframe23a(&adapter
->recvpriv
.uc_swdec_pending_queue
))) {
219 rtw_free_recvframe23a(pending_frame
);
220 DBG_8723A("%s: dequeue uc_swdec_pending_queue\n", __func__
);
227 int rtw_enqueue_recvbuf23a_to_head(struct recv_buf
*precvbuf
, struct rtw_queue
*queue
)
229 spin_lock_bh(&queue
->lock
);
231 list_del_init(&precvbuf
->list
);
232 list_add(&precvbuf
->list
, get_list_head(queue
));
234 spin_unlock_bh(&queue
->lock
);
239 int rtw_enqueue_recvbuf23a(struct recv_buf
*precvbuf
, struct rtw_queue
*queue
)
242 spin_lock_irqsave(&queue
->lock
, irqL
);
244 list_del_init(&precvbuf
->list
);
246 list_add_tail(&precvbuf
->list
, get_list_head(queue
));
247 spin_unlock_irqrestore(&queue
->lock
, irqL
);
251 struct recv_buf
*rtw_dequeue_recvbuf23a (struct rtw_queue
*queue
)
254 struct recv_buf
*precvbuf
;
255 struct list_head
*plist
, *phead
;
257 spin_lock_irqsave(&queue
->lock
, irqL
);
259 if (list_empty(&queue
->queue
)) {
262 phead
= get_list_head(queue
);
266 precvbuf
= container_of(plist
, struct recv_buf
, list
);
268 list_del_init(&precvbuf
->list
);
271 spin_unlock_irqrestore(&queue
->lock
, irqL
);
276 int recvframe_chkmic(struct rtw_adapter
*adapter
,
277 struct recv_frame
*precvframe
);
278 int recvframe_chkmic(struct rtw_adapter
*adapter
,
279 struct recv_frame
*precvframe
) {
281 int i
, res
= _SUCCESS
;
284 u8 bmic_err
= false, brpt_micerror
= true;
285 u8
*pframe
, *payload
,*pframemic
;
287 /* u8 *iv, rxdata_key_idx = 0; */
288 struct sta_info
*stainfo
;
289 struct rx_pkt_attrib
*prxattrib
= &precvframe
->attrib
;
290 struct security_priv
*psecuritypriv
= &adapter
->securitypriv
;
292 struct mlme_ext_priv
*pmlmeext
= &adapter
->mlmeextpriv
;
293 struct mlme_ext_info
*pmlmeinfo
= &pmlmeext
->mlmext_info
;
296 stainfo
= rtw_get_stainfo23a(&adapter
->stapriv
, &prxattrib
->ta
[0]);
298 if (prxattrib
->encrypt
== WLAN_CIPHER_SUITE_TKIP
) {
299 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
300 ("\n recvframe_chkmic:prxattrib->encrypt == WLAN_CIPHER_SUITE_TKIP\n"));
301 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
302 ("\n recvframe_chkmic:da = 0x%02x:0x%02x:0x%02x:0x%02x:"
303 "0x%02x:0x%02x\n", prxattrib
->ra
[0],
304 prxattrib
->ra
[1], prxattrib
->ra
[2], prxattrib
->ra
[3],
305 prxattrib
->ra
[4], prxattrib
->ra
[5]));
307 /* calculate mic code */
308 if (stainfo
!= NULL
) {
309 if (is_multicast_ether_addr(prxattrib
->ra
)) {
310 mickey
= &psecuritypriv
->dot118021XGrprxmickey
[prxattrib
->key_index
].skey
[0];
312 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
313 ("\n recvframe_chkmic: bcmc key\n"));
315 if (!psecuritypriv
->binstallGrpkey
) {
317 RT_TRACE(_module_rtl871x_recv_c_
,
319 ("\n recvframe_chkmic:didn't "
320 "install group key!!!!!!\n"));
321 DBG_8723A("\n recvframe_chkmic:didn't "
322 "install group key!!!!!!\n");
326 mickey
= &stainfo
->dot11tkiprxmickey
.skey
[0];
327 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
328 ("\n recvframe_chkmic: unicast "
332 /* icv_len included the mic code */
333 datalen
= precvframe
->pkt
->len
-prxattrib
->
334 hdrlen
-prxattrib
->iv_len
-prxattrib
->icv_len
- 8;
335 pframe
= precvframe
->pkt
->data
;
336 payload
= pframe
+ prxattrib
->hdrlen
+
339 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
340 ("\n prxattrib->iv_len =%d prxattrib->icv_len ="
341 "%d\n", prxattrib
->iv_len
,
342 prxattrib
->icv_len
));
344 /* care the length of the data */
345 rtw_seccalctkipmic23a(mickey
, pframe
, payload
,
346 datalen
, &miccode
[0],
347 (unsigned char)prxattrib
->priority
);
349 pframemic
= payload
+ datalen
;
353 for (i
= 0; i
< 8; i
++) {
354 if (miccode
[i
] != *(pframemic
+ i
)) {
355 RT_TRACE(_module_rtl871x_recv_c_
,
357 ("recvframe_chkmic:miccode"
358 "[%d](%02x) != *(pframemic+"
359 "%d)(%02x) ", i
, miccode
[i
],
360 i
, *(pframemic
+ i
)));
365 if (bmic_err
== true) {
367 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
368 ("\n *(pframemic-8)-*(pframemic-1) ="
369 "0x%02x:0x%02x:0x%02x:0x%02x:0x%02x:"
370 "0x%02x:0x%02x:0x%02x\n",
371 *(pframemic
- 8), *(pframemic
- 7),
372 *(pframemic
- 6), *(pframemic
- 5),
373 *(pframemic
- 4), *(pframemic
- 3),
374 *(pframemic
- 2), *(pframemic
- 1)));
375 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
376 ("\n *(pframemic-16)-*(pframemic-9) ="
377 "0x%02x:0x%02x:0x%02x:0x%02x:0x%02x:"
378 "0x%02x:0x%02x:0x%02x\n",
379 *(pframemic
- 16), *(pframemic
- 15),
380 *(pframemic
- 14), *(pframemic
- 13),
381 *(pframemic
- 12), *(pframemic
- 11),
382 *(pframemic
- 10), *(pframemic
- 9)));
384 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
385 ("\n ====== demp packet (len =%d) ======"
386 "\n", precvframe
->pkt
->len
));
387 for (i
= 0; i
< precvframe
->pkt
->len
; i
= i
+ 8) {
388 RT_TRACE(_module_rtl871x_recv_c_
,
389 _drv_err_
, ("0x%02x:0x%02x:0x"
393 *(precvframe
->pkt
->data
+i
),*(precvframe
->pkt
->data
+i
+1),
394 *(precvframe
->pkt
->data
+i
+2),*(precvframe
->pkt
->data
+i
+3),
395 *(precvframe
->pkt
->data
+i
+4),*(precvframe
->pkt
->data
+i
+5),
396 *(precvframe
->pkt
->data
+i
+6),*(precvframe
->pkt
->data
+i
+7)));
398 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
399 ("\n ====== demp packet end [len =%d]"
400 "======\n", precvframe
->pkt
->len
));
401 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
405 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
406 ("ra = 0x%.2x 0x%.2x 0x%.2x 0x%.2x 0x%."
407 "2x 0x%.2x psecuritypriv->"
408 "binstallGrpkey =%d ",
409 prxattrib
->ra
[0], prxattrib
->ra
[1],
410 prxattrib
->ra
[2], prxattrib
->ra
[3],
411 prxattrib
->ra
[4], prxattrib
->ra
[5],
412 psecuritypriv
->binstallGrpkey
));
414 /* double check key_index for some timing
415 issue, cannot compare with
416 psecuritypriv->dot118021XGrpKeyid also
417 cause timing issue */
418 if ((is_multicast_ether_addr(prxattrib
->ra
)) &&
419 (prxattrib
->key_index
!=
420 pmlmeinfo
->key_index
))
421 brpt_micerror
= false;
423 if ((prxattrib
->bdecrypted
== true) &&
424 (brpt_micerror
== true)) {
425 rtw_handle_tkip_mic_err23a(adapter
, (u8
)is_multicast_ether_addr(prxattrib
->ra
));
426 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
, (" mic error :prxattrib->bdecrypted =%d ", prxattrib
->bdecrypted
));
427 DBG_8723A(" mic error :prxattrib->"
429 prxattrib
->bdecrypted
);
431 RT_TRACE(_module_rtl871x_recv_c_
,
433 (" mic error :prxattrib->"
435 prxattrib
->bdecrypted
));
436 DBG_8723A(" mic error :prxattrib->"
438 prxattrib
->bdecrypted
);
444 if (!psecuritypriv
->bcheck_grpkey
&&
445 is_multicast_ether_addr(prxattrib
->ra
)) {
446 psecuritypriv
->bcheck_grpkey
= 1;
447 RT_TRACE(_module_rtl871x_recv_c_
,
449 ("psecuritypriv->bcheck_grp"
454 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
455 ("recvframe_chkmic: rtw_get_stainfo23a =="
459 skb_trim(precvframe
->pkt
, precvframe
->pkt
->len
- 8);
469 /* decrypt and set the ivlen, icvlen of the recv_frame */
470 struct recv_frame
*decryptor(struct rtw_adapter
*padapter
,
471 struct recv_frame
*precv_frame
);
472 struct recv_frame
*decryptor(struct rtw_adapter
*padapter
,
473 struct recv_frame
*precv_frame
)
475 struct rx_pkt_attrib
*prxattrib
= &precv_frame
->attrib
;
476 struct security_priv
*psecuritypriv
= &padapter
->securitypriv
;
477 struct recv_frame
*return_packet
= precv_frame
;
480 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
481 ("prxstat->decrypted =%x prxattrib->encrypt = 0x%03x\n",
482 prxattrib
->bdecrypted
, prxattrib
->encrypt
));
484 if (prxattrib
->encrypt
> 0) {
485 u8
*iv
= precv_frame
->pkt
->data
+ prxattrib
->hdrlen
;
486 prxattrib
->key_index
= (((iv
[3]) >> 6) & 0x3);
488 if (prxattrib
->key_index
> WEP_KEYS
) {
489 DBG_8723A("prxattrib->key_index(%d) > WEP_KEYS\n",
490 prxattrib
->key_index
);
492 switch (prxattrib
->encrypt
) {
493 case WLAN_CIPHER_SUITE_WEP40
:
494 case WLAN_CIPHER_SUITE_WEP104
:
495 prxattrib
->key_index
=
496 psecuritypriv
->dot11PrivacyKeyIndex
;
498 case WLAN_CIPHER_SUITE_TKIP
:
499 case WLAN_CIPHER_SUITE_CCMP
:
501 prxattrib
->key_index
=
502 psecuritypriv
->dot118021XGrpKeyid
;
508 if ((prxattrib
->encrypt
> 0) && ((prxattrib
->bdecrypted
== 0))) {
509 psecuritypriv
->hw_decrypted
= 0;
510 switch (prxattrib
->encrypt
) {
511 case WLAN_CIPHER_SUITE_WEP40
:
512 case WLAN_CIPHER_SUITE_WEP104
:
513 rtw_wep_decrypt23a(padapter
, precv_frame
);
515 case WLAN_CIPHER_SUITE_TKIP
:
516 res
= rtw_tkip_decrypt23a(padapter
, precv_frame
);
518 case WLAN_CIPHER_SUITE_CCMP
:
519 res
= rtw_aes_decrypt23a(padapter
, precv_frame
);
524 } else if (prxattrib
->bdecrypted
== 1 && prxattrib
->encrypt
> 0 &&
525 (psecuritypriv
->busetkipkey
== 1 ||
526 prxattrib
->encrypt
!= WLAN_CIPHER_SUITE_TKIP
)) {
527 psecuritypriv
->hw_decrypted
= 1;
531 rtw_free_recvframe23a(return_packet
);
532 return_packet
= NULL
;
537 return return_packet
;
540 /* set the security information in the recv_frame */
541 static struct recv_frame
*portctrl(struct rtw_adapter
*adapter
,
542 struct recv_frame
*precv_frame
)
546 struct recv_frame
*pfhdr
;
547 struct sta_info
*psta
;
548 struct sta_priv
*pstapriv
;
549 struct recv_frame
*prtnframe
;
551 u16 eapol_type
= ETH_P_PAE
;/* for Funia BD's WPA issue */
552 struct rx_pkt_attrib
*pattrib
;
554 pstapriv
= &adapter
->stapriv
;
556 auth_alg
= adapter
->securitypriv
.dot11AuthAlgrthm
;
559 pattrib
= &pfhdr
->attrib
;
560 psta_addr
= pattrib
->ta
;
561 psta
= rtw_get_stainfo23a(pstapriv
, psta_addr
);
563 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
564 ("########portctrl:adapter->securitypriv.dot11AuthAlgrthm ="
565 "%d\n", adapter
->securitypriv
.dot11AuthAlgrthm
));
567 if (auth_alg
== dot11AuthAlgrthm_8021X
) {
569 ptr
= pfhdr
->pkt
->data
+ pfhdr
->attrib
.hdrlen
;
571 ether_type
= (ptr
[6] << 8) | ptr
[7];
573 if ((psta
!= NULL
) && (psta
->ieee8021x_blocked
)) {
575 /* only accept EAPOL frame */
576 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
577 ("########portctrl:psta->ieee8021x_blocked =="
580 if (ether_type
== eapol_type
) {
581 prtnframe
= precv_frame
;
583 /* free this frame */
584 rtw_free_recvframe23a(precv_frame
);
589 /* check decryption status, and decrypt the frame if needed */
590 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
591 ("########portctrl:psta->ieee8021x_blocked =="
593 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
594 ("portctrl:precv_frame->hdr.attrib.privacy ="
595 "%x\n", precv_frame
->attrib
.privacy
));
597 if (pattrib
->bdecrypted
== 0) {
598 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
599 ("portctrl:prxstat->decrypted =%x\n",
600 pattrib
->bdecrypted
));
603 prtnframe
= precv_frame
;
604 /* check is the EAPOL frame or not (Rekey) */
605 if (ether_type
== eapol_type
) {
606 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
607 ("########portctrl:ether_type == "
611 prtnframe
= precv_frame
;
613 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
614 ("########portctrl:ether_type = 0x%04x"
619 prtnframe
= precv_frame
;
625 int recv_decache(struct recv_frame
*precv_frame
, u8 bretry
,
626 struct stainfo_rxcache
*prxcache
);
627 int recv_decache(struct recv_frame
*precv_frame
, u8 bretry
,
628 struct stainfo_rxcache
*prxcache
)
630 int tid
= precv_frame
->attrib
.priority
;
632 u16 seq_ctrl
= ((precv_frame
->attrib
.seq_num
& 0xffff) << 4) |
633 (precv_frame
->attrib
.frag_num
& 0xf);
638 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
639 ("recv_decache, (tid>15)! seq_ctrl = 0x%x, tid = 0x%x\n",
645 if (1) { /* if (bretry) */
646 if (seq_ctrl
== prxcache
->tid_rxseq
[tid
]) {
647 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
648 ("recv_decache, seq_ctrl = 0x%x, tid = 0x%x, "
649 "tid_rxseq = 0x%x\n",
650 seq_ctrl
, tid
, prxcache
->tid_rxseq
[tid
]));
656 prxcache
->tid_rxseq
[tid
] = seq_ctrl
;
663 void process23a_pwrbit_data(struct rtw_adapter
*padapter
,
664 struct recv_frame
*precv_frame
);
665 void process23a_pwrbit_data(struct rtw_adapter
*padapter
,
666 struct recv_frame
*precv_frame
)
668 #ifdef CONFIG_8723AU_AP_MODE
669 unsigned char pwrbit
;
670 struct sk_buff
*skb
= precv_frame
->pkt
;
671 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
672 struct rx_pkt_attrib
*pattrib
= &precv_frame
->attrib
;
673 struct sta_priv
*pstapriv
= &padapter
->stapriv
;
674 struct sta_info
*psta
= NULL
;
676 psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->src
);
679 pwrbit
= ieee80211_has_pm(hdr
->frame_control
);
682 if (!(psta
->state
& WIFI_SLEEP_STATE
))
683 stop_sta_xmit23a(padapter
, psta
);
685 if (psta
->state
& WIFI_SLEEP_STATE
)
686 wakeup_sta_to_xmit23a(padapter
, psta
);
693 void process_wmmps_data(struct rtw_adapter
*padapter
,
694 struct recv_frame
*precv_frame
);
695 void process_wmmps_data(struct rtw_adapter
*padapter
,
696 struct recv_frame
*precv_frame
)
698 #ifdef CONFIG_8723AU_AP_MODE
699 struct rx_pkt_attrib
*pattrib
= &precv_frame
->attrib
;
700 struct sta_priv
*pstapriv
= &padapter
->stapriv
;
701 struct sta_info
*psta
= NULL
;
703 psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->src
);
709 if (!psta
->qos_option
)
712 if (!(psta
->qos_info
& 0xf))
715 if (psta
->state
& WIFI_SLEEP_STATE
) {
718 switch (pattrib
->priority
) {
721 wmmps_ac
= psta
->uapsd_bk
& BIT(1);
725 wmmps_ac
= psta
->uapsd_vi
& BIT(1);
729 wmmps_ac
= psta
->uapsd_vo
& BIT(1);
734 wmmps_ac
= psta
->uapsd_be
& BIT(1);
739 if (psta
->sleepq_ac_len
> 0) {
740 /* process received triggered frame */
741 xmit_delivery_enabled_frames23a(padapter
, psta
);
743 /* issue one qos null frame with More data bit = 0 and the EOSP bit set (= 1) */
744 issue_qos_nulldata23a(padapter
, psta
->hwaddr
,
745 (u16
)pattrib
->priority
,
754 static void count_rx_stats(struct rtw_adapter
*padapter
,
755 struct recv_frame
*prframe
, struct sta_info
*sta
)
758 struct sta_info
*psta
= NULL
;
759 struct stainfo_stats
*pstats
= NULL
;
760 struct rx_pkt_attrib
*pattrib
= & prframe
->attrib
;
761 struct recv_priv
*precvpriv
= &padapter
->recvpriv
;
763 sz
= prframe
->pkt
->len
;
764 precvpriv
->rx_bytes
+= sz
;
766 padapter
->mlmepriv
.LinkDetectInfo
.NumRxOkInPeriod
++;
768 if ((!is_broadcast_ether_addr(pattrib
->dst
)) &&
769 (!is_multicast_ether_addr(pattrib
->dst
)))
770 padapter
->mlmepriv
.LinkDetectInfo
.NumRxUnicastOkInPeriod
++;
775 psta
= prframe
->psta
;
778 pstats
= &psta
->sta_stats
;
780 pstats
->rx_data_pkts
++;
781 pstats
->rx_bytes
+= sz
;
785 static int sta2sta_data_frame(struct rtw_adapter
*adapter
,
786 struct recv_frame
*precv_frame
,
787 struct sta_info
**psta
)
789 struct sk_buff
*skb
= precv_frame
->pkt
;
790 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
792 struct rx_pkt_attrib
*pattrib
= & precv_frame
->attrib
;
793 struct sta_priv
*pstapriv
= &adapter
->stapriv
;
794 struct mlme_priv
*pmlmepriv
= &adapter
->mlmepriv
;
795 u8
*mybssid
= get_bssid(pmlmepriv
);
796 u8
*myhwaddr
= myid(&adapter
->eeprompriv
);
798 int bmcast
= is_multicast_ether_addr(pattrib
->dst
);
802 if (check_fwstate(pmlmepriv
, WIFI_ADHOC_STATE
) ||
803 check_fwstate(pmlmepriv
, WIFI_ADHOC_MASTER_STATE
)) {
805 /* filter packets that SA is myself or multicast or broadcast */
806 if (ether_addr_equal(myhwaddr
, pattrib
->src
)) {
807 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
808 (" SA == myself\n"));
813 if (!ether_addr_equal(myhwaddr
, pattrib
->dst
) && !bmcast
) {
818 if (ether_addr_equal(pattrib
->bssid
, "\x0\x0\x0\x0\x0\x0") ||
819 ether_addr_equal(mybssid
, "\x0\x0\x0\x0\x0\x0") ||
820 !ether_addr_equal(pattrib
->bssid
, mybssid
)) {
825 sta_addr
= pattrib
->src
;
826 } else if (check_fwstate(pmlmepriv
, WIFI_STATION_STATE
)) {
827 /* For Station mode, sa and bssid should always be BSSID,
828 and DA is my mac-address */
829 if (!ether_addr_equal(pattrib
->bssid
, pattrib
->src
)) {
830 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
831 ("bssid != TA under STATION_MODE; drop "
837 sta_addr
= pattrib
->bssid
;
839 } else if (check_fwstate(pmlmepriv
, WIFI_AP_STATE
)) {
841 /* For AP mode, if DA == MCAST, then BSSID should be also MCAST */
842 if (!is_multicast_ether_addr(pattrib
->bssid
)) {
846 } else { /* not mc-frame */
847 /* For AP mode, if DA is non-MCAST, then it must
848 be BSSID, and bssid == BSSID */
849 if (!ether_addr_equal(pattrib
->bssid
, pattrib
->dst
)) {
854 sta_addr
= pattrib
->src
;
856 } else if (check_fwstate(pmlmepriv
, WIFI_MP_STATE
)) {
857 ether_addr_copy(pattrib
->dst
, hdr
->addr1
);
858 ether_addr_copy(pattrib
->src
, hdr
->addr2
);
859 ether_addr_copy(pattrib
->bssid
, hdr
->addr3
);
860 ether_addr_copy(pattrib
->ra
, pattrib
->dst
);
861 ether_addr_copy(pattrib
->ta
, pattrib
->src
);
869 *psta
= rtw_get_bcmc_stainfo23a(adapter
);
871 *psta
= rtw_get_stainfo23a(pstapriv
, sta_addr
); /* get ap_info */
874 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
, ("can't get psta under sta2sta_data_frame ; drop pkt\n"));
884 int ap2sta_data_frame(struct rtw_adapter
*adapter
,
885 struct recv_frame
*precv_frame
,
886 struct sta_info
**psta
);
887 int ap2sta_data_frame(struct rtw_adapter
*adapter
,
888 struct recv_frame
*precv_frame
,
889 struct sta_info
**psta
)
891 struct sk_buff
*skb
= precv_frame
->pkt
;
892 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
893 struct rx_pkt_attrib
*pattrib
= & precv_frame
->attrib
;
895 struct sta_priv
*pstapriv
= &adapter
->stapriv
;
896 struct mlme_priv
*pmlmepriv
= &adapter
->mlmepriv
;
897 u8
*mybssid
= get_bssid(pmlmepriv
);
898 u8
*myhwaddr
= myid(&adapter
->eeprompriv
);
899 int bmcast
= is_multicast_ether_addr(pattrib
->dst
);
903 if (check_fwstate(pmlmepriv
, WIFI_STATION_STATE
) &&
904 (check_fwstate(pmlmepriv
, _FW_LINKED
) ||
905 check_fwstate(pmlmepriv
, _FW_UNDER_LINKING
))) {
907 /* filter packets that SA is myself or multicast or broadcast */
908 if (ether_addr_equal(myhwaddr
, pattrib
->src
)) {
909 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
910 (" SA == myself\n"));
915 /* da should be for me */
916 if (!ether_addr_equal(myhwaddr
, pattrib
->dst
) && !bmcast
) {
917 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
918 (" ap2sta_data_frame: compare DA fail; DA ="
919 MAC_FMT
"\n", MAC_ARG(pattrib
->dst
)));
925 if (ether_addr_equal(pattrib
->bssid
, "\x0\x0\x0\x0\x0\x0") ||
926 ether_addr_equal(mybssid
, "\x0\x0\x0\x0\x0\x0") ||
927 !ether_addr_equal(pattrib
->bssid
, mybssid
)) {
928 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
929 (" ap2sta_data_frame: compare BSSID fail ; "
930 "BSSID ="MAC_FMT
"\n", MAC_ARG(pattrib
->bssid
)));
931 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
932 ("mybssid ="MAC_FMT
"\n", MAC_ARG(mybssid
)));
935 DBG_8723A("issue_deauth23a to the nonassociated "
936 "ap =" MAC_FMT
" for the reason(7)\n",
937 MAC_ARG(pattrib
->bssid
));
938 issue_deauth23a(adapter
, pattrib
->bssid
,
939 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA
);
947 *psta
= rtw_get_bcmc_stainfo23a(adapter
);
950 *psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->bssid
);
953 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
954 ("ap2sta: can't get psta under STATION_MODE ;"
960 if (ieee80211_is_nullfunc(hdr
->frame_control
)) {
961 /* No data, will not indicate to upper layer,
962 temporily count it here */
963 count_rx_stats(adapter
, precv_frame
, *psta
);
964 ret
= RTW_RX_HANDLED
;
968 } else if (check_fwstate(pmlmepriv
, WIFI_MP_STATE
) &&
969 check_fwstate(pmlmepriv
, _FW_LINKED
)) {
970 ether_addr_copy(pattrib
->dst
, hdr
->addr1
);
971 ether_addr_copy(pattrib
->src
, hdr
->addr2
);
972 ether_addr_copy(pattrib
->bssid
, hdr
->addr3
);
973 ether_addr_copy(pattrib
->ra
, pattrib
->dst
);
974 ether_addr_copy(pattrib
->ta
, pattrib
->src
);
977 ether_addr_copy(pattrib
->bssid
, mybssid
);
980 *psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->bssid
);
982 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
983 ("can't get psta under MP_MODE ; drop pkt\n"));
987 } else if (check_fwstate(pmlmepriv
, WIFI_AP_STATE
)) {
989 ret
= RTW_RX_HANDLED
;
992 if (ether_addr_equal(myhwaddr
, pattrib
->dst
) && !bmcast
) {
993 *psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->bssid
);
995 DBG_8723A("issue_deauth23a to the ap =" MAC_FMT
996 " for the reason(7)\n",
997 MAC_ARG(pattrib
->bssid
));
999 issue_deauth23a(adapter
, pattrib
->bssid
,
1000 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA
);
1014 int sta2ap_data_frame(struct rtw_adapter
*adapter
,
1015 struct recv_frame
*precv_frame
,
1016 struct sta_info
**psta
);
1017 int sta2ap_data_frame(struct rtw_adapter
*adapter
,
1018 struct recv_frame
*precv_frame
,
1019 struct sta_info
**psta
)
1021 struct sk_buff
*skb
= precv_frame
->pkt
;
1022 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
1023 struct rx_pkt_attrib
*pattrib
= & precv_frame
->attrib
;
1024 struct sta_priv
*pstapriv
= &adapter
->stapriv
;
1025 struct mlme_priv
*pmlmepriv
= &adapter
->mlmepriv
;
1026 unsigned char *mybssid
= get_bssid(pmlmepriv
);
1031 if (check_fwstate(pmlmepriv
, WIFI_AP_STATE
)) {
1032 /* For AP mode, RA = BSSID, TX = STA(SRC_ADDR), A3 = DST_ADDR */
1033 if (!ether_addr_equal(pattrib
->bssid
, mybssid
)) {
1038 *psta
= rtw_get_stainfo23a(pstapriv
, pattrib
->src
);
1039 if (*psta
== NULL
) {
1040 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1041 ("can't get psta under AP_MODE; drop pkt\n"));
1042 DBG_8723A("issue_deauth23a to sta =" MAC_FMT
1043 " for the reason(7)\n",
1044 MAC_ARG(pattrib
->src
));
1046 issue_deauth23a(adapter
, pattrib
->src
,
1047 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA
);
1049 ret
= RTW_RX_HANDLED
;
1053 process23a_pwrbit_data(adapter
, precv_frame
);
1055 /* We only get here if it's a data frame, so no need to
1056 * confirm data frame type first */
1057 if (ieee80211_is_data_qos(hdr
->frame_control
))
1058 process_wmmps_data(adapter
, precv_frame
);
1060 if (ieee80211_is_nullfunc(hdr
->frame_control
)) {
1061 /* No data, will not indicate to upper layer,
1062 temporily count it here */
1063 count_rx_stats(adapter
, precv_frame
, *psta
);
1064 ret
= RTW_RX_HANDLED
;
1068 u8
*myhwaddr
= myid(&adapter
->eeprompriv
);
1069 if (!ether_addr_equal(pattrib
->ra
, myhwaddr
)) {
1070 ret
= RTW_RX_HANDLED
;
1073 DBG_8723A("issue_deauth23a to sta =" MAC_FMT
" for the reason(7)\n",
1074 MAC_ARG(pattrib
->src
));
1075 issue_deauth23a(adapter
, pattrib
->src
,
1076 WLAN_REASON_CLASS3_FRAME_FROM_NONASSOC_STA
);
1077 ret
= RTW_RX_HANDLED
;
1088 static int validate_recv_ctrl_frame(struct rtw_adapter
*padapter
,
1089 struct recv_frame
*precv_frame
)
1091 #ifdef CONFIG_8723AU_AP_MODE
1092 struct rx_pkt_attrib
*pattrib
= &precv_frame
->attrib
;
1093 struct sta_priv
*pstapriv
= &padapter
->stapriv
;
1094 struct sk_buff
*skb
= precv_frame
->pkt
;
1095 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
1097 if (!ieee80211_is_ctl(hdr
->frame_control
))
1100 /* receive the frames that ra(a1) is my address */
1101 if (!ether_addr_equal(hdr
->addr1
, myid(&padapter
->eeprompriv
)))
1104 /* only handle ps-poll */
1105 if (ieee80211_is_pspoll(hdr
->frame_control
)) {
1106 struct ieee80211_pspoll
*psp
= (struct ieee80211_pspoll
*)hdr
;
1109 struct sta_info
*psta
= NULL
;
1111 aid
= le16_to_cpu(psp
->aid
) & 0x3fff;
1112 psta
= rtw_get_stainfo23a(pstapriv
, hdr
->addr2
);
1114 if (!psta
|| psta
->aid
!= aid
)
1117 /* for rx pkt statistics */
1118 psta
->sta_stats
.rx_ctrl_pkts
++;
1120 switch (pattrib
->priority
) {
1123 wmmps_ac
= psta
->uapsd_bk
& BIT(0);
1127 wmmps_ac
= psta
->uapsd_vi
& BIT(0);
1131 wmmps_ac
= psta
->uapsd_vo
& BIT(0);
1136 wmmps_ac
= psta
->uapsd_be
& BIT(0);
1143 if (psta
->state
& WIFI_STA_ALIVE_CHK_STATE
) {
1144 DBG_8723A("%s alive check-rx ps-poll\n", __func__
);
1145 psta
->expire_to
= pstapriv
->expire_to
;
1146 psta
->state
^= WIFI_STA_ALIVE_CHK_STATE
;
1149 if ((psta
->state
& WIFI_SLEEP_STATE
) &&
1150 (pstapriv
->sta_dz_bitmap
& CHKBIT(psta
->aid
))) {
1151 struct list_head
*xmitframe_plist
, *xmitframe_phead
;
1152 struct xmit_frame
*pxmitframe
;
1153 struct xmit_priv
*pxmitpriv
= &padapter
->xmitpriv
;
1155 spin_lock_bh(&pxmitpriv
->lock
);
1157 xmitframe_phead
= get_list_head(&psta
->sleep_q
);
1158 xmitframe_plist
= xmitframe_phead
->next
;
1160 if (!list_empty(xmitframe_phead
)) {
1161 pxmitframe
= container_of(xmitframe_plist
,
1165 xmitframe_plist
= xmitframe_plist
->next
;
1167 list_del_init(&pxmitframe
->list
);
1171 if (psta
->sleepq_len
>0)
1172 pxmitframe
->attrib
.mdata
= 1;
1174 pxmitframe
->attrib
.mdata
= 0;
1176 pxmitframe
->attrib
.triggered
= 1;
1178 /* DBG_8723A("handling ps-poll, q_len =%d, tim =%x\n", psta->sleepq_len, pstapriv->tim_bitmap); */
1180 rtl8723au_hal_xmitframe_enqueue(padapter
,
1183 if (psta
->sleepq_len
== 0) {
1184 pstapriv
->tim_bitmap
&= ~CHKBIT(psta
->aid
);
1186 /* DBG_8723A("after handling ps-poll, tim =%x\n", pstapriv->tim_bitmap); */
1188 /* upate BCN for TIM IE */
1189 /* update_BCNTIM(padapter); */
1190 update_beacon23a(padapter
, WLAN_EID_TIM
,
1194 /* spin_unlock_bh(&psta->sleep_q.lock); */
1195 spin_unlock_bh(&pxmitpriv
->lock
);
1198 /* spin_unlock_bh(&psta->sleep_q.lock); */
1199 spin_unlock_bh(&pxmitpriv
->lock
);
1201 /* DBG_8723A("no buffered packets to xmit\n"); */
1202 if (pstapriv
->tim_bitmap
& CHKBIT(psta
->aid
)) {
1203 if (psta
->sleepq_len
== 0) {
1204 DBG_8723A("no buffered packets "
1207 /* issue nulldata with More data bit = 0 to indicate we have no buffered packets */
1208 issue_nulldata23a(padapter
,
1212 DBG_8723A("error!psta->sleepq"
1215 psta
->sleepq_len
= 0;
1218 pstapriv
->tim_bitmap
&= ~CHKBIT(psta
->aid
);
1220 /* upate BCN for TIM IE */
1221 /* update_BCNTIM(padapter); */
1222 update_beacon23a(padapter
, WLAN_EID_TIM
,
1233 struct recv_frame
* recvframe_chk_defrag23a(struct rtw_adapter
*padapter
,
1234 struct recv_frame
*precv_frame
);
1235 static int validate_recv_mgnt_frame(struct rtw_adapter
*padapter
,
1236 struct recv_frame
*precv_frame
)
1238 struct sta_info
*psta
;
1239 struct sk_buff
*skb
;
1240 struct ieee80211_hdr
*hdr
;
1241 /* struct mlme_priv *pmlmepriv = &adapter->mlmepriv; */
1243 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1244 ("+validate_recv_mgnt_frame\n"));
1246 precv_frame
= recvframe_chk_defrag23a(padapter
, precv_frame
);
1247 if (precv_frame
== NULL
) {
1248 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
1249 ("%s: fragment packet\n", __func__
));
1253 skb
= precv_frame
->pkt
;
1254 hdr
= (struct ieee80211_hdr
*) skb
->data
;
1256 /* for rx pkt statistics */
1257 psta
= rtw_get_stainfo23a(&padapter
->stapriv
, hdr
->addr2
);
1259 psta
->sta_stats
.rx_mgnt_pkts
++;
1261 if (ieee80211_is_beacon(hdr
->frame_control
))
1262 psta
->sta_stats
.rx_beacon_pkts
++;
1263 else if (ieee80211_is_probe_req(hdr
->frame_control
))
1264 psta
->sta_stats
.rx_probereq_pkts
++;
1265 else if (ieee80211_is_probe_resp(hdr
->frame_control
)) {
1266 if (ether_addr_equal(padapter
->eeprompriv
.mac_addr
,
1268 psta
->sta_stats
.rx_probersp_pkts
++;
1269 else if (is_broadcast_ether_addr(hdr
->addr1
) ||
1270 is_multicast_ether_addr(hdr
->addr1
))
1271 psta
->sta_stats
.rx_probersp_bm_pkts
++;
1273 psta
->sta_stats
.rx_probersp_uo_pkts
++;
1277 mgt_dispatcher23a(padapter
, precv_frame
);
1282 static int validate_recv_data_frame(struct rtw_adapter
*adapter
,
1283 struct recv_frame
*precv_frame
)
1287 struct sta_info
*psta
= NULL
;
1288 struct rx_pkt_attrib
*pattrib
= & precv_frame
->attrib
;
1289 struct security_priv
*psecuritypriv
= &adapter
->securitypriv
;
1291 struct sk_buff
*skb
= precv_frame
->pkt
;
1292 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
1296 bretry
= ieee80211_has_retry(hdr
->frame_control
);
1297 pda
= ieee80211_get_DA(hdr
);
1298 psa
= ieee80211_get_SA(hdr
);
1300 ether_addr_copy(pattrib
->dst
, pda
);
1301 ether_addr_copy(pattrib
->src
, psa
);
1303 switch (hdr
->frame_control
&
1304 cpu_to_le16(IEEE80211_FCTL_TODS
| IEEE80211_FCTL_FROMDS
)) {
1305 case cpu_to_le16(0):
1306 ether_addr_copy(pattrib
->bssid
, hdr
->addr3
);
1307 ether_addr_copy(pattrib
->ra
, pda
);
1308 ether_addr_copy(pattrib
->ta
, psa
);
1309 ret
= sta2sta_data_frame(adapter
, precv_frame
, &psta
);
1312 case cpu_to_le16(IEEE80211_FCTL_FROMDS
):
1313 ether_addr_copy(pattrib
->bssid
, hdr
->addr2
);
1314 ether_addr_copy(pattrib
->ra
, pda
);
1315 ether_addr_copy(pattrib
->ta
, hdr
->addr2
);
1316 ret
= ap2sta_data_frame(adapter
, precv_frame
, &psta
);
1319 case cpu_to_le16(IEEE80211_FCTL_TODS
):
1320 ether_addr_copy(pattrib
->bssid
, hdr
->addr1
);
1321 ether_addr_copy(pattrib
->ra
, hdr
->addr1
);
1322 ether_addr_copy(pattrib
->ta
, psa
);
1323 ret
= sta2ap_data_frame(adapter
, precv_frame
, &psta
);
1326 case cpu_to_le16(IEEE80211_FCTL_TODS
| IEEE80211_FCTL_FROMDS
):
1328 * There is no BSSID in this case, but the driver has been
1329 * using addr1 so far, so keep it for now.
1331 ether_addr_copy(pattrib
->bssid
, hdr
->addr1
);
1332 ether_addr_copy(pattrib
->ra
, hdr
->addr1
);
1333 ether_addr_copy(pattrib
->ta
, hdr
->addr2
);
1335 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
, (" case 3\n"));
1339 if ((ret
== _FAIL
) || (ret
== RTW_RX_HANDLED
))
1343 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1344 (" after to_fr_ds_chk; psta == NULL\n"));
1349 /* psta->rssi = prxcmd->rssi; */
1350 /* psta->signal_quality = prxcmd->sq; */
1351 precv_frame
->psta
= psta
;
1353 pattrib
->hdrlen
= sizeof(struct ieee80211_hdr_3addr
);
1354 if (ieee80211_has_a4(hdr
->frame_control
))
1355 pattrib
->hdrlen
+= ETH_ALEN
;
1357 /* parsing QC field */
1358 if (pattrib
->qos
== 1) {
1359 __le16
*qptr
= (__le16
*)ieee80211_get_qos_ctl(hdr
);
1360 u16 qos_ctrl
= le16_to_cpu(*qptr
);
1362 pattrib
->priority
= qos_ctrl
& IEEE80211_QOS_CTL_TID_MASK
;
1363 pattrib
->ack_policy
= (qos_ctrl
>> 5) & 3;
1365 (qos_ctrl
& IEEE80211_QOS_CTL_A_MSDU_PRESENT
) >> 7;
1366 pattrib
->hdrlen
+= IEEE80211_QOS_CTL_LEN
;
1368 if (pattrib
->priority
!= 0 && pattrib
->priority
!= 3) {
1369 adapter
->recvpriv
.bIsAnyNonBEPkts
= true;
1372 pattrib
->priority
= 0;
1373 pattrib
->ack_policy
= 0;
1377 if (pattrib
->order
) { /* HT-CTRL 11n */
1378 pattrib
->hdrlen
+= 4;
1381 precv_frame
->preorder_ctrl
= &psta
->recvreorder_ctrl
[pattrib
->priority
];
1383 /* decache, drop duplicate recv packets */
1384 if (recv_decache(precv_frame
, bretry
, &psta
->sta_recvpriv
.rxcache
) ==
1386 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1387 ("decache : drop pkt\n"));
1392 if (pattrib
->privacy
) {
1393 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1394 ("validate_recv_data_frame:pattrib->privacy =%x\n",
1396 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1397 ("\n ^^^^^^^^^^^is_multicast_ether_addr"
1398 "(pattrib->ra(0x%02x)) =%d^^^^^^^^^^^^^^^6\n",
1400 is_multicast_ether_addr(pattrib
->ra
)));
1402 GET_ENCRY_ALGO(psecuritypriv
, psta
, pattrib
->encrypt
,
1403 is_multicast_ether_addr(pattrib
->ra
));
1405 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1406 ("\n pattrib->encrypt =%d\n", pattrib
->encrypt
));
1408 switch (pattrib
->encrypt
)
1410 case WLAN_CIPHER_SUITE_WEP40
:
1411 case WLAN_CIPHER_SUITE_WEP104
:
1412 pattrib
->iv_len
= IEEE80211_WEP_IV_LEN
;
1413 pattrib
->icv_len
= IEEE80211_WEP_ICV_LEN
;
1415 case WLAN_CIPHER_SUITE_TKIP
:
1416 pattrib
->iv_len
= IEEE80211_TKIP_IV_LEN
;
1417 pattrib
->icv_len
= IEEE80211_TKIP_ICV_LEN
;
1419 case WLAN_CIPHER_SUITE_CCMP
:
1420 pattrib
->iv_len
= IEEE80211_CCMP_HDR_LEN
;
1421 pattrib
->icv_len
= IEEE80211_CCMP_MIC_LEN
;
1424 pattrib
->iv_len
= 0;
1425 pattrib
->icv_len
= 0;
1429 pattrib
->encrypt
= 0;
1430 pattrib
->iv_len
= 0;
1431 pattrib
->icv_len
= 0;
1441 static void dump_rx_pkt(struct sk_buff
*skb
, u16 type
, int level
)
1447 ((level
== 2) && (type
== IEEE80211_FTYPE_MGMT
)) ||
1448 ((level
== 3) && (type
== IEEE80211_FTYPE_DATA
))) {
1452 DBG_8723A("#############################\n");
1454 for (i
= 0; i
< 64; i
= i
+ 8)
1455 DBG_8723A("%02X:%02X:%02X:%02X:%02X:%02X:%02X:%02X:\n",
1456 *(ptr
+ i
), *(ptr
+ i
+ 1), *(ptr
+ i
+ 2),
1457 *(ptr
+ i
+ 3), *(ptr
+ i
+ 4),
1458 *(ptr
+ i
+ 5), *(ptr
+ i
+ 6),
1460 DBG_8723A("#############################\n");
1464 static int validate_recv_frame(struct rtw_adapter
*adapter
,
1465 struct recv_frame
*precv_frame
)
1467 /* shall check frame subtype, to / from ds, da, bssid */
1469 /* then call check if rx seq/frag. duplicated. */
1472 int retval
= _SUCCESS
;
1473 struct rx_pkt_attrib
*pattrib
= & precv_frame
->attrib
;
1474 struct sk_buff
*skb
= precv_frame
->pkt
;
1475 struct ieee80211_hdr
*hdr
= (struct ieee80211_hdr
*) skb
->data
;
1480 fctl
= le16_to_cpu(hdr
->frame_control
);
1481 ver
= fctl
& IEEE80211_FCTL_VERS
;
1482 type
= fctl
& IEEE80211_FCTL_FTYPE
;
1483 subtype
= fctl
& IEEE80211_FCTL_STYPE
;
1485 /* add version chk */
1487 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1488 ("validate_recv_data_frame fail! (ver!= 0)\n"));
1493 seq_ctrl
= le16_to_cpu(hdr
->seq_ctrl
);
1494 pattrib
->frag_num
= seq_ctrl
& IEEE80211_SCTL_FRAG
;
1495 pattrib
->seq_num
= seq_ctrl
>> 4;
1497 pattrib
->pw_save
= ieee80211_has_pm(hdr
->frame_control
);
1498 pattrib
->mfrag
= ieee80211_has_morefrags(hdr
->frame_control
);
1499 pattrib
->mdata
= ieee80211_has_moredata(hdr
->frame_control
);
1500 pattrib
->privacy
= ieee80211_has_protected(hdr
->frame_control
);
1501 pattrib
->order
= ieee80211_has_order(hdr
->frame_control
);
1503 GetHalDefVar8192CUsb(adapter
, HAL_DEF_DBG_DUMP_RXPKT
, &bDumpRxPkt
);
1505 if (unlikely(bDumpRxPkt
== 1))
1506 dump_rx_pkt(skb
, type
, bDumpRxPkt
);
1510 case IEEE80211_FTYPE_MGMT
:
1511 retval
= validate_recv_mgnt_frame(adapter
, precv_frame
);
1512 if (retval
== _FAIL
) {
1513 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1514 ("validate_recv_mgnt_frame fail\n"));
1516 retval
= _FAIL
; /* only data frame return _SUCCESS */
1518 case IEEE80211_FTYPE_CTL
:
1519 retval
= validate_recv_ctrl_frame(adapter
, precv_frame
);
1520 if (retval
== _FAIL
) {
1521 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1522 ("validate_recv_ctrl_frame fail\n"));
1524 retval
= _FAIL
; /* only data frame return _SUCCESS */
1526 case IEEE80211_FTYPE_DATA
:
1527 rtw_led_control(adapter
, LED_CTL_RX
);
1528 pattrib
->qos
= (subtype
& IEEE80211_STYPE_QOS_DATA
) ? 1 : 0;
1529 retval
= validate_recv_data_frame(adapter
, precv_frame
);
1530 if (retval
== _FAIL
) {
1531 struct recv_priv
*precvpriv
= &adapter
->recvpriv
;
1532 /* RT_TRACE(_module_rtl871x_recv_c_, _drv_err_, ("validate_recv_data_frame fail\n")); */
1533 precvpriv
->rx_drop
++;
1537 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1538 ("validate_recv_data_frame fail! type = 0x%x\n", type
));
1547 /* remove the wlanhdr and add the eth_hdr */
1549 static int wlanhdr_to_ethhdr (struct recv_frame
*precvframe
)
1551 u16 eth_type
, len
, hdrlen
;
1556 struct rtw_adapter
*adapter
= precvframe
->adapter
;
1557 struct mlme_priv
*pmlmepriv
= &adapter
->mlmepriv
;
1559 struct sk_buff
*skb
= precvframe
->pkt
;
1561 struct rx_pkt_attrib
*pattrib
= &precvframe
->attrib
;
1566 hdrlen
= pattrib
->hdrlen
;
1567 psnap
= ptr
+ hdrlen
;
1568 eth_type
= (psnap
[6] << 8) | psnap
[7];
1569 /* convert hdr + possible LLC headers into Ethernet header */
1570 /* eth_type = (psnap_type[0] << 8) | psnap_type[1]; */
1571 if ((ether_addr_equal(psnap
, rfc1042_header
) &&
1572 eth_type
!= ETH_P_AARP
&& eth_type
!= ETH_P_IPX
) ||
1573 ether_addr_equal(psnap
, bridge_tunnel_header
)) {
1574 /* remove RFC1042 or Bridge-Tunnel encapsulation
1575 and replace EtherType */
1577 hdrlen
+= SNAP_SIZE
;
1579 /* Leave Ethernet header part of hdr and full payload */
1581 eth_type
= (psnap
[0] << 8) | psnap
[1];
1584 len
= skb
->len
- hdrlen
;
1586 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1587 ("\n === pattrib->hdrlen: %x, pattrib->iv_len:%x ===\n\n",
1588 pattrib
->hdrlen
, pattrib
->iv_len
));
1590 pattrib
->eth_type
= eth_type
;
1591 if (check_fwstate(pmlmepriv
, WIFI_MP_STATE
)) {
1597 /* append rx status for mp test packets */
1599 ptr
= skb_pull(skb
, (hdrlen
- sizeof(struct ethhdr
) + 2) - 24);
1600 memcpy(ptr
, skb
->head
, 24);
1603 ptr
= skb_pull(skb
, (hdrlen
- sizeof(struct ethhdr
) +
1607 ether_addr_copy(ptr
, pattrib
->dst
);
1608 ether_addr_copy(ptr
+ ETH_ALEN
, pattrib
->src
);
1612 memcpy(ptr
+ 12, &len
, 2);
1619 /* perform defrag */
1620 struct recv_frame
*recvframe_defrag(struct rtw_adapter
*adapter
,
1621 struct rtw_queue
*defrag_q
);
1622 struct recv_frame
*recvframe_defrag(struct rtw_adapter
*adapter
,
1623 struct rtw_queue
*defrag_q
)
1625 struct list_head
*plist
, *phead
, *ptmp
;
1626 u8
*data
, wlanhdr_offset
;
1628 struct recv_frame
*pnfhdr
;
1629 struct recv_frame
*prframe
, *pnextrframe
;
1630 struct rtw_queue
*pfree_recv_queue
;
1631 struct sk_buff
*skb
;
1636 pfree_recv_queue
= &adapter
->recvpriv
.free_recv_queue
;
1638 phead
= get_list_head(defrag_q
);
1639 plist
= phead
->next
;
1640 prframe
= container_of(plist
, struct recv_frame
, list
);
1641 list_del_init(&prframe
->list
);
1644 if (curfragnum
!= prframe
->attrib
.frag_num
) {
1645 /* the first fragment number must be 0 */
1646 /* free the whole queue */
1647 rtw_free_recvframe23a(prframe
);
1648 rtw_free_recvframe23a_queue(defrag_q
);
1655 phead
= get_list_head(defrag_q
);
1657 data
= prframe
->pkt
->data
;
1659 list_for_each_safe(plist
, ptmp
, phead
) {
1660 pnfhdr
= container_of(plist
, struct recv_frame
, list
);
1661 pnextrframe
= (struct recv_frame
*)pnfhdr
;
1662 /* check the fragment sequence (2nd ~n fragment frame) */
1664 if (curfragnum
!= pnfhdr
->attrib
.frag_num
) {
1665 /* the fragment number must be increasing
1667 /* release the defrag_q & prframe */
1668 rtw_free_recvframe23a(prframe
);
1669 rtw_free_recvframe23a_queue(defrag_q
);
1675 /* copy the 2nd~n fragment frame's payload to the
1677 /* get the 2nd~last fragment frame's payload */
1679 wlanhdr_offset
= pnfhdr
->attrib
.hdrlen
+ pnfhdr
->attrib
.iv_len
;
1681 skb_pull(pnfhdr
->pkt
, wlanhdr_offset
);
1683 /* append to first fragment frame's tail
1684 (if privacy frame, pull the ICV) */
1686 skb_trim(skb
, skb
->len
- prframe
->attrib
.icv_len
);
1688 memcpy(skb_tail_pointer(skb
), pnfhdr
->pkt
->data
,
1691 skb_put(skb
, pnfhdr
->pkt
->len
);
1693 prframe
->attrib
.icv_len
= pnfhdr
->attrib
.icv_len
;
1696 /* free the defrag_q queue and return the prframe */
1697 rtw_free_recvframe23a_queue(defrag_q
);
1699 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1700 ("Performance defrag!!!!!\n"));
1707 /* check if need to defrag, if needed queue the frame to defrag_q */
1708 struct recv_frame
* recvframe_chk_defrag23a(struct rtw_adapter
*padapter
,
1709 struct recv_frame
*precv_frame
)
1714 struct recv_frame
*pfhdr
;
1715 struct sta_info
*psta
;
1716 struct sta_priv
*pstapriv
;
1717 struct list_head
*phead
;
1718 struct recv_frame
*prtnframe
= NULL
;
1719 struct rtw_queue
*pfree_recv_queue
, *pdefrag_q
;
1723 pstapriv
= &padapter
->stapriv
;
1725 pfhdr
= precv_frame
;
1727 pfree_recv_queue
= &padapter
->recvpriv
.free_recv_queue
;
1729 /* need to define struct of wlan header frame ctrl */
1730 ismfrag
= pfhdr
->attrib
.mfrag
;
1731 fragnum
= pfhdr
->attrib
.frag_num
;
1733 psta_addr
= pfhdr
->attrib
.ta
;
1734 psta
= rtw_get_stainfo23a(pstapriv
, psta_addr
);
1736 struct ieee80211_hdr
*hdr
=
1737 (struct ieee80211_hdr
*) pfhdr
->pkt
->data
;
1738 if (!ieee80211_is_data(hdr
->frame_control
)) {
1739 psta
= rtw_get_bcmc_stainfo23a(padapter
);
1740 pdefrag_q
= &psta
->sta_recvpriv
.defrag_q
;
1744 pdefrag_q
= &psta
->sta_recvpriv
.defrag_q
;
1746 if ((ismfrag
== 0) && (fragnum
== 0)) {
1747 prtnframe
= precv_frame
;/* isn't a fragment frame */
1751 /* 0~(n-1) fragment frame */
1752 /* enqueue to defraf_g */
1753 if (pdefrag_q
!= NULL
) {
1755 /* the first fragment */
1756 if (!list_empty(&pdefrag_q
->queue
)) {
1757 /* free current defrag_q */
1758 rtw_free_recvframe23a_queue(pdefrag_q
);
1762 /* Then enqueue the 0~(n-1) fragment into the
1765 /* spin_lock(&pdefrag_q->lock); */
1766 phead
= get_list_head(pdefrag_q
);
1767 list_add_tail(&pfhdr
->list
, phead
);
1768 /* spin_unlock(&pdefrag_q->lock); */
1770 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1771 ("Enqueuq: ismfrag = %d, fragnum = %d\n",
1777 /* can't find this ta's defrag_queue,
1778 so free this recv_frame */
1779 rtw_free_recvframe23a(precv_frame
);
1781 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1782 ("Free because pdefrag_q == NULL: ismfrag = "
1783 "%d, fragnum = %d\n", ismfrag
, fragnum
));
1787 if ((ismfrag
== 0) && (fragnum
!= 0)) {
1788 /* the last fragment frame */
1789 /* enqueue the last fragment */
1790 if (pdefrag_q
!= NULL
) {
1791 /* spin_lock(&pdefrag_q->lock); */
1792 phead
= get_list_head(pdefrag_q
);
1793 list_add_tail(&pfhdr
->list
, phead
);
1794 /* spin_unlock(&pdefrag_q->lock); */
1796 /* call recvframe_defrag to defrag */
1797 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
1798 ("defrag: ismfrag = %d, fragnum = %d\n",
1800 precv_frame
= recvframe_defrag(padapter
, pdefrag_q
);
1801 prtnframe
= precv_frame
;
1803 /* can't find this ta's defrag_queue,
1804 so free this recv_frame */
1805 rtw_free_recvframe23a(precv_frame
);
1807 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1808 ("Free because pdefrag_q == NULL: ismfrag = "
1809 "%d, fragnum = %d\n", ismfrag
, fragnum
));
1814 if ((prtnframe
!= NULL
) && (prtnframe
->attrib
.privacy
)) {
1815 /* after defrag we must check tkip mic code */
1816 if (recvframe_chkmic(padapter
, prtnframe
) == _FAIL
) {
1817 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
1818 ("recvframe_chkmic(padapter, prtnframe) =="
1820 rtw_free_recvframe23a(prtnframe
);
1830 int amsdu_to_msdu(struct rtw_adapter
*padapter
, struct recv_frame
*prframe
);
1831 int amsdu_to_msdu(struct rtw_adapter
*padapter
, struct recv_frame
*prframe
)
1833 struct rx_pkt_attrib
*pattrib
;
1834 struct sk_buff
*skb
, *sub_skb
;
1835 struct sk_buff_head skb_list
;
1837 pattrib
= &prframe
->attrib
;
1840 skb_pull(skb
, prframe
->attrib
.hdrlen
);
1841 __skb_queue_head_init(&skb_list
);
1843 ieee80211_amsdu_to_8023s(skb
, &skb_list
, NULL
, 0, 0, false);
1845 while (!skb_queue_empty(&skb_list
)) {
1846 sub_skb
= __skb_dequeue(&skb_list
);
1848 sub_skb
->protocol
= eth_type_trans(sub_skb
, padapter
->pnetdev
);
1849 sub_skb
->dev
= padapter
->pnetdev
;
1851 sub_skb
->ip_summed
= CHECKSUM_NONE
;
1856 prframe
->pkt
= NULL
;
1857 rtw_free_recvframe23a(prframe
);
1861 int check_indicate_seq(struct recv_reorder_ctrl
*preorder_ctrl
, u16 seq_num
);
1862 int check_indicate_seq(struct recv_reorder_ctrl
*preorder_ctrl
, u16 seq_num
)
1864 u8 wsize
= preorder_ctrl
->wsize_b
;
1865 u16 wend
= (preorder_ctrl
->indicate_seq
+ wsize
-1) & 0xFFF;
1867 /* Rx Reorder initialize condition. */
1868 if (preorder_ctrl
->indicate_seq
== 0xFFFF)
1869 preorder_ctrl
->indicate_seq
= seq_num
;
1871 /* Drop out the packet which SeqNum is smaller than WinStart */
1872 if (SN_LESS(seq_num
, preorder_ctrl
->indicate_seq
))
1876 /* Sliding window manipulation. Conditions includes: */
1877 /* 1. Incoming SeqNum is equal to WinStart =>Window shift 1 */
1878 /* 2. Incoming SeqNum is larger than the WinEnd => Window shift N */
1880 if (SN_EQUAL(seq_num
, preorder_ctrl
->indicate_seq
)) {
1881 preorder_ctrl
->indicate_seq
=
1882 (preorder_ctrl
->indicate_seq
+ 1) & 0xFFF;
1883 } else if (SN_LESS(wend
, seq_num
)) {
1884 /* boundary situation, when seq_num cross 0xFFF */
1885 if (seq_num
>= (wsize
- 1))
1886 preorder_ctrl
->indicate_seq
= seq_num
+ 1 -wsize
;
1888 preorder_ctrl
->indicate_seq
= 0xFFF - (wsize
- (seq_num
+ 1)) + 1;
1893 static int enqueue_reorder_recvframe23a(struct recv_reorder_ctrl
*preorder_ctrl
,
1894 struct recv_frame
*prframe
)
1896 struct rx_pkt_attrib
*pattrib
= &prframe
->attrib
;
1897 struct rtw_queue
*ppending_recvframe_queue
;
1898 struct list_head
*phead
, *plist
, *ptmp
;
1899 struct recv_frame
*hdr
;
1900 struct rx_pkt_attrib
*pnextattrib
;
1902 ppending_recvframe_queue
= &preorder_ctrl
->pending_recvframe_queue
;
1903 /* DbgPrint("+enqueue_reorder_recvframe23a()\n"); */
1905 /* spin_lock_irqsave(&ppending_recvframe_queue->lock); */
1906 /* spin_lock_ex(&ppending_recvframe_queue->lock); */
1908 phead
= get_list_head(ppending_recvframe_queue
);
1910 list_for_each_safe(plist
, ptmp
, phead
) {
1911 hdr
= container_of(plist
, struct recv_frame
, list
);
1912 pnextattrib
= &hdr
->attrib
;
1914 if (SN_LESS(pnextattrib
->seq_num
, pattrib
->seq_num
)) {
1916 } else if (SN_EQUAL(pnextattrib
->seq_num
, pattrib
->seq_num
)) {
1917 /* Duplicate entry is found!! Do not insert current entry. */
1918 /* RT_TRACE(COMP_RX_REORDER, DBG_TRACE, ("InsertRxReorderList(): Duplicate packet is dropped!! IndicateSeq: %d, NewSeq: %d\n", pTS->RxIndicateSeq, SeqNum)); */
1920 /* spin_unlock_irqrestore(&ppending_recvframe_queue->lock); */
1926 /* DbgPrint("enqueue_reorder_recvframe23a():while\n"); */
1929 /* spin_lock_irqsave(&ppending_recvframe_queue->lock); */
1930 /* spin_lock_ex(&ppending_recvframe_queue->lock); */
1932 list_del_init(&prframe
->list
);
1934 list_add_tail(&prframe
->list
, plist
);
1936 /* spin_unlock_ex(&ppending_recvframe_queue->lock); */
1937 /* spin_unlock_irqrestore(&ppending_recvframe_queue->lock); */
1939 /* RT_TRACE(COMP_RX_REORDER, DBG_TRACE, ("InsertRxReorderList(): Pkt insert into buffer!! IndicateSeq: %d, NewSeq: %d\n", pTS->RxIndicateSeq, SeqNum)); */
1943 int recv_indicatepkts_in_order(struct rtw_adapter
*padapter
,
1944 struct recv_reorder_ctrl
*preorder_ctrl
,
1946 int recv_indicatepkts_in_order(struct rtw_adapter
*padapter
,
1947 struct recv_reorder_ctrl
*preorder_ctrl
,
1950 /* u8 bcancelled; */
1951 struct list_head
*phead
, *plist
;
1952 struct recv_frame
*prframe
;
1953 struct rx_pkt_attrib
*pattrib
;
1955 int bPktInBuf
= false;
1956 struct recv_priv
*precvpriv
;
1957 struct rtw_queue
*ppending_recvframe_queue
;
1959 precvpriv
= &padapter
->recvpriv
;
1960 ppending_recvframe_queue
= &preorder_ctrl
->pending_recvframe_queue
;
1961 /* DbgPrint("+recv_indicatepkts_in_order\n"); */
1963 /* spin_lock_irqsave(&ppending_recvframe_queue->lock); */
1964 /* spin_lock_ex(&ppending_recvframe_queue->lock); */
1966 phead
= get_list_head(ppending_recvframe_queue
);
1967 plist
= phead
->next
;
1969 /* Handling some condition for forced indicate case. */
1971 if (list_empty(phead
)) {
1972 /* spin_unlock_irqrestore(&ppending_recvframe_queue->lock); */
1973 /* spin_unlock_ex(&ppending_recvframe_queue->lock); */
1977 prframe
= container_of(plist
, struct recv_frame
, list
);
1978 pattrib
= &prframe
->attrib
;
1979 preorder_ctrl
->indicate_seq
= pattrib
->seq_num
;
1982 /* Prepare indication list and indication. */
1983 /* Check if there is any packet need indicate. */
1984 while (!list_empty(phead
)) {
1986 prframe
= container_of(plist
, struct recv_frame
, list
);
1987 pattrib
= &prframe
->attrib
;
1989 if (!SN_LESS(preorder_ctrl
->indicate_seq
, pattrib
->seq_num
)) {
1990 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
1991 ("recv_indicatepkts_in_order: indicate =%d "
1992 "seq =%d amsdu =%d\n",
1993 preorder_ctrl
->indicate_seq
,
1994 pattrib
->seq_num
, pattrib
->amsdu
));
1996 plist
= plist
->next
;
1997 list_del_init(&prframe
->list
);
1999 if (SN_EQUAL(preorder_ctrl
->indicate_seq
,
2000 pattrib
->seq_num
)) {
2001 preorder_ctrl
->indicate_seq
=
2002 (preorder_ctrl
->indicate_seq
+ 1)&0xFFF;
2005 if (!pattrib
->amsdu
) {
2006 if ((padapter
->bDriverStopped
== false) &&
2007 (padapter
->bSurpriseRemoved
== false)) {
2008 rtw_recv_indicatepkt23a(padapter
, prframe
);
2011 if (amsdu_to_msdu(padapter
, prframe
) !=
2013 rtw_free_recvframe23a(prframe
);
2016 /* Update local variables. */
2024 /* DbgPrint("recv_indicatepkts_in_order():while\n"); */
2027 /* spin_unlock_ex(&ppending_recvframe_queue->lock); */
2028 /* spin_unlock_irqrestore(&ppending_recvframe_queue->lock); */
2033 int recv_indicatepkt_reorder(struct rtw_adapter
*padapter
,
2034 struct recv_frame
*prframe
);
2035 int recv_indicatepkt_reorder(struct rtw_adapter
*padapter
,
2036 struct recv_frame
*prframe
)
2038 int retval
= _SUCCESS
;
2039 struct rx_pkt_attrib
*pattrib
;
2040 struct recv_reorder_ctrl
*preorder_ctrl
;
2041 struct rtw_queue
*ppending_recvframe_queue
;
2043 pattrib
= &prframe
->attrib
;
2044 preorder_ctrl
= prframe
->preorder_ctrl
;
2045 ppending_recvframe_queue
= &preorder_ctrl
->pending_recvframe_queue
;
2047 if (!pattrib
->amsdu
) {
2049 wlanhdr_to_ethhdr(prframe
);
2051 if ((pattrib
->qos
!= 1) || (pattrib
->eth_type
== ETH_P_ARP
) ||
2052 (pattrib
->ack_policy
!= 0)) {
2053 if ((padapter
->bDriverStopped
== false) &&
2054 (padapter
->bSurpriseRemoved
== false)) {
2055 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
2056 ("@@@@ recv_indicatepkt_reorder -"
2057 "recv_func recv_indicatepkt\n"));
2059 rtw_recv_indicatepkt23a(padapter
, prframe
);
2066 if (preorder_ctrl
->enable
== false) {
2067 /* indicate this recv_frame */
2068 preorder_ctrl
->indicate_seq
= pattrib
->seq_num
;
2069 rtw_recv_indicatepkt23a(padapter
, prframe
);
2071 preorder_ctrl
->indicate_seq
=
2072 (preorder_ctrl
->indicate_seq
+ 1) % 4096;
2076 /* temp filter -> means didn't support A-MSDUs in a A-MPDU */
2077 if (preorder_ctrl
->enable
== false) {
2078 preorder_ctrl
->indicate_seq
= pattrib
->seq_num
;
2079 retval
= amsdu_to_msdu(padapter
, prframe
);
2081 preorder_ctrl
->indicate_seq
=
2082 (preorder_ctrl
->indicate_seq
+ 1) % 4096;
2087 spin_lock_bh(&ppending_recvframe_queue
->lock
);
2089 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
2090 ("recv_indicatepkt_reorder: indicate =%d seq =%d\n",
2091 preorder_ctrl
->indicate_seq
, pattrib
->seq_num
));
2093 /* s2. check if winstart_b(indicate_seq) needs to been updated */
2094 if (!check_indicate_seq(preorder_ctrl
, pattrib
->seq_num
)) {
2098 /* s3. Insert all packet into Reorder Queue to maintain its ordering. */
2099 if (!enqueue_reorder_recvframe23a(preorder_ctrl
, prframe
)) {
2104 /* Indication process. */
2105 /* After Packet dropping and Sliding Window shifting as above,
2106 we can now just indicate the packets */
2107 /* with the SeqNum smaller than latest WinStart and buffer
2110 /* For Rx Reorder condition: */
2111 /* 1. All packets with SeqNum smaller than WinStart => Indicate */
2112 /* 2. All packets with SeqNum larger than or equal to WinStart =>
2116 if (recv_indicatepkts_in_order(padapter
, preorder_ctrl
, false) == true) {
2117 mod_timer(&preorder_ctrl
->reordering_ctrl_timer
,
2118 jiffies
+ msecs_to_jiffies(REORDER_WAIT_TIME
));
2119 spin_unlock_bh(&ppending_recvframe_queue
->lock
);
2121 spin_unlock_bh(&ppending_recvframe_queue
->lock
);
2122 del_timer_sync(&preorder_ctrl
->reordering_ctrl_timer
);
2128 spin_unlock_bh(&ppending_recvframe_queue
->lock
);
2132 void rtw_reordering_ctrl_timeout_handler23a(unsigned long pcontext
)
2134 struct recv_reorder_ctrl
*preorder_ctrl
;
2135 struct rtw_adapter
*padapter
;
2136 struct rtw_queue
*ppending_recvframe_queue
;
2138 preorder_ctrl
= (struct recv_reorder_ctrl
*)pcontext
;
2139 padapter
= preorder_ctrl
->padapter
;
2140 ppending_recvframe_queue
= &preorder_ctrl
->pending_recvframe_queue
;
2142 if (padapter
->bDriverStopped
|| padapter
->bSurpriseRemoved
) {
2146 /* DBG_8723A("+rtw_reordering_ctrl_timeout_handler23a() =>\n"); */
2148 spin_lock_bh(&ppending_recvframe_queue
->lock
);
2150 if (recv_indicatepkts_in_order(padapter
, preorder_ctrl
, true) == true) {
2151 mod_timer(&preorder_ctrl
->reordering_ctrl_timer
,
2152 jiffies
+ msecs_to_jiffies(REORDER_WAIT_TIME
));
2155 spin_unlock_bh(&ppending_recvframe_queue
->lock
);
2158 int process_recv_indicatepkts(struct rtw_adapter
*padapter
,
2159 struct recv_frame
*prframe
);
2160 int process_recv_indicatepkts(struct rtw_adapter
*padapter
,
2161 struct recv_frame
*prframe
)
2163 int retval
= _SUCCESS
;
2164 /* struct recv_priv *precvpriv = &padapter->recvpriv; */
2165 /* struct rx_pkt_attrib *pattrib = &prframe->attrib; */
2166 struct mlme_priv
*pmlmepriv
= &padapter
->mlmepriv
;
2167 struct ht_priv
*phtpriv
= &pmlmepriv
->htpriv
;
2169 if (phtpriv
->ht_option
== true) { /* B/G/N Mode */
2170 /* prframe->preorder_ctrl = &precvpriv->recvreorder_ctrl[pattrib->priority]; */
2172 /* including perform A-MPDU Rx Ordering Buffer Control */
2173 if (recv_indicatepkt_reorder(padapter
, prframe
) != _SUCCESS
) {
2174 if ((padapter
->bDriverStopped
== false) &&
2175 (padapter
->bSurpriseRemoved
== false)) {
2180 } else /* B/G mode */
2182 retval
= wlanhdr_to_ethhdr(prframe
);
2183 if (retval
!= _SUCCESS
) {
2184 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
2185 ("wlanhdr_to_ethhdr: drop pkt\n"));
2189 if ((padapter
->bDriverStopped
== false) &&
2190 (padapter
->bSurpriseRemoved
== false)) {
2191 /* indicate this recv_frame */
2192 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
2193 ("@@@@ process_recv_indicatepkts- "
2194 "recv_func recv_indicatepkt\n"));
2195 rtw_recv_indicatepkt23a(padapter
, prframe
);
2197 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
2198 ("@@@@ process_recv_indicatepkts- "
2199 "recv_func free_indicatepkt\n"));
2201 RT_TRACE(_module_rtl871x_recv_c_
, _drv_notice_
,
2202 ("recv_func:bDriverStopped(%d) OR "
2203 "bSurpriseRemoved(%d)",
2204 padapter
->bDriverStopped
,
2205 padapter
->bSurpriseRemoved
));
2215 static int recv_func_prehandle(struct rtw_adapter
*padapter
,
2216 struct recv_frame
*rframe
)
2220 /* check the frame crtl field and decache */
2221 ret
= validate_recv_frame(padapter
, rframe
);
2222 if (ret
!= _SUCCESS
) {
2223 RT_TRACE(_module_rtl871x_recv_c_
, _drv_info_
,
2224 ("recv_func: validate_recv_frame fail! drop pkt\n"));
2225 rtw_free_recvframe23a(rframe
);
2233 static int recv_func_posthandle(struct rtw_adapter
*padapter
,
2234 struct recv_frame
*prframe
)
2237 struct recv_frame
*orig_prframe
= prframe
;
2238 struct recv_priv
*precvpriv
= &padapter
->recvpriv
;
2241 rtw_led_control(padapter
, LED_CTL_RX
);
2243 prframe
= decryptor(padapter
, prframe
);
2244 if (prframe
== NULL
) {
2245 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
2246 ("decryptor: drop pkt\n"));
2248 goto _recv_data_drop
;
2251 prframe
= recvframe_chk_defrag23a(padapter
, prframe
);
2253 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
2254 ("recvframe_chk_defrag23a: drop pkt\n"));
2255 goto _recv_data_drop
;
2259 * Pull off crypto headers
2261 if (prframe
->attrib
.iv_len
> 0) {
2262 skb_pull(prframe
->pkt
, prframe
->attrib
.iv_len
);
2265 if (prframe
->attrib
.icv_len
> 0) {
2266 skb_trim(prframe
->pkt
,
2267 prframe
->pkt
->len
- prframe
->attrib
.icv_len
);
2270 prframe
= portctrl(padapter
, prframe
);
2272 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
2273 ("portctrl: drop pkt\n"));
2275 goto _recv_data_drop
;
2278 count_rx_stats(padapter
, prframe
, NULL
);
2280 ret
= process_recv_indicatepkts(padapter
, prframe
);
2281 if (ret
!= _SUCCESS
) {
2282 RT_TRACE(_module_rtl871x_recv_c_
, _drv_err_
,
2283 ("recv_func: process_recv_indicatepkts fail!\n"));
2284 rtw_free_recvframe23a(orig_prframe
);/* free this recv_frame */
2285 goto _recv_data_drop
;
2290 precvpriv
->rx_drop
++;
2294 int rtw_recv_entry23a(struct recv_frame
*rframe
)
2297 struct rtw_adapter
*padapter
= rframe
->adapter
;
2298 struct rx_pkt_attrib
*prxattrib
= &rframe
->attrib
;
2299 struct recv_priv
*recvpriv
= &padapter
->recvpriv
;
2300 struct security_priv
*psecuritypriv
= &padapter
->securitypriv
;
2301 struct mlme_priv
*mlmepriv
= &padapter
->mlmepriv
;
2303 /* check if need to handle uc_swdec_pending_queue*/
2304 if (check_fwstate(mlmepriv
, WIFI_STATION_STATE
) &&
2305 psecuritypriv
->busetkipkey
) {
2306 struct recv_frame
*pending_frame
;
2308 while ((pending_frame
= rtw_alloc_recvframe23a(&padapter
->recvpriv
.uc_swdec_pending_queue
))) {
2309 r
= recv_func_posthandle(padapter
, pending_frame
);
2311 DBG_8723A("%s: dequeue uc_swdec_pending_queue\n", __func__
);
2315 ret
= recv_func_prehandle(padapter
, rframe
);
2317 if (ret
== _SUCCESS
) {
2318 /* check if need to enqueue into uc_swdec_pending_queue*/
2319 if (check_fwstate(mlmepriv
, WIFI_STATION_STATE
) &&
2320 !is_multicast_ether_addr(prxattrib
->ra
) &&
2321 prxattrib
->encrypt
> 0 &&
2322 (prxattrib
->bdecrypted
== 0) &&
2323 !is_wep_enc(psecuritypriv
->dot11PrivacyAlgrthm
) &&
2324 !psecuritypriv
->busetkipkey
) {
2325 rtw_enqueue_recvframe23a(rframe
, &padapter
->recvpriv
.uc_swdec_pending_queue
);
2326 DBG_8723A("%s: no key, enqueue uc_swdec_pending_queue\n", __func__
);
2330 ret
= recv_func_posthandle(padapter
, rframe
);
2332 recvpriv
->rx_pkts
++;
2339 void rtw_signal_stat_timer_hdl23a(unsigned long data
)
2341 struct rtw_adapter
*adapter
= (struct rtw_adapter
*)data
;
2342 struct recv_priv
*recvpriv
= &adapter
->recvpriv
;
2345 u8 avg_signal_strength
= 0;
2346 u8 avg_signal_qual
= 0;
2347 u32 num_signal_strength
= 0;
2348 u32 num_signal_qual
= 0;
2349 u8 _alpha
= 3; /* this value is based on converging_constant = 5000 */
2350 /* and sampling_interval = 1000 */
2352 if (adapter
->recvpriv
.is_signal_dbg
) {
2353 /* update the user specific value, signal_strength_dbg, */
2354 /* to signal_strength, rssi */
2355 adapter
->recvpriv
.signal_strength
=
2356 adapter
->recvpriv
.signal_strength_dbg
;
2357 adapter
->recvpriv
.rssi
=
2358 (s8
)translate_percentage_to_dbm((u8
)adapter
->recvpriv
.signal_strength_dbg
);
2360 if (recvpriv
->signal_strength_data
.update_req
== 0) {
2361 /* update_req is clear, means we got rx */
2362 avg_signal_strength
=
2363 recvpriv
->signal_strength_data
.avg_val
;
2364 num_signal_strength
=
2365 recvpriv
->signal_strength_data
.total_num
;
2366 /* after avg_vals are accquired, we can re-stat */
2367 /* the signal values */
2368 recvpriv
->signal_strength_data
.update_req
= 1;
2371 if (recvpriv
->signal_qual_data
.update_req
== 0) {
2372 /* update_req is clear, means we got rx */
2373 avg_signal_qual
= recvpriv
->signal_qual_data
.avg_val
;
2374 num_signal_qual
= recvpriv
->signal_qual_data
.total_num
;
2375 /* after avg_vals are accquired, we can re-stat */
2376 /*the signal values */
2377 recvpriv
->signal_qual_data
.update_req
= 1;
2380 /* update value of signal_strength, rssi, signal_qual */
2381 if (!check_fwstate(&adapter
->mlmepriv
, _FW_UNDER_SURVEY
)) {
2382 tmp_s
= (avg_signal_strength
+ (_alpha
- 1) *
2383 recvpriv
->signal_strength
);
2385 tmp_s
= tmp_s
/ _alpha
+ 1;
2387 tmp_s
= tmp_s
/ _alpha
;
2391 tmp_q
= (avg_signal_qual
+ (_alpha
- 1) *
2392 recvpriv
->signal_qual
);
2394 tmp_q
= tmp_q
/ _alpha
+ 1;
2396 tmp_q
= tmp_q
/ _alpha
;
2400 recvpriv
->signal_strength
= tmp_s
;
2401 recvpriv
->rssi
= (s8
)translate_percentage_to_dbm(tmp_s
);
2402 recvpriv
->signal_qual
= tmp_q
;
2404 DBG_8723A("%s signal_strength:%3u, rssi:%3d, "
2405 "signal_qual:%3u, num_signal_strength:%u, "
2406 "num_signal_qual:%u\n",
2407 __func__
, recvpriv
->signal_strength
,
2408 recvpriv
->rssi
, recvpriv
->signal_qual
,
2409 num_signal_strength
, num_signal_qual
2413 rtw_set_signal_stat_timer(recvpriv
);