Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
nu11secur1ty
GitHub Repository: nu11secur1ty/Kali-Linux
Path: blob/master/ALFA-W1F1/RTL8814AU/os_dep/linux/xmit_linux.c
1307 views
1
/******************************************************************************
2
*
3
* Copyright(c) 2007 - 2017 Realtek Corporation.
4
*
5
* This program is free software; you can redistribute it and/or modify it
6
* under the terms of version 2 of the GNU General Public License as
7
* published by the Free Software Foundation.
8
*
9
* This program is distributed in the hope that it will be useful, but WITHOUT
10
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
12
* more details.
13
*
14
*****************************************************************************/
15
#define _XMIT_OSDEP_C_
16
17
#include <drv_types.h>
18
19
#define DBG_DUMP_OS_QUEUE_CTL 0
20
21
uint rtw_remainder_len(struct pkt_file *pfile)
22
{
23
return pfile->buf_len - ((SIZE_PTR)(pfile->cur_addr) - (SIZE_PTR)(pfile->buf_start));
24
}
25
26
void _rtw_open_pktfile(_pkt *pktptr, struct pkt_file *pfile)
27
{
28
29
pfile->pkt = pktptr;
30
pfile->cur_addr = pfile->buf_start = pktptr->data;
31
pfile->pkt_len = pfile->buf_len = pktptr->len;
32
33
pfile->cur_buffer = pfile->buf_start ;
34
35
}
36
37
uint _rtw_pktfile_read(struct pkt_file *pfile, u8 *rmem, uint rlen)
38
{
39
uint len = 0;
40
41
42
len = rtw_remainder_len(pfile);
43
len = (rlen > len) ? len : rlen;
44
45
if (rmem)
46
skb_copy_bits(pfile->pkt, pfile->buf_len - pfile->pkt_len, rmem, len);
47
48
pfile->cur_addr += len;
49
pfile->pkt_len -= len;
50
51
52
return len;
53
}
54
55
sint rtw_endofpktfile(struct pkt_file *pfile)
56
{
57
58
if (pfile->pkt_len == 0) {
59
return _TRUE;
60
}
61
62
63
return _FALSE;
64
}
65
66
void rtw_set_tx_chksum_offload(_pkt *pkt, struct pkt_attrib *pattrib)
67
{
68
#ifdef CONFIG_TCP_CSUM_OFFLOAD_TX
69
struct sk_buff *skb = (struct sk_buff *)pkt;
70
struct iphdr *iph = NULL;
71
struct ipv6hdr *i6ph = NULL;
72
struct udphdr *uh = NULL;
73
struct tcphdr *th = NULL;
74
u8 protocol = 0xFF;
75
76
if (skb->protocol == htons(ETH_P_IP)) {
77
iph = (struct iphdr *)skb_network_header(skb);
78
protocol = iph->protocol;
79
} else if (skb->protocol == htons(ETH_P_IPV6)) {
80
i6ph = (struct ipv6hdr *)skb_network_header(skb);
81
protocol = i6ph->nexthdr;
82
} else
83
{}
84
85
/* HW unable to compute CSUM if header & payload was be encrypted by SW(cause TXDMA error) */
86
if (pattrib->bswenc == _TRUE) {
87
if (skb->ip_summed == CHECKSUM_PARTIAL)
88
skb_checksum_help(skb);
89
return;
90
}
91
92
/* For HW rule, clear ipv4_csum & UDP/TCP_csum if it is UDP/TCP packet */
93
switch (protocol) {
94
case IPPROTO_UDP:
95
uh = (struct udphdr *)skb_transport_header(skb);
96
uh->check = 0;
97
if (iph)
98
iph->check = 0;
99
pattrib->hw_csum = _TRUE;
100
break;
101
case IPPROTO_TCP:
102
th = (struct tcphdr *)skb_transport_header(skb);
103
th->check = 0;
104
if (iph)
105
iph->check = 0;
106
pattrib->hw_csum = _TRUE;
107
break;
108
default:
109
break;
110
}
111
#endif
112
113
}
114
115
int rtw_os_xmit_resource_alloc(_adapter *padapter, struct xmit_buf *pxmitbuf, u32 alloc_sz, u8 flag)
116
{
117
if (alloc_sz > 0) {
118
#ifdef CONFIG_USE_USB_BUFFER_ALLOC_TX
119
struct dvobj_priv *pdvobjpriv = adapter_to_dvobj(padapter);
120
struct usb_device *pusbd = pdvobjpriv->pusbdev;
121
122
pxmitbuf->pallocated_buf = rtw_usb_buffer_alloc(pusbd, (size_t)alloc_sz, &pxmitbuf->dma_transfer_addr);
123
pxmitbuf->pbuf = pxmitbuf->pallocated_buf;
124
if (pxmitbuf->pallocated_buf == NULL)
125
return _FAIL;
126
#else /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
127
128
pxmitbuf->pallocated_buf = rtw_zmalloc(alloc_sz);
129
if (pxmitbuf->pallocated_buf == NULL)
130
return _FAIL;
131
132
pxmitbuf->pbuf = (u8 *)N_BYTE_ALIGMENT((SIZE_PTR)(pxmitbuf->pallocated_buf), XMITBUF_ALIGN_SZ);
133
134
#endif /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
135
}
136
137
if (flag) {
138
#ifdef CONFIG_USB_HCI
139
int i;
140
for (i = 0; i < 8; i++) {
141
pxmitbuf->pxmit_urb[i] = usb_alloc_urb(0, GFP_KERNEL);
142
if (pxmitbuf->pxmit_urb[i] == NULL) {
143
RTW_INFO("pxmitbuf->pxmit_urb[i]==NULL");
144
return _FAIL;
145
}
146
}
147
#endif
148
}
149
150
return _SUCCESS;
151
}
152
153
void rtw_os_xmit_resource_free(_adapter *padapter, struct xmit_buf *pxmitbuf, u32 free_sz, u8 flag)
154
{
155
if (flag) {
156
#ifdef CONFIG_USB_HCI
157
int i;
158
159
for (i = 0; i < 8; i++) {
160
if (pxmitbuf->pxmit_urb[i]) {
161
/* usb_kill_urb(pxmitbuf->pxmit_urb[i]); */
162
usb_free_urb(pxmitbuf->pxmit_urb[i]);
163
}
164
}
165
#endif
166
}
167
168
if (free_sz > 0) {
169
#ifdef CONFIG_USE_USB_BUFFER_ALLOC_TX
170
struct dvobj_priv *pdvobjpriv = adapter_to_dvobj(padapter);
171
struct usb_device *pusbd = pdvobjpriv->pusbdev;
172
173
rtw_usb_buffer_free(pusbd, (size_t)free_sz, pxmitbuf->pallocated_buf, pxmitbuf->dma_transfer_addr);
174
pxmitbuf->pallocated_buf = NULL;
175
pxmitbuf->dma_transfer_addr = 0;
176
#else /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
177
if (pxmitbuf->pallocated_buf)
178
rtw_mfree(pxmitbuf->pallocated_buf, free_sz);
179
#endif /* CONFIG_USE_USB_BUFFER_ALLOC_TX */
180
}
181
}
182
183
void dump_os_queue(void *sel, _adapter *padapter)
184
{
185
struct net_device *ndev = padapter->pnetdev;
186
187
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
188
int i;
189
190
for (i = 0; i < 4; i++) {
191
RTW_PRINT_SEL(sel, "os_queue[%d]:%s\n"
192
, i, __netif_subqueue_stopped(ndev, i) ? "stopped" : "waked");
193
}
194
#else
195
RTW_PRINT_SEL(sel, "os_queue:%s\n"
196
, netif_queue_stopped(ndev) ? "stopped" : "waked");
197
#endif
198
}
199
200
#define WMM_XMIT_THRESHOLD (NR_XMITFRAME*2/5)
201
202
static inline bool rtw_os_need_wake_queue(_adapter *padapter, u16 qidx)
203
{
204
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
205
struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
206
207
if (padapter->registrypriv.wifi_spec) {
208
if (pxmitpriv->hwxmits[qidx].accnt < WMM_XMIT_THRESHOLD)
209
return _TRUE;
210
#ifdef DBG_CONFIG_ERROR_DETECT
211
#ifdef DBG_CONFIG_ERROR_RESET
212
} else if (rtw_hal_sreset_inprogress(padapter) == _TRUE) {
213
return _FALSE;
214
#endif/* #ifdef DBG_CONFIG_ERROR_RESET */
215
#endif/* #ifdef DBG_CONFIG_ERROR_DETECT */
216
} else {
217
#ifdef CONFIG_MCC_MODE
218
if (MCC_EN(padapter)) {
219
if (rtw_hal_check_mcc_status(padapter, MCC_STATUS_DOING_MCC)
220
&& MCC_STOP(padapter))
221
return _FALSE;
222
}
223
#endif /* CONFIG_MCC_MODE */
224
return _TRUE;
225
}
226
return _FALSE;
227
#else
228
#ifdef CONFIG_MCC_MODE
229
if (MCC_EN(padapter)) {
230
if (rtw_hal_check_mcc_status(padapter, MCC_STATUS_DOING_MCC)
231
&& MCC_STOP(padapter))
232
return _FALSE;
233
}
234
#endif /* CONFIG_MCC_MODE */
235
return _TRUE;
236
#endif
237
}
238
239
static inline bool rtw_os_need_stop_queue(_adapter *padapter, u16 qidx)
240
{
241
struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
242
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
243
if (padapter->registrypriv.wifi_spec) {
244
/* No free space for Tx, tx_worker is too slow */
245
if (pxmitpriv->hwxmits[qidx].accnt > WMM_XMIT_THRESHOLD)
246
return _TRUE;
247
} else {
248
if (pxmitpriv->free_xmitframe_cnt <= 4)
249
return _TRUE;
250
}
251
#else
252
if (pxmitpriv->free_xmitframe_cnt <= 4)
253
return _TRUE;
254
#endif
255
return _FALSE;
256
}
257
258
void rtw_os_pkt_complete(_adapter *padapter, _pkt *pkt)
259
{
260
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
261
u16 qidx;
262
263
qidx = skb_get_queue_mapping(pkt);
264
if (rtw_os_need_wake_queue(padapter, qidx)) {
265
if (DBG_DUMP_OS_QUEUE_CTL)
266
RTW_INFO(FUNC_ADPT_FMT": netif_wake_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), qidx);
267
netif_wake_subqueue(padapter->pnetdev, qidx);
268
}
269
#else
270
if (rtw_os_need_wake_queue(padapter, 0)) {
271
if (DBG_DUMP_OS_QUEUE_CTL)
272
RTW_INFO(FUNC_ADPT_FMT": netif_wake_queue\n", FUNC_ADPT_ARG(padapter));
273
netif_wake_queue(padapter->pnetdev);
274
}
275
#endif
276
277
rtw_skb_free(pkt);
278
}
279
280
void rtw_os_xmit_complete(_adapter *padapter, struct xmit_frame *pxframe)
281
{
282
if (pxframe->pkt)
283
rtw_os_pkt_complete(padapter, pxframe->pkt);
284
285
pxframe->pkt = NULL;
286
}
287
288
void rtw_os_xmit_schedule(_adapter *padapter)
289
{
290
#if defined(CONFIG_SDIO_HCI) || defined(CONFIG_GSPI_HCI)
291
_adapter *pri_adapter;
292
293
if (!padapter)
294
return;
295
pri_adapter = GET_PRIMARY_ADAPTER(padapter);
296
297
if (_rtw_queue_empty(&padapter->xmitpriv.pending_xmitbuf_queue) == _FALSE)
298
_rtw_up_sema(&pri_adapter->xmitpriv.xmit_sema);
299
300
301
#else
302
_irqL irqL;
303
struct xmit_priv *pxmitpriv;
304
305
if (!padapter)
306
return;
307
308
pxmitpriv = &padapter->xmitpriv;
309
310
_enter_critical_bh(&pxmitpriv->lock, &irqL);
311
312
if (rtw_txframes_pending(padapter))
313
tasklet_hi_schedule(&pxmitpriv->xmit_tasklet);
314
315
_exit_critical_bh(&pxmitpriv->lock, &irqL);
316
317
#if defined(CONFIG_PCI_HCI) && defined(CONFIG_XMIT_THREAD_MODE)
318
if (_rtw_queue_empty(&padapter->xmitpriv.pending_xmitbuf_queue) == _FALSE)
319
_rtw_up_sema(&padapter->xmitpriv.xmit_sema);
320
#endif
321
322
323
#endif
324
}
325
326
static bool rtw_check_xmit_resource(_adapter *padapter, _pkt *pkt)
327
{
328
bool busy = _FALSE;
329
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
330
u16 qidx;
331
332
qidx = skb_get_queue_mapping(pkt);
333
if (rtw_os_need_stop_queue(padapter, qidx)) {
334
if (DBG_DUMP_OS_QUEUE_CTL)
335
RTW_INFO(FUNC_ADPT_FMT": netif_stop_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), qidx);
336
netif_stop_subqueue(padapter->pnetdev, qidx);
337
busy = _TRUE;
338
}
339
#else
340
if (rtw_os_need_stop_queue(padapter, 0)) {
341
if (DBG_DUMP_OS_QUEUE_CTL)
342
RTW_INFO(FUNC_ADPT_FMT": netif_stop_queue\n", FUNC_ADPT_ARG(padapter));
343
rtw_netif_stop_queue(padapter->pnetdev);
344
busy = _TRUE;
345
}
346
#endif
347
return busy;
348
}
349
350
void rtw_os_wake_queue_at_free_stainfo(_adapter *padapter, int *qcnt_freed)
351
{
352
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 35))
353
int i;
354
355
for (i = 0; i < 4; i++) {
356
if (qcnt_freed[i] == 0)
357
continue;
358
359
if (rtw_os_need_wake_queue(padapter, i)) {
360
if (DBG_DUMP_OS_QUEUE_CTL)
361
RTW_INFO(FUNC_ADPT_FMT": netif_wake_subqueue[%d]\n", FUNC_ADPT_ARG(padapter), i);
362
netif_wake_subqueue(padapter->pnetdev, i);
363
}
364
}
365
#else
366
if (qcnt_freed[0] || qcnt_freed[1] || qcnt_freed[2] || qcnt_freed[3]) {
367
if (rtw_os_need_wake_queue(padapter, 0)) {
368
if (DBG_DUMP_OS_QUEUE_CTL)
369
RTW_INFO(FUNC_ADPT_FMT": netif_wake_queue\n", FUNC_ADPT_ARG(padapter));
370
netif_wake_queue(padapter->pnetdev);
371
}
372
}
373
#endif
374
}
375
376
#ifdef CONFIG_TX_MCAST2UNI
377
int rtw_mlcst2unicst(_adapter *padapter, struct sk_buff *skb)
378
{
379
struct sta_priv *pstapriv = &padapter->stapriv;
380
struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
381
_irqL irqL;
382
_list *phead, *plist;
383
struct sk_buff *newskb;
384
struct sta_info *psta = NULL;
385
u8 chk_alive_num = 0;
386
char chk_alive_list[NUM_STA];
387
u8 bc_addr[6] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
388
u8 null_addr[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
389
390
int i;
391
s32 res;
392
393
DBG_COUNTER(padapter->tx_logs.os_tx_m2u);
394
395
_enter_critical_bh(&pstapriv->asoc_list_lock, &irqL);
396
phead = &pstapriv->asoc_list;
397
plist = get_next(phead);
398
399
/* free sta asoc_queue */
400
while ((rtw_end_of_queue_search(phead, plist)) == _FALSE) {
401
int stainfo_offset;
402
psta = LIST_CONTAINOR(plist, struct sta_info, asoc_list);
403
plist = get_next(plist);
404
405
stainfo_offset = rtw_stainfo_offset(pstapriv, psta);
406
if (stainfo_offset_valid(stainfo_offset))
407
chk_alive_list[chk_alive_num++] = stainfo_offset;
408
}
409
_exit_critical_bh(&pstapriv->asoc_list_lock, &irqL);
410
411
for (i = 0; i < chk_alive_num; i++) {
412
psta = rtw_get_stainfo_by_offset(pstapriv, chk_alive_list[i]);
413
if (!(psta->state & _FW_LINKED)) {
414
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_ignore_fw_linked);
415
continue;
416
}
417
418
/* avoid come from STA1 and send back STA1 */
419
if (_rtw_memcmp(psta->cmn.mac_addr, &skb->data[6], ETH_ALEN) == _TRUE
420
|| _rtw_memcmp(psta->cmn.mac_addr, null_addr, ETH_ALEN) == _TRUE
421
|| _rtw_memcmp(psta->cmn.mac_addr, bc_addr, ETH_ALEN) == _TRUE
422
) {
423
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_ignore_self);
424
continue;
425
}
426
427
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry);
428
429
newskb = rtw_skb_copy(skb);
430
431
if (newskb) {
432
_rtw_memcpy(newskb->data, psta->cmn.mac_addr, ETH_ALEN);
433
res = rtw_xmit(padapter, &newskb);
434
if (res < 0) {
435
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry_err_xmit);
436
RTW_INFO("%s()-%d: rtw_xmit() return error! res=%d\n", __FUNCTION__, __LINE__, res);
437
pxmitpriv->tx_drop++;
438
rtw_skb_free(newskb);
439
}
440
} else {
441
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_entry_err_skb);
442
RTW_INFO("%s-%d: rtw_skb_copy() failed!\n", __FUNCTION__, __LINE__);
443
pxmitpriv->tx_drop++;
444
/* rtw_skb_free(skb); */
445
return _FALSE; /* Caller shall tx this multicast frame via normal way. */
446
}
447
}
448
449
rtw_skb_free(skb);
450
return _TRUE;
451
}
452
#endif /* CONFIG_TX_MCAST2UNI */
453
454
455
int _rtw_xmit_entry(_pkt *pkt, _nic_hdl pnetdev)
456
{
457
_adapter *padapter = (_adapter *)rtw_netdev_priv(pnetdev);
458
struct xmit_priv *pxmitpriv = &padapter->xmitpriv;
459
#ifdef CONFIG_TX_MCAST2UNI
460
extern int rtw_mc2u_disable;
461
#endif /* CONFIG_TX_MCAST2UNI */
462
#ifdef CONFIG_TCP_CSUM_OFFLOAD_TX
463
struct sk_buff *skb = pkt;
464
struct sk_buff *segs, *nskb;
465
netdev_features_t features = padapter->pnetdev->features;
466
#endif
467
s32 res = 0;
468
469
if (padapter->registrypriv.mp_mode) {
470
RTW_INFO("MP_TX_DROP_OS_FRAME\n");
471
goto drop_packet;
472
}
473
DBG_COUNTER(padapter->tx_logs.os_tx);
474
475
if (rtw_if_up(padapter) == _FALSE) {
476
DBG_COUNTER(padapter->tx_logs.os_tx_err_up);
477
#ifdef DBG_TX_DROP_FRAME
478
RTW_INFO("DBG_TX_DROP_FRAME %s if_up fail\n", __FUNCTION__);
479
#endif
480
goto drop_packet;
481
}
482
483
rtw_check_xmit_resource(padapter, pkt);
484
485
#ifdef CONFIG_TX_MCAST2UNI
486
if (!rtw_mc2u_disable
487
&& MLME_IS_AP(padapter)
488
&& (IP_MCAST_MAC(pkt->data)
489
|| ICMPV6_MCAST_MAC(pkt->data)
490
#ifdef CONFIG_TX_BCAST2UNI
491
|| is_broadcast_mac_addr(pkt->data)
492
#endif
493
)
494
&& (padapter->registrypriv.wifi_spec == 0)
495
) {
496
if (pxmitpriv->free_xmitframe_cnt > (NR_XMITFRAME / 4)) {
497
res = rtw_mlcst2unicst(padapter, pkt);
498
if (res == _TRUE)
499
goto exit;
500
} else {
501
/* RTW_INFO("Stop M2U(%d, %d)! ", pxmitpriv->free_xmitframe_cnt, pxmitpriv->free_xmitbuf_cnt); */
502
/* RTW_INFO("!m2u ); */
503
DBG_COUNTER(padapter->tx_logs.os_tx_m2u_stop);
504
}
505
}
506
#endif /* CONFIG_TX_MCAST2UNI */
507
508
#ifdef CONFIG_TCP_CSUM_OFFLOAD_TX
509
if (skb_shinfo(skb)->gso_size) {
510
/* split a big(65k) skb into several small(1.5k) skbs */
511
features &= ~(NETIF_F_TSO | NETIF_F_TSO6);
512
segs = skb_gso_segment(skb, features);
513
if (IS_ERR(segs) || !segs)
514
goto drop_packet;
515
516
do {
517
nskb = segs;
518
segs = segs->next;
519
nskb->next = NULL;
520
rtw_mstat_update( MSTAT_TYPE_SKB, MSTAT_ALLOC_SUCCESS, nskb->truesize);
521
res = rtw_xmit(padapter, &nskb);
522
if (res < 0) {
523
#ifdef DBG_TX_DROP_FRAME
524
RTW_INFO("DBG_TX_DROP_FRAME %s rtw_xmit fail\n", __FUNCTION__);
525
#endif
526
pxmitpriv->tx_drop++;
527
rtw_os_pkt_complete(padapter, nskb);
528
}
529
} while (segs);
530
rtw_os_pkt_complete(padapter, skb);
531
goto exit;
532
}
533
#endif
534
535
res = rtw_xmit(padapter, &pkt);
536
if (res < 0) {
537
#ifdef DBG_TX_DROP_FRAME
538
RTW_INFO("DBG_TX_DROP_FRAME %s rtw_xmit fail\n", __FUNCTION__);
539
#endif
540
goto drop_packet;
541
}
542
543
goto exit;
544
545
drop_packet:
546
pxmitpriv->tx_drop++;
547
rtw_os_pkt_complete(padapter, pkt);
548
549
exit:
550
551
552
return 0;
553
}
554
555
#ifdef CONFIG_CUSTOMER_ALIBABA_GENERAL
556
int check_alibaba_meshpkt(struct sk_buff *skb)
557
{
558
u16 protocol;
559
560
if (skb)
561
return (htons(skb->protocol) == ETH_P_ALL);
562
563
return _FALSE;
564
}
565
566
s32 rtw_alibaba_mesh_xmit_entry(_pkt *pkt, struct net_device *ndev)
567
{
568
u16 frame_ctl;
569
570
_adapter *padapter = (_adapter *)rtw_netdev_priv(ndev);
571
struct pkt_file pktfile;
572
struct rtw_ieee80211_hdr *pwlanhdr;
573
struct pkt_attrib *pattrib;
574
struct xmit_frame *pmgntframe;
575
struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
576
struct xmit_priv *pxmitpriv = &(padapter->xmitpriv);
577
unsigned char *pframe;
578
struct sk_buff *skb = (struct sk_buff *)pkt;
579
int len = skb->len;
580
581
rtw_mstat_update(MSTAT_TYPE_SKB, MSTAT_ALLOC_SUCCESS, skb->truesize);
582
583
pmgntframe = alloc_mgtxmitframe(pxmitpriv);
584
if (pmgntframe == NULL) {
585
goto fail;
586
return -1;
587
}
588
589
pattrib = &pmgntframe->attrib;
590
update_mgntframe_attrib(padapter, pattrib);
591
_rtw_memset(pmgntframe->buf_addr, 0, WLANHDR_OFFSET + TXDESC_OFFSET);
592
pframe = (u8 *)(pmgntframe->buf_addr) + TXDESC_OFFSET;
593
pwlanhdr = (struct rtw_ieee80211_hdr *)pframe;
594
595
_rtw_open_pktfile(pkt, &pktfile);
596
_rtw_pktfile_read(&pktfile, pframe, len);
597
598
pattrib->type = pframe[0] & 0x0C;
599
pattrib->subtype = pframe[0] & 0xF0;
600
pattrib->raid = rtw_get_mgntframe_raid(padapter, WIRELESS_11G);
601
pattrib->rate = MGN_24M;
602
pattrib->pktlen = len;
603
SetSeqNum(pwlanhdr, pmlmeext->mgnt_seq);
604
pmlmeext->mgnt_seq++;
605
606
RTW_DBG_DUMP("rtw_alibaba_mesh_xmit_entry payload:", skb->data, len);
607
608
pattrib->last_txcmdsz = pattrib->pktlen;
609
dump_mgntframe(padapter, pmgntframe);
610
611
fail:
612
rtw_skb_free(skb);
613
return 0;
614
}
615
#endif
616
617
int rtw_xmit_entry(_pkt *pkt, _nic_hdl pnetdev)
618
{
619
_adapter *padapter = (_adapter *)rtw_netdev_priv(pnetdev);
620
struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
621
int ret = 0;
622
623
if (pkt) {
624
#ifdef CONFIG_CUSTOMER_ALIBABA_GENERAL
625
if (check_alibaba_meshpkt(pkt)) {
626
return rtw_alibaba_mesh_xmit_entry(pkt, pnetdev);
627
}
628
#endif
629
if (check_fwstate(pmlmepriv, WIFI_MONITOR_STATE) == _TRUE) {
630
#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 24))
631
ret = rtw_monitor_xmit_entry((struct sk_buff *)pkt, pnetdev);
632
#endif
633
} else {
634
rtw_mstat_update(MSTAT_TYPE_SKB, MSTAT_ALLOC_SUCCESS, pkt->truesize);
635
ret = _rtw_xmit_entry(pkt, pnetdev);
636
}
637
638
}
639
640
return ret;
641
}
642
643