1 /*
2 * rx.c
3 *
4 * Copyright(c) 1998 - 2009 Texas Instruments. All rights reserved.
5 * All rights reserved.
6 *
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
9 * are met:
10 *
11 * * Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * * Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in
15 * the documentation and/or other materials provided with the
16 * distribution.
17 * * Neither the name Texas Instruments nor the names of its
18 * contributors may be used to endorse or promote products derived
19 * from this software without specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
24 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
25 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
26 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
27 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
28 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
29 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
30 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
31 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
32 */
33
34 /***************************************************************************/
35 /* */
36 /* MODULE: Rx.c */
37 /* PURPOSE: Rx module functions */
38 /* */
39 /***************************************************************************/
40 #define __FILE_ID__ FILE_ID_54
41 #include "tidef.h"
42 #include "paramOut.h"
43 #include "rx.h"
44 #include "osApi.h"
45 #include "timer.h"
46 #include "DataCtrl_Api.h"
47 #include "Ctrl.h"
48 #include "802_11Defs.h"
49 #include "Ethernet.h"
50 #include "report.h"
51 #include "rate.h"
52 #include "mlmeApi.h"
53 #include "rsnApi.h"
54 #include "smeApi.h"
55 #include "siteMgrApi.h"
56 #include "GeneralUtil.h"
57 #include "EvHandler.h"
58 #ifdef XCC_MODULE_INCLUDED
59 #include "XCCMngr.h"
60 #endif
61 #include "TWDriver.h"
62 #include "RxBuf.h"
63 #include "DrvMainModules.h"
64 #include "bmtrace_api.h"
65 #include "PowerMgr_API.h"
66
67
68 #define EAPOL_PACKET 0x888E
69 #define IAPP_PACKET 0x0000
70 #define PREAUTH_EAPOL_PACKET 0x88C7
71
72
73 #define RX_DATA_FILTER_FLAG_NO_BIT_MASK 0
74 #define RX_DATA_FILTER_FLAG_USE_BIT_MASK 1
75 #define RX_DATA_FILTER_FLAG_IP_HEADER 0
76 #define RX_DATA_FILTER_FLAG_ETHERNET_HEADER 2
77 #define RX_DATA_FILTER_ETHERNET_HEADER_BOUNDARY 14
78
79 #define PADDING_ETH_PACKET_SIZE 2
80
81 /* CallBack for recieving packet from rxXfer */
82 static void rxData_ReceivePacket (TI_HANDLE hRxData, void *pBuffer);
83
84 static ERxBufferStatus rxData_RequestForBuffer (TI_HANDLE hRxData, void **pBuf, TI_UINT16 aLength, TI_UINT32 uEncryptionFlag,PacketClassTag_e ePacketClassTag);
85
86 #if 0
87 static TI_STATUS rxData_checkBssIdAndBssType(TI_HANDLE hRxData,
88 dot11_header_t* dot11_header,
89 TMacAddr **rxBssid,
90 ScanBssType_e *currBssType,
91 TMacAddr *currBssId);
92 #endif
93 static TI_STATUS rxData_convertWlanToEthHeader (TI_HANDLE hRxData, void *pBuffer, TI_UINT16 * etherType);
94 static TI_STATUS rxData_ConvertAmsduToEthPackets (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
95 static void rxData_dataPacketDisptcher (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
96 static void rxData_discardPacket (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
97 static void rxData_discardPacketVlan (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
98 static void rxData_rcvPacketInOpenNotify (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
99 static void rxData_rcvPacketEapol (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
100 static void rxData_rcvPacketData (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
101
102 static TI_STATUS rxData_enableDisableRxDataFilters(TI_HANDLE hRxData, TI_BOOL enabled);
103 static TI_STATUS rxData_addRxDataFilter(TI_HANDLE hRxData, TRxDataFilterRequest* request);
104 static TI_STATUS rxData_removeRxDataFilter(TI_HANDLE hRxData, TRxDataFilterRequest* request);
105
106
107 #ifdef XCC_MODULE_INCLUDED
108 static void rxData_rcvPacketIapp(TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr);
109 #endif
110 #ifdef TI_DBG
111 static void rxData_printRxThroughput(TI_HANDLE hRxData, TI_BOOL bTwdInitOccured);
112 #endif
113
114 static void rxData_StartReAuthActiveTimer(TI_HANDLE hRxData);
115 static void reAuthTimeout(TI_HANDLE hRxData, TI_BOOL bTwdInitOccured);
116 static void rxData_ReauthEnablePriority(TI_HANDLE hRxData);
117
118
119 /*************************************************************************
120 * rxData_create *
121 **************************************************************************
122 * DESCRIPTION: This function initializes the Rx data module.
123 *
124 * INPUT: hOs - handle to Os Abstraction Layer
125 *
126 * OUTPUT:
127 *
128 * RETURN: Handle to the allocated Rx data control block
129 ************************************************************************/
rxData_create(TI_HANDLE hOs)130 TI_HANDLE rxData_create (TI_HANDLE hOs)
131 {
132 rxData_t *pRxData;
133
134 /* check parameters validity */
135 if (hOs == NULL)
136 {
137 WLAN_OS_REPORT(("FATAL ERROR: rxData_create(): OS handle Error - Aborting\n"));
138 return NULL;
139 }
140
141
142 /* alocate Rx module control block */
143 pRxData = os_memoryAlloc(hOs, (sizeof(rxData_t)));
144
145 if (!pRxData)
146 {
147 WLAN_OS_REPORT(("FATAL ERROR: rxData_create(): Error Creating Rx Module - Aborting\n"));
148 return NULL;
149 }
150
151 /* reset Rx control block */
152 os_memoryZero (hOs, pRxData, (sizeof(rxData_t)));
153
154 pRxData->RxEventDistributor = DistributorMgr_Create (hOs, MAX_RX_NOTIF_REQ_ELMENTS);
155
156 pRxData->hOs = hOs;
157
158 return (TI_HANDLE)pRxData;
159 }
160
161 /***************************************************************************
162 * rxData_config *
163 ****************************************************************************
164 * DESCRIPTION: This function configures the Rx Data module
165 *
166 * INPUTS: pStadHandles - The driver modules handles
167 *
168 * OUTPUT:
169 *
170 * RETURNS: void
171 ***************************************************************************/
rxData_init(TStadHandlesList * pStadHandles)172 void rxData_init (TStadHandlesList *pStadHandles)
173 {
174 rxData_t *pRxData = (rxData_t *)(pStadHandles->hRxData);
175
176 pRxData->hCtrlData = pStadHandles->hCtrlData;
177 pRxData->hTWD = pStadHandles->hTWD;
178 pRxData->hMlme = pStadHandles->hMlmeSm;
179 pRxData->hRsn = pStadHandles->hRsn;
180 pRxData->hSiteMgr = pStadHandles->hSiteMgr;
181 pRxData->hOs = pStadHandles->hOs;
182 pRxData->hReport = pStadHandles->hReport;
183 pRxData->hXCCMgr = pStadHandles->hXCCMngr;
184 pRxData->hEvHandler = pStadHandles->hEvHandler;
185 pRxData->hTimer = pStadHandles->hTimer;
186 pRxData->hPowerMgr = pStadHandles->hPowerMgr;
187
188 pRxData->rxDataExcludeUnencrypted = DEF_EXCLUDE_UNENCYPTED;
189 pRxData->rxDataExludeBroadcastUnencrypted = DEF_EXCLUDE_UNENCYPTED;
190 pRxData->rxDataEapolDestination = DEF_EAPOL_DESTINATION;
191 pRxData->rxDataPortStatus = DEF_RX_PORT_STATUS;
192
193 /*
194 * configure rx data dispatcher
195 */
196
197 /* port status close */
198 pRxData->rxData_dispatchBuffer[CLOSE][DATA_DATA_PACKET] = &rxData_discardPacket; /* data */
199 pRxData->rxData_dispatchBuffer[CLOSE][DATA_EAPOL_PACKET] = &rxData_discardPacket; /* eapol */
200 pRxData->rxData_dispatchBuffer[CLOSE][DATA_IAPP_PACKET] = &rxData_discardPacket; /* Iapp */
201 pRxData->rxData_dispatchBuffer[CLOSE][DATA_VLAN_PACKET] = &rxData_discardPacketVlan; /* VLAN */
202
203 /* port status open notify */
204 pRxData->rxData_dispatchBuffer[OPEN_NOTIFY][DATA_DATA_PACKET] = &rxData_rcvPacketInOpenNotify; /* data */
205 pRxData->rxData_dispatchBuffer[OPEN_NOTIFY][DATA_EAPOL_PACKET] = &rxData_rcvPacketInOpenNotify; /* eapol */
206 pRxData->rxData_dispatchBuffer[OPEN_NOTIFY][DATA_IAPP_PACKET] = &rxData_rcvPacketInOpenNotify; /* Iapp */
207 pRxData->rxData_dispatchBuffer[OPEN_NOTIFY][DATA_VLAN_PACKET] = &rxData_discardPacketVlan; /* VLAN */
208
209 /* port status open eapol */
210 pRxData->rxData_dispatchBuffer[OPEN_EAPOL][DATA_DATA_PACKET] = &rxData_discardPacket; /* data */
211 pRxData->rxData_dispatchBuffer[OPEN_EAPOL][DATA_EAPOL_PACKET] = &rxData_rcvPacketEapol; /* eapol */
212 pRxData->rxData_dispatchBuffer[OPEN_EAPOL][DATA_IAPP_PACKET] = &rxData_discardPacket; /* Iapp */
213 pRxData->rxData_dispatchBuffer[OPEN_EAPOL][DATA_VLAN_PACKET] = &rxData_discardPacketVlan; /* VLAN */
214
215 /* port status open */
216 pRxData->rxData_dispatchBuffer[OPEN][DATA_DATA_PACKET] = &rxData_rcvPacketData; /* data */
217 pRxData->rxData_dispatchBuffer[OPEN][DATA_EAPOL_PACKET] = &rxData_rcvPacketEapol; /* eapol */
218 #ifdef XCC_MODULE_INCLUDED
219 pRxData->rxData_dispatchBuffer[OPEN][DATA_IAPP_PACKET] = &rxData_rcvPacketIapp; /* Iapp */
220 #else
221 pRxData->rxData_dispatchBuffer[OPEN][DATA_IAPP_PACKET] = &rxData_rcvPacketData; /* Iapp */
222 #endif
223 pRxData->rxData_dispatchBuffer[OPEN][DATA_VLAN_PACKET] = &rxData_discardPacketVlan;/* VLAN */
224
225 /* register CB's for request buffer and receive CB to the lower layers */
226 TWD_RegisterCb (pRxData->hTWD,
227 TWD_EVENT_RX_RECEIVE_PACKET,
228 (void *)rxData_ReceivePacket,
229 pStadHandles->hRxData);
230
231 TWD_RegisterCb (pRxData->hTWD,
232 TWD_EVENT_RX_REQUEST_FOR_BUFFER,
233 (void*)rxData_RequestForBuffer,
234 pStadHandles->hRxData);
235 }
236
237
rxData_SetDefaults(TI_HANDLE hRxData,rxDataInitParams_t * rxDataInitParams)238 TI_STATUS rxData_SetDefaults (TI_HANDLE hRxData, rxDataInitParams_t * rxDataInitParams)
239 {
240 rxData_t *pRxData = (rxData_t *)hRxData;
241 int i;
242
243 /* init rx data filters */
244 pRxData->filteringEnabled = rxDataInitParams->rxDataFiltersEnabled;
245 pRxData->filteringDefaultAction = rxDataInitParams->rxDataFiltersDefaultAction;
246 TWD_CfgEnableRxDataFilter (pRxData->hTWD, pRxData->filteringEnabled, pRxData->filteringDefaultAction);
247
248 for (i = 0; i < MAX_DATA_FILTERS; ++i)
249 {
250 if (rxDataInitParams->rxDataFilterRequests[i].maskLength > 0)
251 {
252 if (rxData_addRxDataFilter(hRxData, &rxDataInitParams->rxDataFilterRequests[i]) != TI_OK)
253 {
254 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR, ": Invalid Rx Data Filter configured at init stage (at index %d)!\n", i);
255 }
256 }
257 }
258
259 pRxData->reAuthActiveTimer = tmr_CreateTimer (pRxData->hTimer);
260 if (pRxData->reAuthActiveTimer == NULL)
261 {
262 WLAN_OS_REPORT(("rxData_SetDefaults(): Failed to create reAuthActiveTimer!\n"));
263 return TI_NOK;
264 }
265
266 pRxData->reAuthActiveTimeout = rxDataInitParams->reAuthActiveTimeout;
267
268 rxData_SetReAuthInProgress(pRxData, TI_FALSE);
269
270 #ifdef TI_DBG
271 /* reset counters */
272 rxData_resetCounters(pRxData);
273 rxData_resetDbgCounters(pRxData);
274
275 /* allocate timer for debug throughput */
276 pRxData->hThroughputTimer = tmr_CreateTimer (pRxData->hTimer);
277 if (pRxData->hThroughputTimer == NULL)
278 {
279 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_SetDefaults(): Failed to create hThroughputTimer!\n");
280 return TI_NOK;
281 }
282 pRxData->rxThroughputTimerEnable = TI_FALSE;
283 #endif
284
285
286 TRACE0(pRxData->hReport, REPORT_SEVERITY_INIT, ".....Rx Data configured successfully\n");
287
288 return TI_OK;
289 }
290
291 /***************************************************************************
292 * rxData_unLoad *
293 ****************************************************************************
294 * DESCRIPTION: This function unload the Rx data module.
295 *
296 * INPUTS: hRxData - the object
297 *
298 * OUTPUT:
299 *
300 * RETURNS: TI_OK - Unload succesfull
301 * TI_NOK - Unload unsuccesfull
302 ***************************************************************************/
rxData_unLoad(TI_HANDLE hRxData)303 TI_STATUS rxData_unLoad(TI_HANDLE hRxData)
304 {
305 rxData_t *pRxData = (rxData_t *)hRxData;
306
307 /* check parameters validity */
308 if (pRxData == NULL)
309 {
310 return TI_NOK;
311 }
312
313 DistributorMgr_Destroy(pRxData->RxEventDistributor);
314
315 #ifdef TI_DBG
316 /* destroy periodic rx throughput timer */
317 if (pRxData->hThroughputTimer)
318 {
319 tmr_DestroyTimer (pRxData->hThroughputTimer);
320 }
321 #endif
322
323 if (pRxData->reAuthActiveTimer)
324 {
325 tmr_DestroyTimer (pRxData->reAuthActiveTimer);
326 }
327
328 /* free Rx Data controll block */
329 os_memoryFree(pRxData->hOs, pRxData, sizeof(rxData_t));
330
331 return TI_OK;
332 }
333
334
335 /***************************************************************************
336 * rxData_stop *
337 ****************************************************************************
338 * DESCRIPTION: this function stop the rx data.
339 *
340 * INPUTS: hRxData - the object
341 *
342 * OUTPUT:
343 *
344 * RETURNS: TI_OK - stop succesfull
345 * TI_NOK - stop unsuccesfull
346 ***************************************************************************/
rxData_stop(TI_HANDLE hRxData)347 TI_STATUS rxData_stop (TI_HANDLE hRxData)
348 {
349 rxData_t *pRxData = (rxData_t *)hRxData;
350
351 /* check parameters validity */
352 if (pRxData == NULL)
353 {
354 return TI_NOK;
355 }
356
357 pRxData->rxDataExcludeUnencrypted = DEF_EXCLUDE_UNENCYPTED;
358 pRxData->rxDataExludeBroadcastUnencrypted = DEF_EXCLUDE_UNENCYPTED;
359 pRxData->rxDataEapolDestination = DEF_EAPOL_DESTINATION;
360 pRxData->rxDataPortStatus = DEF_RX_PORT_STATUS;
361
362 #ifdef TI_DBG
363 /* reset counters */
364 /*rxData_resetCounters(pRxData);*/
365 /*rxData_resetDbgCounters(pRxData);*/
366
367 /* stop throughput timer */
368 if (pRxData->rxThroughputTimerEnable)
369 {
370 tmr_StopTimer (pRxData->hThroughputTimer);
371 pRxData->rxThroughputTimerEnable = TI_FALSE;
372 }
373 #endif
374
375 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_stop() : Succeeded.\n");
376
377 return TI_OK;
378
379 }
380
381 /***************************************************************************
382 * rxData_getParam *
383 ****************************************************************************
384 * DESCRIPTION: get a specific parameter
385 *
386 * INPUTS: hRxData - the object
387 *
388 * OUTPUT: pParamInfo - structure which include the value of
389 * the requested parameter
390 *
391 * RETURNS: TI_OK
392 * TI_NOK
393 ***************************************************************************/
rxData_getParam(TI_HANDLE hRxData,paramInfo_t * pParamInfo)394 TI_STATUS rxData_getParam(TI_HANDLE hRxData, paramInfo_t *pParamInfo)
395 {
396 rxData_t *pRxData = (rxData_t *)hRxData;
397
398 /* check handle validity */
399 if (pRxData == NULL)
400 {
401 return TI_NOK;
402 }
403
404 switch (pParamInfo->paramType)
405 {
406 case RX_DATA_EXCLUDE_UNENCRYPTED_PARAM:
407 pParamInfo->content.rxDataExcludeUnencrypted = pRxData->rxDataExcludeUnencrypted;
408 break;
409
410 case RX_DATA_EAPOL_DESTINATION_PARAM:
411 pParamInfo->content.rxDataEapolDestination = pRxData->rxDataEapolDestination;
412 break;
413
414 case RX_DATA_PORT_STATUS_PARAM:
415 pParamInfo->content.rxDataPortStatus = pRxData->rxDataPortStatus;
416 break;
417
418 case RX_DATA_COUNTERS_PARAM:
419 pParamInfo->content.siteMgrTiWlanCounters.RecvOk = pRxData->rxDataCounters.RecvOk;
420 pParamInfo->content.siteMgrTiWlanCounters.DirectedBytesRecv = pRxData->rxDataCounters.DirectedBytesRecv;
421 pParamInfo->content.siteMgrTiWlanCounters.DirectedFramesRecv = pRxData->rxDataCounters.DirectedFramesRecv;
422 pParamInfo->content.siteMgrTiWlanCounters.MulticastBytesRecv = pRxData->rxDataCounters.MulticastBytesRecv;
423 pParamInfo->content.siteMgrTiWlanCounters.MulticastFramesRecv = pRxData->rxDataCounters.MulticastFramesRecv;
424 pParamInfo->content.siteMgrTiWlanCounters.BroadcastBytesRecv = pRxData->rxDataCounters.BroadcastBytesRecv;
425 pParamInfo->content.siteMgrTiWlanCounters.BroadcastFramesRecv = pRxData->rxDataCounters.BroadcastFramesRecv;
426 break;
427
428 case RX_DATA_GET_RX_DATA_FILTERS_STATISTICS:
429 TWD_ItrDataFilterStatistics (pRxData->hTWD,
430 pParamInfo->content.interogateCmdCBParams.fCb,
431 pParamInfo->content.interogateCmdCBParams.hCb,
432 pParamInfo->content.interogateCmdCBParams.pCb);
433 break;
434
435 case RX_DATA_RATE_PARAM:
436 pParamInfo->content.siteMgrCurrentRxRate = pRxData->uLastDataPktRate;
437 break;
438
439 default:
440 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, " rxData_getParam() : PARAMETER NOT SUPPORTED \n");
441 return (PARAM_NOT_SUPPORTED);
442 }
443
444 return TI_OK;
445 }
446
447 /***************************************************************************
448 * rxData_setParam *
449 ****************************************************************************
450 * DESCRIPTION: set a specific parameter
451 *
452 * INPUTS: hRxData - the object
453 * pParamInfo - structure which include the value to set for
454 * the requested parameter
455 *
456 * OUTPUT:
457 *
458 * RETURNS: TI_OK
459 * TI_NOK
460 ***************************************************************************/
rxData_setParam(TI_HANDLE hRxData,paramInfo_t * pParamInfo)461 TI_STATUS rxData_setParam(TI_HANDLE hRxData, paramInfo_t *pParamInfo)
462 {
463 rxData_t *pRxData = (rxData_t *)hRxData;
464
465 /* check handle validity */
466 if( pRxData == NULL )
467 {
468 return TI_NOK;
469 }
470
471 switch (pParamInfo->paramType)
472 {
473 case RX_DATA_EXCLUDE_UNENCRYPTED_PARAM:
474 pRxData->rxDataExcludeUnencrypted = pParamInfo->content.rxDataExcludeUnencrypted;
475 break;
476 case RX_DATA_EXCLUDE_BROADCAST_UNENCRYPTED_PARAM:
477 pRxData->rxDataExludeBroadcastUnencrypted = pParamInfo->content.rxDataExcludeUnencrypted;
478 break;
479 case RX_DATA_EAPOL_DESTINATION_PARAM:
480 pRxData->rxDataEapolDestination = pParamInfo->content.rxDataEapolDestination;
481 break;
482
483 case RX_DATA_PORT_STATUS_PARAM:
484 pRxData->rxDataPortStatus = pParamInfo->content.rxDataPortStatus;
485 break;
486
487 case RX_DATA_ENABLE_DISABLE_RX_DATA_FILTERS:
488 return rxData_enableDisableRxDataFilters(hRxData, pParamInfo->content.rxDataFilterEnableDisable);
489
490 case RX_DATA_ADD_RX_DATA_FILTER:
491 {
492 TRxDataFilterRequest* pRequest = &pParamInfo->content.rxDataFilterRequest;
493
494 return rxData_addRxDataFilter(hRxData, pRequest);
495 }
496
497 case RX_DATA_REMOVE_RX_DATA_FILTER:
498 {
499 TRxDataFilterRequest* pRequest = &pParamInfo->content.rxDataFilterRequest;
500
501 return rxData_removeRxDataFilter(hRxData, pRequest);
502 }
503
504 default:
505 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, " rxData_setParam() : PARAMETER NOT SUPPORTED \n");
506 return (PARAM_NOT_SUPPORTED);
507 }
508
509 return TI_OK;
510 }
511
512 /***************************************************************************
513 * rxData_enableDisableRxDataFilters *
514 ****************************************************************************
515 * DESCRIPTION:
516 *
517 *
518 * INPUTS:
519 *
520 *
521 *
522 * OUTPUT:
523 *
524 * RETURNS:
525 *
526 ***************************************************************************/
rxData_enableDisableRxDataFilters(TI_HANDLE hRxData,TI_BOOL enabled)527 static TI_STATUS rxData_enableDisableRxDataFilters(TI_HANDLE hRxData, TI_BOOL enabled)
528 {
529 rxData_t * pRxData = (rxData_t *) hRxData;
530
531 /* assert 0 or 1 */
532 if (enabled != TI_FALSE)
533 enabled = 1;
534
535 if (enabled == pRxData->filteringEnabled)
536 return TI_OK;
537
538 pRxData->filteringEnabled = enabled;
539
540 return TWD_CfgEnableRxDataFilter (pRxData->hTWD, pRxData->filteringEnabled, pRxData->filteringDefaultAction);
541 }
542
543 /***************************************************************************
544 * findFilterRequest *
545 ****************************************************************************
546 * DESCRIPTION:
547 *
548 *
549 * INPUTS:
550 *
551 *
552 *
553 * OUTPUT:
554 *
555 * RETURNS:
556 *
557 ***************************************************************************/
findFilterRequest(TI_HANDLE hRxData,TRxDataFilterRequest * request)558 static int findFilterRequest(TI_HANDLE hRxData, TRxDataFilterRequest* request)
559 {
560 rxData_t * pRxData = (rxData_t *) hRxData;
561 int i;
562
563 for (i = 0; i < MAX_DATA_FILTERS; ++i)
564 {
565 if (pRxData->isFilterSet[i])
566 {
567 if ((pRxData->filterRequests[i].offset == request->offset) &&
568 (pRxData->filterRequests[i].maskLength == request->maskLength) &&
569 (pRxData->filterRequests[i].patternLength == request->patternLength))
570 {
571 if ((os_memoryCompare(pRxData->hOs, pRxData->filterRequests[i].mask, request->mask, request->maskLength) == 0) &&
572 (os_memoryCompare(pRxData->hOs, pRxData->filterRequests[i].pattern, request->pattern, request->patternLength) == 0))
573 return i;
574 }
575 }
576 }
577
578 return -1;
579 }
580
581 /***************************************************************************
582 * closeFieldPattern *
583 ****************************************************************************
584 * DESCRIPTION:
585 *
586 *
587 * INPUTS:
588 *
589 *
590 *
591 * OUTPUT:
592 *
593 * RETURNS:
594 *
595 ***************************************************************************/
closeFieldPattern(rxData_t * pRxData,rxDataFilterFieldPattern_t * fieldPattern,TI_UINT8 * fieldPatterns,TI_UINT8 * lenFieldPatterns)596 static void closeFieldPattern (rxData_t * pRxData, rxDataFilterFieldPattern_t * fieldPattern, TI_UINT8 * fieldPatterns, TI_UINT8 * lenFieldPatterns)
597 {
598 fieldPatterns[*lenFieldPatterns] = fieldPattern->offset;
599 *lenFieldPatterns += sizeof(fieldPattern->offset);
600
601 fieldPatterns[*lenFieldPatterns] = fieldPattern->length;
602 *lenFieldPatterns += sizeof(fieldPattern->length);
603
604 fieldPatterns[*lenFieldPatterns] = fieldPattern->flag;
605 *lenFieldPatterns += sizeof(fieldPattern->flag);
606
607 os_memoryCopy(pRxData->hOs, fieldPatterns + *lenFieldPatterns, fieldPattern->pattern, fieldPattern->length);
608 *lenFieldPatterns += fieldPattern->length;
609
610 /* if the pattern bit mask is enabled add it to the end of the request */
611 if ((fieldPattern->flag & RX_DATA_FILTER_FLAG_USE_BIT_MASK) == RX_DATA_FILTER_FLAG_USE_BIT_MASK)
612 {
613 os_memoryCopy(pRxData->hOs, fieldPatterns + *lenFieldPatterns, fieldPattern->mask, fieldPattern->length);
614 *lenFieldPatterns += fieldPattern->length;
615 }
616
617 TRACE3(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": Closed field pattern, length = %d, total length = %d, pattern bit mask = %d.\n", fieldPattern->length, *lenFieldPatterns, ((fieldPattern->flag & RX_DATA_FILTER_FLAG_USE_BIT_MASK) == RX_DATA_FILTER_FLAG_USE_BIT_MASK));
618 }
619
620
621 /***************************************************************************
622 * parseRxDataFilterRequest *
623 ****************************************************************************
624 * DESCRIPTION:
625 *
626 *
627 * INPUTS:
628 *
629 *
630 *
631 * OUTPUT:
632 *
633 * RETURNS:
634 *
635 ***************************************************************************/
parseRxDataFilterRequest(TI_HANDLE hRxData,TRxDataFilterRequest * request,TI_UINT8 * numFieldPatterns,TI_UINT8 * lenFieldPatterns,TI_UINT8 * fieldPatterns)636 static int parseRxDataFilterRequest(TI_HANDLE hRxData, TRxDataFilterRequest* request, TI_UINT8 * numFieldPatterns, TI_UINT8 * lenFieldPatterns, TI_UINT8 * fieldPatterns)
637 {
638 rxData_t * pRxData = (rxData_t *) hRxData;
639
640 int maskIter;
641 int patternIter = 0;
642
643 /* used to store field patterns while they are built */
644 TI_BOOL isBuildingFieldPattern = TI_FALSE;
645 rxDataFilterFieldPattern_t fieldPattern;
646
647 for (maskIter = 0; maskIter < request->maskLength * 8; ++maskIter)
648 {
649 /* which byte in the mask and which bit in the byte we're at */
650 int bit = maskIter % 8;
651 int byte = maskIter / 8;
652
653 /* is the bit in the mask set */
654 TI_BOOL isSet = ((request->mask[byte] & (1 << bit)) == (1 << bit));
655
656 TRACE4(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": MaskIter = %d, Byte = %d, Bit = %d, isSet = %d\n", maskIter, byte, bit, isSet);
657
658 /* if we're in the midst of building a field pattern, we need to close in case */
659 /* the current bit is not set or we've reached the ethernet header boundary */
660 if (isBuildingFieldPattern)
661 {
662 if ((isSet == TI_FALSE) || (request->offset + maskIter == RX_DATA_FILTER_ETHERNET_HEADER_BOUNDARY))
663 {
664 closeFieldPattern(hRxData, &fieldPattern, fieldPatterns, lenFieldPatterns);
665
666 isBuildingFieldPattern = TI_FALSE;
667 }
668 }
669
670 /* nothing to do in case the bit is not set */
671 if (isSet)
672 {
673 /* if not already building a field pattern, create a new one */
674 if (isBuildingFieldPattern == TI_FALSE)
675 {
676 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": Creating a new field pattern.\n");
677
678 isBuildingFieldPattern = TI_TRUE;
679 ++(*numFieldPatterns);
680
681 if (*numFieldPatterns > RX_DATA_FILTER_MAX_FIELD_PATTERNS)
682 {
683 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR, ": Invalid filter, too many field patterns, maximum of %u is allowed!\n", RX_DATA_FILTER_MAX_FIELD_PATTERNS);
684
685 return TI_NOK;
686 }
687
688 fieldPattern.offset = request->offset + maskIter;
689 fieldPattern.length = 0;
690
691 /* we don't support the mask per bit feature yet. */
692 fieldPattern.flag = RX_DATA_FILTER_FLAG_NO_BIT_MASK;
693
694 /* first 14 bits are used for the Ethernet header, rest for the IP header */
695 if (fieldPattern.offset < RX_DATA_FILTER_ETHERNET_HEADER_BOUNDARY)
696 {
697 fieldPattern.flag |= RX_DATA_FILTER_FLAG_ETHERNET_HEADER;
698 }
699 else
700 {
701 fieldPattern.flag |= RX_DATA_FILTER_FLAG_IP_HEADER;
702 fieldPattern.offset -= RX_DATA_FILTER_ETHERNET_HEADER_BOUNDARY;
703 }
704
705 TRACE2(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": offset = %d, flag = %d.\n", fieldPattern.offset, fieldPattern.flag);
706 }
707
708 /* check that the pattern is long enough */
709 if (patternIter > request->patternLength)
710 {
711 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, ": Invalid filter, mask and pattern length are not consistent!\n");
712
713 return TI_NOK;
714 }
715
716 /* add the current pattern byte to the field pattern */
717 fieldPattern.pattern[fieldPattern.length++] = request->pattern[patternIter++];
718
719 /* check pattern matching boundary */
720 if (fieldPattern.offset + fieldPattern.length >= RX_DATA_FILTER_FILTER_BOUNDARY)
721 {
722 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR, ": Invalid filter, pattern matching cannot exceed first %u characters.\n", RX_DATA_FILTER_FILTER_BOUNDARY);
723
724 return TI_NOK;
725 }
726 }
727 }
728
729 /* check that the pattern is long enough */
730 if (patternIter != request->patternLength)
731 {
732 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, ": Invalid filter, mask and pattern lengths are not consistent!\n");
733
734 return TI_NOK;
735 }
736
737 /* close the last field pattern if needed */
738 if (isBuildingFieldPattern)
739 {
740 closeFieldPattern (hRxData, &fieldPattern, fieldPatterns, lenFieldPatterns);
741 }
742
743 return TI_OK;
744 }
745
746
747 /***************************************************************************
748 * rxData_setRxDataFilter *
749 ****************************************************************************
750 * DESCRIPTION:
751 *
752 *
753 * INPUTS:
754 *
755 *
756 *
757 * OUTPUT:
758 *
759 * RETURNS:
760 *
761 ***************************************************************************/
rxData_addRxDataFilter(TI_HANDLE hRxData,TRxDataFilterRequest * request)762 static TI_STATUS rxData_addRxDataFilter (TI_HANDLE hRxData, TRxDataFilterRequest* request)
763 {
764 rxData_t * pRxData = (rxData_t *) hRxData;
765
766 /* firmware request fields */
767 TI_UINT8 index = 0;
768 TI_UINT8 numFieldPatterns = 0;
769 TI_UINT8 lenFieldPatterns = 0;
770 TI_UINT8 fieldPatterns[MAX_DATA_FILTER_SIZE];
771
772 /* does the filter already exist? */
773 if (findFilterRequest(hRxData, request) >= 0)
774 {
775 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": Filter already exists.\n");
776
777 return RX_FILTER_ALREADY_EXISTS;
778 }
779
780 /* find place for insertion */
781 for (index = 0; index < MAX_DATA_FILTERS; ++index)
782 {
783 if (pRxData->isFilterSet[index] == TI_FALSE)
784 break;
785 }
786
787 /* are all filter slots taken? */
788 if (index == MAX_DATA_FILTERS)
789 {
790 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, ": No place to insert filter!\n");
791
792 return RX_NO_AVAILABLE_FILTERS;
793 }
794
795 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": Inserting filter at index %d.\n", index);
796
797 /* parse the filter request into discrete field patterns */
798 if (parseRxDataFilterRequest(hRxData, request, &numFieldPatterns, &lenFieldPatterns, fieldPatterns) != TI_OK)
799 return TI_NOK;
800
801 if (numFieldPatterns == 0)
802 return TI_NOK;
803
804 /* Store configuration for future manipulation */
805 pRxData->isFilterSet[index] = TI_TRUE;
806 os_memoryCopy(pRxData->hOs, &pRxData->filterRequests[index], request, sizeof(pRxData->filterRequests[index]));
807
808 /* Send configuration to firmware */
809 return TWD_CfgRxDataFilter (pRxData->hTWD,
810 index,
811 ADD_FILTER,
812 FILTER_SIGNAL,
813 numFieldPatterns,
814 lenFieldPatterns,
815 fieldPatterns);
816
817 }
818
819 /***************************************************************************
820 * rxData_removeRxDataFilter *
821 ****************************************************************************
822 * DESCRIPTION:
823 *
824 *
825 * INPUTS:
826 *
827 *
828 *
829 * OUTPUT:
830 *
831 * RETURNS:
832 *
833 ***************************************************************************/
rxData_removeRxDataFilter(TI_HANDLE hRxData,TRxDataFilterRequest * request)834 static TI_STATUS rxData_removeRxDataFilter (TI_HANDLE hRxData, TRxDataFilterRequest* request)
835 {
836 rxData_t * pRxData = (rxData_t *) hRxData;
837
838 int index = findFilterRequest(hRxData, request);
839
840 /* does the filter exist? */
841 if (index < 0)
842 {
843 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, ": Remove data filter request received but the specified filter was not found!");
844
845 return RX_FILTER_DOES_NOT_EXIST;
846 }
847
848 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, ": Removing filter at index %d.", index);
849
850 pRxData->isFilterSet[index] = TI_FALSE;
851
852 return TWD_CfgRxDataFilter (pRxData->hTWD,
853 index,
854 REMOVE_FILTER,
855 FILTER_SIGNAL,
856 0,
857 0,
858 NULL);
859 }
860
861 /***************************************************************************
862 * rxData_DistributorRxEvent *
863 ****************************************************************************
864 * DESCRIPTION:
865 *
866 *
867 * INPUTS:
868 *
869 *
870 *
871 * OUTPUT:
872 *
873 * RETURNS:
874 *
875 ***************************************************************************/
rxData_DistributorRxEvent(rxData_t * pRxData,TI_UINT16 Mask,int DataLen)876 static void rxData_DistributorRxEvent (rxData_t *pRxData, TI_UINT16 Mask, int DataLen)
877 {
878 DistributorMgr_EventCall (pRxData->RxEventDistributor, Mask, DataLen);
879 }
880
881 /***************************************************************************
882 * rxData_RegNotif *
883 ****************************************************************************/
rxData_RegNotif(TI_HANDLE hRxData,TI_UINT16 EventMask,GeneralEventCall_t CallBack,TI_HANDLE context,TI_UINT32 Cookie)884 TI_HANDLE rxData_RegNotif (TI_HANDLE hRxData, TI_UINT16 EventMask, GeneralEventCall_t CallBack, TI_HANDLE context, TI_UINT32 Cookie)
885 {
886 rxData_t *pRxData = (rxData_t *)hRxData;
887
888 if (!hRxData)
889 return NULL;
890
891 return DistributorMgr_Reg (pRxData->RxEventDistributor, EventMask, (TI_HANDLE)CallBack, context, Cookie);
892 }
893
894 /***************************************************************************
895 * rxData_AddToNotifMask *
896 ****************************************************************************/
rxData_AddToNotifMask(TI_HANDLE hRxData,TI_HANDLE Notifh,TI_UINT16 EventMask)897 TI_STATUS rxData_AddToNotifMask (TI_HANDLE hRxData, TI_HANDLE Notifh, TI_UINT16 EventMask)
898 {
899 rxData_t *pRxData = (rxData_t *)hRxData;
900
901 if (!hRxData)
902 return TI_NOK;
903
904 return DistributorMgr_AddToMask (pRxData->RxEventDistributor, Notifh, EventMask);
905 }
906
907
908 /***************************************************************************
909 * rxData_UnRegNotif *
910 ****************************************************************************/
rxData_UnRegNotif(TI_HANDLE hRxData,TI_HANDLE RegEventHandle)911 TI_STATUS rxData_UnRegNotif(TI_HANDLE hRxData,TI_HANDLE RegEventHandle)
912 {
913 rxData_t *pRxData = (rxData_t *)hRxData;
914
915 if (!hRxData)
916 return TI_NOK;
917
918 return DistributorMgr_UnReg (pRxData->RxEventDistributor, RegEventHandle);
919 }
920
921
922 /***************************************************************************
923 * rxData_receivePacketFromWlan *
924 ****************************************************************************
925 * DESCRIPTION: this function is called by the GWSI for each received Buffer.
926 * It filter and distribute the received Buffer.
927 *
928 * INPUTS: hRxData - the object
929 * pBuffer - the received Buffer.
930 * pRxAttr - Rx attributes
931 *
932 * OUTPUT:
933 *
934 * RETURNS:
935 ***************************************************************************/
rxData_receivePacketFromWlan(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)936 void rxData_receivePacketFromWlan (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
937 {
938 rxData_t *pRxData = (rxData_t *)hRxData;
939 TMacAddr address3;
940 dot11_header_t *pDot11Hdr;
941
942 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_receivePacketFromWlan() : pRxAttr->packetType = %d\n", pRxAttr->ePacketType);
943
944 switch (pRxAttr->ePacketType)
945 {
946 case TAG_CLASS_MANAGEMENT:
947 case TAG_CLASS_BCN_PRBRSP:
948
949 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, "rxData_receivePacketFromWlan(): Received management Buffer len = %d\n", RX_BUF_LEN(pBuffer));
950
951 /* update siteMngr
952 *
953 * the BSSID in mgmt frames is always addr3 in the header
954 * must copy address3 since Buffer is freed in mlmeParser_recv
955 */
956 pDot11Hdr = (dot11_header_t*)RX_BUF_DATA(pBuffer);
957
958 os_memoryCopy(pRxData->hOs, &address3, &pDot11Hdr->address3, sizeof(address3));
959
960 /* distribute mgmt pBuffer to mlme */
961 if( mlmeParser_recv(pRxData->hMlme, pBuffer, pRxAttr) != TI_OK )
962 {
963 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_receivePacketFromWlan() : MLME returned error \n");
964 }
965 break;
966
967 case TAG_CLASS_DATA:
968 case TAG_CLASS_QOS_DATA:
969 case TAG_CLASS_AMSDU:
970 case TAG_CLASS_EAPOL:
971 {
972 CL_TRACE_START_L3();
973 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_receivePacketFromWlan() : Received Data Buffer len = %d\n", RX_BUF_LEN(pBuffer));
974
975 /* send pBuffer to data dispatcher */
976 rxData_dataPacketDisptcher(hRxData, pBuffer, pRxAttr);
977 CL_TRACE_END_L3("tiwlan_drv.ko", "INHERIT", "RX", ".DataPacket");
978 break;
979 }
980
981 default:
982 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_receivePacketFromWlan(): Received unspecified packet type !!! \n");
983 RxBufFree(pRxData->hOs, pBuffer);
984 break;
985 }
986 }
987
988 /***************************************************************************
989 * rxData_dataPacketDisptcher *
990 ****************************************************************************
991 * DESCRIPTION: this function is called upon receving data Buffer,
992 * it dispatches the packet to the approciate function according to
993 * data packet type and rx port status.
994 *
995 * INPUTS: hRxData - the object
996 * pBuffer - the received Buffer.
997 * pRxAttr - Rx attributes
998 *
999 * OUTPUT:
1000 *
1001 * RETURNS:
1002 ***************************************************************************/
1003
rxData_dataPacketDisptcher(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1004 static void rxData_dataPacketDisptcher (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1005 {
1006 rxData_t *pRxData = (rxData_t *)hRxData;
1007 portStatus_e DataPortStatus;
1008 rxDataPacketType_e DataPacketType;
1009
1010 /* get rx port status */
1011 DataPortStatus = pRxData->rxDataPortStatus;
1012
1013 /* discard data packets received while rx data port is closed */
1014 if (DataPortStatus == CLOSE)
1015 {
1016 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_dataPacketDisptcher() : Received Data Buffer while Rx data port is closed \n");
1017
1018 rxData_discardPacket (hRxData, pBuffer, pRxAttr);
1019 return;
1020 }
1021
1022 /* get data packet type */
1023
1024 pRxData->uLastDataPktRate = pRxAttr->Rate; /* save Rx packet rate for statistics */
1025
1026 #ifdef XCC_MODULE_INCLUDED
1027 if (XCCMngr_isIappPacket (pRxData->hXCCMgr, pBuffer) == TI_TRUE)
1028 {
1029 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_dataPacketDisptcher() : Received Iapp Buffer \n");
1030
1031 DataPacketType = DATA_IAPP_PACKET;
1032
1033 /* dispatch Buffer according to packet type and current rx data port status */
1034 pRxData->rxData_dispatchBuffer[DataPortStatus][DataPacketType] (hRxData, pBuffer, pRxAttr);
1035 }
1036 else
1037 #endif
1038 {
1039 /* A-MSDU ? */
1040 if (TAG_CLASS_AMSDU == pRxAttr->ePacketType)
1041 {
1042 rxData_ConvertAmsduToEthPackets (hRxData, pBuffer, pRxAttr);
1043 }
1044 else
1045 {
1046 TI_UINT16 etherType = 0;
1047 TEthernetHeader * pEthernetHeader;
1048
1049 /*
1050 * if Host processes received packets, the header translation
1051 * from WLAN to ETH is done here. The conversion has been moved
1052 * here so that IAPP packets aren't converted.
1053 */
1054 rxData_convertWlanToEthHeader (hRxData, pBuffer, ðerType);
1055
1056 pEthernetHeader = (TEthernetHeader *)RX_ETH_PKT_DATA(pBuffer);
1057
1058 if (etherType == ETHERTYPE_802_1D)
1059 {
1060 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_dataPacketDisptcher() : Received VLAN packet \n");
1061
1062 DataPacketType = DATA_VLAN_PACKET;
1063 }
1064 else if (HTOWLANS(pEthernetHeader->type) == EAPOL_PACKET)
1065 {
1066 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_dataPacketDisptcher() : Received Eapol packet \n");
1067
1068 if (rxData_IsReAuthInProgress(pRxData))
1069 {
1070 /* ReAuth already in progress, restart timer */
1071 rxData_StopReAuthActiveTimer(pRxData);
1072 rxData_StartReAuthActiveTimer(pRxData);
1073 }
1074 else
1075 {
1076 if (PowerMgr_getReAuthActivePriority(pRxData->hPowerMgr))
1077 {
1078 /* ReAuth not in progress yet, force active, set flag, restart timer, send event */
1079 rxData_SetReAuthInProgress(pRxData, TI_TRUE);
1080 rxData_StartReAuthActiveTimer(pRxData);
1081 rxData_ReauthEnablePriority(pRxData);
1082 EvHandlerSendEvent(pRxData->hEvHandler, IPC_EVENT_RE_AUTH_STARTED, NULL, 0);
1083 }
1084 }
1085
1086 DataPacketType = DATA_EAPOL_PACKET;
1087 }
1088 else
1089 {
1090 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_dataPacketDisptcher() : Received Data packet \n");
1091
1092 DataPacketType = DATA_DATA_PACKET;
1093 }
1094
1095 /* dispatch Buffer according to packet type and current rx data port status */
1096 pRxData->rxData_dispatchBuffer[DataPortStatus][DataPacketType] (hRxData, pBuffer, pRxAttr);
1097 }
1098 }
1099
1100 }
1101
1102 /***************************************************************************
1103 * rxData_discardPacket *
1104 ****************************************************************************
1105 * DESCRIPTION: this function is called to discard Buffer
1106 *
1107 * INPUTS: hRxData - the object
1108 * pBuffer - the received Buffer.
1109 * pRxAttr - Rx attributes
1110 *
1111 * OUTPUT:
1112 *
1113 * RETURNS:
1114 ***************************************************************************/
rxData_discardPacket(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1115 static void rxData_discardPacket (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1116 {
1117 rxData_t *pRxData = (rxData_t *)hRxData;
1118
1119 TRACE2(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_discardPacket: rx port status = %d , Buffer status = %d \n", pRxData->rxDataPortStatus, pRxAttr->status);
1120
1121 pRxData->rxDataDbgCounters.excludedFrameCounter++;
1122
1123 /* free Buffer */
1124 RxBufFree(pRxData->hOs, pBuffer);
1125
1126 }
1127
1128 /***************************************************************************
1129 * rxData_discardPacketVlan *
1130 ****************************************************************************
1131 * DESCRIPTION: this function is called to discard Buffer
1132 *
1133 * INPUTS: hRxData - the object
1134 * pBuffer - the received Buffer.
1135 * pRxAttr - Rx attributes
1136 *
1137 * OUTPUT:
1138 *
1139 * RETURNS:
1140 ***************************************************************************/
rxData_discardPacketVlan(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1141 static void rxData_discardPacketVlan (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1142 {
1143 rxData_t *pRxData = (rxData_t *)hRxData;
1144
1145 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_discardPacketVlan : drop packet that contains VLAN tag\n");
1146
1147 pRxData->rxDataDbgCounters.rxDroppedDueToVLANIncludedCnt++;
1148
1149 /* free Buffer */
1150 RxBufFree(pRxData->hOs, pBuffer);
1151 }
1152
1153
1154 /***************************************************************************
1155 * rxData_rcvPacketInOpenNotify *
1156 ****************************************************************************
1157 * DESCRIPTION: this function is called upon receving data Eapol packet type
1158 * while rx port status is "open notify"
1159 *
1160 * INPUTS: hRxData - the object
1161 * pBuffer - the received Buffer.
1162 * pRxAttr - Rx attributes
1163 *
1164 * OUTPUT:
1165 *
1166 * RETURNS:
1167 ***************************************************************************/
rxData_rcvPacketInOpenNotify(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1168 static void rxData_rcvPacketInOpenNotify (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1169 {
1170 rxData_t *pRxData = (rxData_t *)hRxData;
1171
1172 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_rcvPacketInOpenNotify: receiving data packet while in rx port status is open notify\n");
1173
1174 pRxData->rxDataDbgCounters.rcvUnicastFrameInOpenNotify++;
1175
1176 /* free Buffer */
1177 RxBufFree(pRxData->hOs, pBuffer);
1178 }
1179
1180
1181 /***************************************************************************
1182 * rxData_rcvPacketEapol *
1183 ****************************************************************************
1184 * DESCRIPTION: this function is called upon receving data Eapol packet type
1185 * while rx port status is "open eapol"
1186 *
1187 * INPUTS: hRxData - the object
1188 * pBuffer - the received Buffer.
1189 * pRxAttr - Rx attributes
1190 *
1191 * OUTPUT:
1192 *
1193 * RETURNS:
1194 ***************************************************************************/
rxData_rcvPacketEapol(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1195 static void rxData_rcvPacketEapol(TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1196 {
1197 rxData_t *pRxData = (rxData_t *)hRxData;
1198
1199 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketEapol() : Received an EAPOL frame tranferred to OS\n");
1200
1201 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketEapol() : Received an EAPOL frame tranferred to OS\n");
1202
1203 EvHandlerSendEvent (pRxData->hEvHandler, IPC_EVENT_EAPOL, NULL, 0);
1204 os_receivePacket (pRxData->hOs, pBuffer, (TI_UINT16)RX_ETH_PKT_LEN(pBuffer));
1205
1206 }
1207
1208 /***************************************************************************
1209 * rxData_rcvPacketData *
1210 ****************************************************************************
1211 * DESCRIPTION: this function is called upon receving data "data" packet type
1212 * while rx port status is "open"
1213 *
1214 * INPUTS: hRxData - the object
1215 * pBuffer - the received Buffer.
1216 * pRxAttr - Rx attributes
1217 *
1218 * OUTPUT:
1219 *
1220 * RETURNS:
1221 ***************************************************************************/
rxData_rcvPacketData(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1222 static void rxData_rcvPacketData(TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1223 {
1224 rxData_t *pRxData = (rxData_t *)hRxData;
1225 TEthernetHeader *pEthernetHeader;
1226 TI_UINT16 EventMask = 0;
1227 TFwInfo *pFwInfo;
1228
1229 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketData() : Received DATA frame tranferred to OS\n");
1230
1231 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketData() : Received DATA frame tranferred to OS\n");
1232
1233 /* check encryption status */
1234 pEthernetHeader = (TEthernetHeader *)RX_ETH_PKT_DATA(pBuffer);
1235 if (!MAC_MULTICAST (pEthernetHeader->dst))
1236 { /* unicast frame */
1237 if((pRxData->rxDataExcludeUnencrypted) && (!(pRxAttr->packetInfo & RX_DESC_ENCRYPT_MASK)))
1238 {
1239 pRxData->rxDataDbgCounters.excludedFrameCounter++;
1240 /* free Buffer */
1241 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_rcvPacketData() : exclude unicast unencrypted is TI_TRUE & packet encryption is OFF\n");
1242
1243 RxBufFree(pRxData->hOs, pBuffer);
1244 return;
1245 }
1246 }
1247 else
1248 { /* broadcast frame */
1249 if ((pRxData->rxDataExludeBroadcastUnencrypted) && (!(pRxAttr->packetInfo & RX_DESC_ENCRYPT_MASK)))
1250 {
1251 pRxData->rxDataDbgCounters.excludedFrameCounter++;
1252 /* free Buffer */
1253 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, " rxData_rcvPacketData() : exclude broadcast unencrypted is TI_TRUE & packet encryption is OFF\n");
1254
1255 RxBufFree(pRxData->hOs, pBuffer);
1256 return;
1257 }
1258
1259 /*
1260 * Discard multicast/broadcast frames that we sent ourselves.
1261 * Per IEEE 802.11-2007 section 9.2.7: "STAs shall filter out
1262 * broadcast/multicast messages that contain their address as
1263 * the source address."
1264 */
1265 pFwInfo = TWD_GetFWInfo (pRxData->hTWD);
1266 if (MAC_EQUAL(pFwInfo->macAddress, pEthernetHeader->src))
1267 {
1268 pRxData->rxDataDbgCounters.excludedFrameCounter++;
1269 /* free Buffer */
1270 RxBufFree(pRxData->hOs, pBuffer);
1271 return;
1272 }
1273 }
1274
1275 /* update traffic monitor parameters */
1276 pRxData->rxDataCounters.RecvOk++;
1277 EventMask |= RECV_OK;
1278
1279 if (!MAC_MULTICAST (pEthernetHeader->dst))
1280 {
1281 /* Directed frame */
1282 pRxData->rxDataCounters.DirectedFramesRecv++;
1283 pRxData->rxDataCounters.DirectedBytesRecv += RX_ETH_PKT_LEN(pBuffer);
1284 EventMask |= DIRECTED_BYTES_RECV;
1285 EventMask |= DIRECTED_FRAMES_RECV;
1286 }
1287 else if (MAC_BROADCAST (pEthernetHeader->dst))
1288 {
1289 /* Broadcast frame */
1290 pRxData->rxDataCounters.BroadcastFramesRecv++;
1291 pRxData->rxDataCounters.BroadcastBytesRecv += RX_ETH_PKT_LEN(pBuffer);
1292 EventMask |= BROADCAST_BYTES_RECV;
1293 EventMask |= BROADCAST_FRAMES_RECV;
1294 }
1295 else
1296 {
1297 /* Multicast Address */
1298 pRxData->rxDataCounters.MulticastFramesRecv++;
1299 pRxData->rxDataCounters.MulticastBytesRecv += RX_ETH_PKT_LEN(pBuffer);
1300 EventMask |= MULTICAST_BYTES_RECV;
1301 EventMask |= MULTICAST_FRAMES_RECV;
1302 }
1303 pRxData->rxDataCounters.LastSecBytesRecv += RX_ETH_PKT_LEN(pBuffer);
1304
1305 /*Handle PREAUTH_EAPOL_PACKET*/
1306 if (HTOWLANS(pEthernetHeader->type) == PREAUTH_EAPOL_PACKET)
1307 {
1308 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketData() : Received an Pre-Auth EAPOL frame tranferred to OS\n");
1309 }
1310
1311 rxData_DistributorRxEvent (pRxData, EventMask, RX_ETH_PKT_LEN(pBuffer));
1312
1313 /* deliver packet to os */
1314 os_receivePacket (pRxData->hOs, pBuffer, (TI_UINT16)RX_ETH_PKT_LEN(pBuffer));
1315 }
1316
1317
1318 /***************************************************************************
1319 * rxData_rcvPacketIapp *
1320 ****************************************************************************
1321 * DESCRIPTION: this function is called upon receving data IAPP packet type
1322 * while rx port status is "open"
1323 *
1324 * INPUTS: hRxData - the object
1325 * pBuffer - the received Buffer.
1326 * pRxAttr - Rx attributes
1327 *
1328 * OUTPUT:
1329 *
1330 * RETURNS:
1331 ***************************************************************************/
1332 #ifdef XCC_MODULE_INCLUDED
1333
rxData_rcvPacketIapp(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1334 static void rxData_rcvPacketIapp(TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1335 {
1336 rxData_t *pRxData = (rxData_t *)hRxData;
1337
1338 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketIapp() : Received IAPP frame tranferred to XCCMgr\n");
1339
1340 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_rcvPacketIapp() : Received IAPP frame tranferred to XCCMgr\n");
1341
1342 /* tranfer packet to XCCMgr */
1343 XCCMngr_recvIAPPPacket (pRxData->hXCCMgr, pBuffer, pRxAttr);
1344
1345 /* free Buffer */
1346 RxBufFree(pRxData->hOs, pBuffer);
1347 }
1348
1349 #endif
1350
1351
1352 /****************************************************************************
1353 * rxData_convertWlanToEthHeader *
1354 *****************************************************************************
1355 * DESCRIPTION: this function convert the Packet header from 802.11 header
1356 * format to ethernet format
1357 *
1358 * INPUTS: hRxData - the object
1359 * pBuffer - the received pBuffer in 802.11 format
1360 *
1361 * OUTPUT: pEthPacket - pointer to the received pBuffer in ethernet format
1362 * uEthLength - ethernet packet length
1363 *
1364 * RETURNS: TI_OK/TI_NOK
1365 ***************************************************************************/
rxData_convertWlanToEthHeader(TI_HANDLE hRxData,void * pBuffer,TI_UINT16 * etherType)1366 static TI_STATUS rxData_convertWlanToEthHeader (TI_HANDLE hRxData, void *pBuffer, TI_UINT16 * etherType)
1367 {
1368 TEthernetHeader EthHeader;
1369 Wlan_LlcHeader_T *pWlanSnapHeader;
1370 TI_UINT8 *dataBuf;
1371 dot11_header_t *pDot11Header;
1372 TI_UINT32 lengthDelta;
1373 TI_UINT16 swapedTypeLength;
1374 TI_UINT32 headerLength;
1375 TI_UINT8 createEtherIIHeader;
1376 rxData_t *pRxData = (rxData_t *)hRxData;
1377
1378 dataBuf = (TI_UINT8 *)RX_BUF_DATA(pBuffer);
1379
1380 /* Setting the mac header len according to the received FrameControl field in the Mac Header */
1381 GET_MAX_HEADER_SIZE (dataBuf, &headerLength);
1382 pDot11Header = (dot11_header_t*) dataBuf;
1383 pWlanSnapHeader = (Wlan_LlcHeader_T*)((TI_UINT32)dataBuf + (TI_UINT32)headerLength);
1384
1385 swapedTypeLength = WLANTOHS (pWlanSnapHeader->Type);
1386 *etherType = swapedTypeLength;
1387
1388 /* Prepare the Ethernet header. */
1389 if( ENDIAN_HANDLE_WORD(pDot11Header->fc) & DOT11_FC_FROM_DS)
1390 { /* Infrastructure bss */
1391 MAC_COPY (EthHeader.dst, pDot11Header->address1);
1392 MAC_COPY (EthHeader.src, pDot11Header->address3);
1393 }
1394 else
1395 { /* Independent bss */
1396 MAC_COPY (EthHeader.dst, pDot11Header->address1);
1397 MAC_COPY (EthHeader.src, pDot11Header->address2);
1398 }
1399
1400 createEtherIIHeader = TI_FALSE;
1401 /* See if the LLC header in the frame shows the SAP SNAP... */
1402 if((SNAP_CHANNEL_ID == pWlanSnapHeader->DSAP) &&
1403 (SNAP_CHANNEL_ID == pWlanSnapHeader->SSAP) &&
1404 (LLC_CONTROL_UNNUMBERED_INFORMATION == pWlanSnapHeader->Control))
1405 {
1406 /* Check for the Bridge Tunnel OUI in the SNAP Header... */
1407 if((SNAP_OUI_802_1H_BYTE0 == pWlanSnapHeader->OUI[ 0 ]) &&
1408 (SNAP_OUI_802_1H_BYTE1 == pWlanSnapHeader->OUI[ 1 ]) &&
1409 (SNAP_OUI_802_1H_BYTE2 == pWlanSnapHeader->OUI[ 2 ]))
1410 {
1411 /* Strip the SNAP header by skipping over it. */
1412 /* Start moving data from the Ethertype field in the SNAP */
1413 /* header. Move to the TypeLength field in the 802.3 header. */
1414 createEtherIIHeader = TI_TRUE;
1415 }
1416 /* Check for the RFC 1042 OUI in the SNAP Header */
1417 else
1418 {
1419 /* Check for the RFC 1042 OUI in the SNAP Header */
1420 if( (SNAP_OUI_RFC1042_BYTE0 == pWlanSnapHeader->OUI[ 0 ]) &&
1421 (SNAP_OUI_RFC1042_BYTE1 == pWlanSnapHeader->OUI[ 1 ]) &&
1422 (SNAP_OUI_RFC1042_BYTE2 == pWlanSnapHeader->OUI[ 2 ]))
1423 {
1424 /* See if the Ethertype is in our selective translation table */
1425 /* (Appletalk AARP and DIX II IPX are the two protocols in */
1426 /* our 'table') */
1427 if((ETHERTYPE_APPLE_AARP != swapedTypeLength) &&
1428 (ETHERTYPE_DIX_II_IPX != swapedTypeLength))
1429 {
1430 /* Strip the SNAP header by skipping over it. */
1431 createEtherIIHeader = TI_TRUE;
1432 }
1433 }
1434 }
1435 }
1436
1437 if( createEtherIIHeader == TI_TRUE )
1438 {
1439 /* The LEN/TYPE bytes are set to TYPE, the entire WLAN+SNAP is removed.*/
1440 lengthDelta = headerLength + WLAN_SNAP_HDR_LEN - ETHERNET_HDR_LEN;
1441 EthHeader.type = pWlanSnapHeader->Type;
1442 }
1443 else
1444 {
1445 /* The LEN/TYPE bytes are set to frame LEN, only the WLAN header is removed, */
1446 /* the entire 802.3 or 802.2 header is not removed.*/
1447 lengthDelta = headerLength - ETHERNET_HDR_LEN;
1448 EthHeader.type = WLANTOHS((TI_UINT16)(RX_BUF_LEN(pBuffer) - headerLength));
1449 }
1450
1451 /* Replace the 802.11 header and the LLC with Ethernet packet. */
1452 dataBuf += lengthDelta;
1453 os_memoryCopy (pRxData->hOs, dataBuf, (void*)&EthHeader, ETHERNET_HDR_LEN);
1454 RX_ETH_PKT_DATA(pBuffer) = dataBuf;
1455 RX_ETH_PKT_LEN(pBuffer) = RX_BUF_LEN(pBuffer) - lengthDelta;
1456
1457 return TI_OK;
1458 }
1459
1460
1461 /**
1462 * \brief convert A-MSDU to several ethernet packets
1463 *
1464 * \param hRxData - the object
1465 * \param pBuffer - the received Buffer in A-MSDU 802.11n format
1466 * \param pRxAttr - Rx attributes
1467 * \return TI_OK on success or TI_NOK on failure
1468 *
1469 * \par Description
1470 * Static function
1471 * This function convert the A-MSDU Packet from A-MSDU 802.11n packet
1472 * format to several ethernet packets format and pass them to the OS layer
1473 *
1474 * \sa
1475 */
rxData_ConvertAmsduToEthPackets(TI_HANDLE hRxData,void * pBuffer,TRxAttr * pRxAttr)1476 static TI_STATUS rxData_ConvertAmsduToEthPackets (TI_HANDLE hRxData, void *pBuffer, TRxAttr* pRxAttr)
1477 {
1478
1479 TEthernetHeader *pMsduEthHeader;
1480 TEthernetHeader *pEthHeader;
1481 Wlan_LlcHeader_T *pWlanSnapHeader;
1482 TI_UINT8 *pAmsduDataBuf;
1483 TI_UINT16 uAmsduDataLen;
1484 void *pDataBuf;
1485 TI_UINT16 uDataLen;
1486 TI_UINT32 lengthDelta;
1487 TI_UINT16 swapedTypeLength;
1488 TI_UINT32 headerLength;
1489 rxDataPacketType_e DataPacketType;
1490 rxData_t *pRxData = (rxData_t *)hRxData;
1491
1492 /* total AMPDU header */
1493 pAmsduDataBuf = (TI_UINT8 *)RX_BUF_DATA(pBuffer);
1494 /* Setting the mac header len according to the received FrameControl field in the Mac Header */
1495 GET_MAX_HEADER_SIZE (pAmsduDataBuf, &headerLength);
1496
1497 /*
1498 * init loop setting
1499 */
1500 /* total AMPDU size */
1501 uAmsduDataLen = (TI_UINT16)(RX_BUF_LEN(pBuffer) - headerLength);
1502 /* ETH header */
1503 pMsduEthHeader = (TEthernetHeader *)(pAmsduDataBuf + headerLength);
1504 /* ETH length, in A-MSDU the MSDU header type contain the MSDU length and not the type */
1505 uDataLen = WLANTOHS(pMsduEthHeader->type);
1506
1507 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, "rxData_ConvertAmsduToEthPackets(): A-MSDU received in length %d \n",uAmsduDataLen);
1508
1509 /* if we have another packet at the AMSDU */
1510 while((uDataLen < uAmsduDataLen) && (uAmsduDataLen > ETHERNET_HDR_LEN + FCS_SIZE))
1511 {
1512 /* allocate a new buffer */
1513 /* RxBufAlloc() add an extra word for alignment the MAC payload */
1514 rxData_RequestForBuffer (hRxData, &pDataBuf, sizeof(RxIfDescriptor_t) + WLAN_SNAP_HDR_LEN + ETHERNET_HDR_LEN + uDataLen, 0, TAG_CLASS_AMSDU);
1515 if (NULL == pDataBuf)
1516 {
1517 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_ConvertAmsduToEthPackets(): cannot alloc MSDU packet. length %d \n",uDataLen);
1518 rxData_discardPacket (hRxData, pBuffer, pRxAttr);
1519 return TI_NOK;
1520 }
1521
1522 /* read packet type from LLC */
1523 pWlanSnapHeader = (Wlan_LlcHeader_T*)((TI_UINT8*)pMsduEthHeader + ETHERNET_HDR_LEN);
1524 swapedTypeLength = WLANTOHS (pWlanSnapHeader->Type);
1525
1526 /* copy the RxIfDescriptor */
1527 os_memoryCopy (pRxData->hOs, pDataBuf, pBuffer, sizeof(RxIfDescriptor_t));
1528
1529 /* update length, in the RxIfDescriptor the Len in words (4B) */
1530 ((RxIfDescriptor_t *)pDataBuf)->length = (sizeof(RxIfDescriptor_t) + WLAN_SNAP_HDR_LEN + ETHERNET_HDR_LEN + uDataLen) >> 2;
1531 ((RxIfDescriptor_t *)pDataBuf)->extraBytes = 4 - ((sizeof(RxIfDescriptor_t) + WLAN_SNAP_HDR_LEN + ETHERNET_HDR_LEN + uDataLen) & 0x3);
1532
1533 /* Prepare the Ethernet header pointer. */
1534 /* add padding in the start of the buffer in order to align ETH payload */
1535 pEthHeader = (TEthernetHeader *)((TI_UINT8 *)(RX_BUF_DATA(pDataBuf)) +
1536 WLAN_SNAP_HDR_LEN +
1537 PADDING_ETH_PACKET_SIZE);
1538
1539 /* copy the Ethernet header */
1540 os_memoryCopy (pRxData->hOs, pEthHeader, pMsduEthHeader, ETHERNET_HDR_LEN);
1541
1542 /* The LEN/TYPE bytes are set to TYPE */
1543 pEthHeader->type = pWlanSnapHeader->Type;
1544
1545 /* Delta length for the next packet */
1546 lengthDelta = ETHERNET_HDR_LEN + uDataLen;
1547
1548 /* copy the packet payload */
1549 if (uDataLen > WLAN_SNAP_HDR_LEN)
1550 os_memoryCopy (pRxData->hOs,
1551 (((TI_UINT8*)pEthHeader) + ETHERNET_HDR_LEN),
1552 ((TI_UINT8*)pMsduEthHeader) + ETHERNET_HDR_LEN + WLAN_SNAP_HDR_LEN,
1553 uDataLen - WLAN_SNAP_HDR_LEN);
1554
1555 /* set the packet type */
1556 if (swapedTypeLength == ETHERTYPE_802_1D)
1557 {
1558 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_ConvertAmsduToEthPackets() : Received VLAN Buffer \n");
1559
1560 DataPacketType = DATA_VLAN_PACKET;
1561 }
1562 else if (HTOWLANS(pEthHeader->type) == EAPOL_PACKET)
1563 {
1564 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_ConvertAmsduToEthPackets() : Received Eapol pBuffer \n");
1565
1566 DataPacketType = DATA_EAPOL_PACKET;
1567 }
1568 else
1569 {
1570 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, " rxData_ConvertAmsduToEthPackets() : Received Data pBuffer \n");
1571
1572 DataPacketType = DATA_DATA_PACKET;
1573 }
1574
1575 /* update buffer setting */
1576 /* save the ETH packet address */
1577 RX_ETH_PKT_DATA(pDataBuf) = pEthHeader;
1578 /* save the ETH packet size */
1579 RX_ETH_PKT_LEN(pDataBuf) = uDataLen + ETHERNET_HDR_LEN - WLAN_SNAP_HDR_LEN;
1580
1581 /* dispatch Buffer according to packet type and current rx data port status */
1582 pRxData->rxData_dispatchBuffer[pRxData->rxDataPortStatus][DataPacketType] (hRxData, pDataBuf, pRxAttr);
1583
1584 /* star of MSDU packet always align acceding to 11n spec */
1585 lengthDelta = (lengthDelta + ALIGN_4BYTE_MASK) & ~ALIGN_4BYTE_MASK;
1586 pMsduEthHeader = (TEthernetHeader *)(((TI_UINT8*)pMsduEthHeader) + lengthDelta);
1587
1588 if(uAmsduDataLen > lengthDelta)
1589 {
1590 /* swich to the next MSDU */
1591 uAmsduDataLen = uAmsduDataLen - lengthDelta;
1592 }
1593 else
1594 {
1595 /* no more MSDU */
1596 uAmsduDataLen = 0;
1597 break;
1598 }
1599
1600
1601 /* in A-MSDU the MSDU header type contain the MSDU length and not the type */
1602 uDataLen = WLANTOHS(pMsduEthHeader->type);
1603
1604 } /* while end */
1605
1606
1607 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION, "rxData_ConvertAmsduToEthPackets(): A-MSDU Packe conversion done.\n");
1608
1609 /* free the A-MSDU packet */
1610 RxBufFree(pRxData->hOs, pBuffer);
1611
1612 return TI_OK;
1613 }
1614
1615 /****************************************************************************************
1616 * rxData_ReceivePacket *
1617 ****************************************************************************************
1618 DESCRIPTION: receive packet CB from RxXfer.
1619 parse the status and other parameters and forward the frame to
1620 rxData_receivePacketFromWlan()
1621
1622 INPUT: Rx frame with its parameters
1623
1624 OUTPUT:
1625
1626 RETURN:
1627
1628 ************************************************************************/
rxData_ReceivePacket(TI_HANDLE hRxData,void * pBuffer)1629 static void rxData_ReceivePacket (TI_HANDLE hRxData,
1630 void *pBuffer)
1631 {
1632 rxData_t *pRxData = (rxData_t *)hRxData;
1633
1634 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION, "rxData_ReceivePacket: Received BUF %x\n", pBuffer);
1635
1636 if (pBuffer)
1637 {
1638 RxIfDescriptor_t *pRxParams = (RxIfDescriptor_t*)pBuffer;
1639 TRxAttr RxAttr;
1640 ERate appRate;
1641 dot11_header_t *pHdr;
1642
1643 /*
1644 * First thing we do is getting the dot11_header, and than we check the status, since the header is
1645 * needed for RX_MIC_FAILURE_ERROR
1646 */
1647
1648 pHdr = (dot11_header_t *)RX_BUF_DATA(pBuffer);
1649
1650 /* Check status */
1651 switch (pRxParams->status & RX_DESC_STATUS_MASK)
1652 {
1653 case RX_DESC_STATUS_SUCCESS:
1654 break;
1655
1656 case RX_DESC_STATUS_DECRYPT_FAIL:
1657 {
1658 /* This error is not important before the Connection, so we ignore it when portStatus is not OPEN */
1659 if (pRxData->rxDataPortStatus == OPEN)
1660 {
1661 TRACE0(pRxData->hReport, REPORT_SEVERITY_WARNING, "rxData_ReceivePacket: Received Packet with RX_DESC_DECRYPT_FAIL\n");
1662 }
1663
1664 RxBufFree(pRxData->hOs, pBuffer);
1665 return;
1666 }
1667 case RX_DESC_STATUS_MIC_FAIL:
1668 {
1669 TI_UINT8 uKeyType;
1670 paramInfo_t Param;
1671 TMacAddr* pMac = &pHdr->address1; /* hold the first mac address */
1672 /* Get BSS type */
1673 Param.paramType = SITE_MGR_CURRENT_BSS_TYPE_PARAM;
1674 siteMgr_getParam (pRxData->hSiteMgr, &Param);
1675
1676 /* For multicast/broadcast frames or in IBSS the key used is GROUP, else - it's Pairwise */
1677 if (MAC_MULTICAST(*pMac) || Param.content.siteMgrCurrentBSSType == BSS_INDEPENDENT)
1678 {
1679 uKeyType = (TI_UINT8)KEY_TKIP_MIC_GROUP;
1680 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_ReceivePacket: Received Packet MIC failure. Type = Group\n");
1681 }
1682 /* Unicast on infrastructure */
1683 else
1684 {
1685 uKeyType = (TI_UINT8)KEY_TKIP_MIC_PAIRWISE;
1686 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_ReceivePacket: Received Packet MIC failure. Type = Pairwise\n");
1687 }
1688
1689
1690 rsn_reportMicFailure (pRxData->hRsn, &uKeyType, sizeof(uKeyType));
1691 RxBufFree(pRxData->hOs, pBuffer);
1692 return;
1693 }
1694
1695 case RX_DESC_STATUS_DRIVER_RX_Q_FAIL:
1696 {
1697 /* Rx queue error - free packet and return */
1698 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_ReceivePacket: Received Packet with Rx queue error \n");
1699
1700 RxBufFree(pRxData->hOs, pBuffer);
1701 return;
1702 }
1703
1704 default:
1705 /* Unknown error - free packet and return */
1706 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR, "rxData_ReceivePacket: Received Packet with unknown status = %d\n", (pRxParams->status & RX_DESC_STATUS_MASK));
1707
1708 RxBufFree(pRxData->hOs, pBuffer);
1709 return;
1710 }
1711
1712 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION , "Receive good Packet\n");
1713
1714 if (rate_PolicyToDrv (pRxParams->rate, &appRate) != TI_OK)
1715 {
1716 TRACE1(pRxData->hReport, REPORT_SEVERITY_ERROR , "rxData_ReceivePacket: can't convert hwRate=0x%x\n", pRxParams->rate);
1717 }
1718
1719 /*
1720 * Set rx attributes
1721 */
1722 RxAttr.channel = pRxParams->channel;
1723 RxAttr.packetInfo = pRxParams->flags;
1724 RxAttr.ePacketType= pRxParams->packet_class_tag;
1725 RxAttr.Rate = appRate;
1726 RxAttr.Rssi = pRxParams->rx_level;
1727 RxAttr.SNR = pRxParams->rx_snr;
1728 RxAttr.status = pRxParams->status & RX_DESC_STATUS_MASK;
1729 /* for now J band not implemented */
1730 RxAttr.band = ((pRxParams->flags & RX_DESC_BAND_MASK) == RX_DESC_BAND_A) ?
1731 RADIO_BAND_5_0_GHZ : RADIO_BAND_2_4_GHZ ;
1732 RxAttr.eScanTag = pRxParams->proccess_id_tag;
1733 /* timestamp is 32 bit so do bytes copy to avoid exception in case the RxInfo is in 2 bytes offset */
1734 os_memoryCopy (pRxData->hOs,
1735 (void *)&(RxAttr.TimeStamp),
1736 (void *)&(pRxParams->timestamp),
1737 sizeof(pRxParams->timestamp) );
1738 RxAttr.TimeStamp = ENDIAN_HANDLE_LONG(RxAttr.TimeStamp);
1739
1740 TRACE8(pRxData->hReport, REPORT_SEVERITY_INFORMATION, "rxData_ReceivePacket: channel=%d, info=0x%x, type=%d, rate=0x%x, RSSI=%d, SNR=%d, status=%d, scan tag=%d\n", RxAttr.channel, RxAttr.packetInfo, RxAttr.ePacketType, RxAttr.Rate, RxAttr.Rssi, RxAttr.SNR, RxAttr.status, RxAttr.eScanTag);
1741
1742 rxData_receivePacketFromWlan (hRxData, pBuffer, &RxAttr);
1743
1744 /*
1745 * Buffer MUST be freed until now
1746 */
1747 }
1748 else
1749 {
1750 TRACE0(pRxData->hReport, REPORT_SEVERITY_ERROR , "rxData_ReceivePacket: null Buffer received");
1751 }
1752 }
1753
1754
1755 /****************************************************************************************
1756 * rxData_RequestForBuffer *
1757 ****************************************************************************************
1758 DESCRIPTION: RX request for buffer
1759 uEncryptionflag API are for GWSI use.
1760 INPUT:
1761
1762 OUTPUT:
1763
1764 RETURN:
1765
1766 ************************************************************************/
rxData_RequestForBuffer(TI_HANDLE hRxData,void ** pBuf,TI_UINT16 aLength,TI_UINT32 uEncryptionflag,PacketClassTag_e ePacketClassTag)1767 static ERxBufferStatus rxData_RequestForBuffer (TI_HANDLE hRxData,
1768 void **pBuf,
1769 TI_UINT16 aLength,
1770 TI_UINT32 uEncryptionflag,
1771 PacketClassTag_e ePacketClassTag)
1772 {
1773 rxData_t *pRxData = (rxData_t *)hRxData;
1774
1775 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION , " RequestForBuffer, length = %d \n",aLength);
1776
1777 *pBuf = RxBufAlloc (pRxData->hOs, aLength, ePacketClassTag);
1778
1779 if (*pBuf)
1780 {
1781 return RX_BUF_ALLOC_COMPLETE;
1782 }
1783 else
1784 {
1785 return RX_BUF_ALLOC_OUT_OF_MEM;
1786 }
1787 }
1788
1789
1790 /*******************************************************************
1791 * DEBUG FUNCTIONS *
1792 *******************************************************************/
1793
1794 #ifdef TI_DBG
1795
1796 /***************************************************************************
1797 * rxData_resetCounters *
1798 ****************************************************************************
1799 * DESCRIPTION: This function reset the Rx Data module counters
1800 *
1801 * INPUTS: hRxData - the object
1802 *
1803 * OUTPUT:
1804 *
1805 * RETURNS: void
1806 ***************************************************************************/
rxData_resetCounters(TI_HANDLE hRxData)1807 void rxData_resetCounters(TI_HANDLE hRxData)
1808 {
1809 rxData_t *pRxData = (rxData_t *)hRxData;
1810
1811 os_memoryZero(pRxData->hOs, &pRxData->rxDataCounters, sizeof(rxDataCounters_t));
1812 }
1813
1814 /***************************************************************************
1815 * rxData_resetDbgCounters *
1816 ****************************************************************************
1817 * DESCRIPTION: This function reset the Rx Data module debug counters
1818 *
1819 * INPUTS: hRxData - the object
1820 *
1821 * OUTPUT:
1822 *
1823 * RETURNS: void
1824 ***************************************************************************/
1825
rxData_resetDbgCounters(TI_HANDLE hRxData)1826 void rxData_resetDbgCounters(TI_HANDLE hRxData)
1827 {
1828 rxData_t *pRxData = (rxData_t *)hRxData;
1829
1830 os_memoryZero(pRxData->hOs, &pRxData->rxDataDbgCounters, sizeof(rxDataDbgCounters_t));
1831 }
1832
1833
1834 /***************************************************************************
1835 * test functions *
1836 ***************************************************************************/
rxData_printRxCounters(TI_HANDLE hRxData)1837 void rxData_printRxCounters (TI_HANDLE hRxData)
1838 {
1839 #ifdef REPORT_LOG
1840 rxData_t *pRxData = (rxData_t *)hRxData;
1841
1842 if (pRxData)
1843 {
1844 WLAN_OS_REPORT(("RecvOk = %d\n", pRxData->rxDataCounters.RecvOk));
1845 WLAN_OS_REPORT(("DirectedBytesRecv = %d\n", pRxData->rxDataCounters.DirectedBytesRecv));
1846 WLAN_OS_REPORT(("DirectedFramesRecv = %d\n", pRxData->rxDataCounters.DirectedFramesRecv));
1847 WLAN_OS_REPORT(("MulticastBytesRecv = %d\n", pRxData->rxDataCounters.MulticastBytesRecv));
1848 WLAN_OS_REPORT(("MulticastFramesRecv = %d\n", pRxData->rxDataCounters.MulticastFramesRecv));
1849 WLAN_OS_REPORT(("BroadcastBytesRecv = %d\n", pRxData->rxDataCounters.BroadcastBytesRecv));
1850 WLAN_OS_REPORT(("BroadcastFramesRecv = %d\n", pRxData->rxDataCounters.BroadcastFramesRecv));
1851
1852 /* debug counters */
1853 WLAN_OS_REPORT(("excludedFrameCounter = %d\n", pRxData->rxDataDbgCounters.excludedFrameCounter));
1854 WLAN_OS_REPORT(("rxDroppedDueToVLANIncludedCnt = %d\n", pRxData->rxDataDbgCounters.rxDroppedDueToVLANIncludedCnt));
1855 WLAN_OS_REPORT(("rxWrongBssTypeCounter = %d\n", pRxData->rxDataDbgCounters.rxWrongBssTypeCounter));
1856 WLAN_OS_REPORT(("rxWrongBssIdCounter = %d\n", pRxData->rxDataDbgCounters.rxWrongBssIdCounter));
1857 WLAN_OS_REPORT(("rcvUnicastFrameInOpenNotify = %d\n", pRxData->rxDataDbgCounters.rcvUnicastFrameInOpenNotify));
1858 }
1859 #endif
1860 }
1861
1862
rxData_printRxBlock(TI_HANDLE hRxData)1863 void rxData_printRxBlock(TI_HANDLE hRxData)
1864 {
1865 #ifdef REPORT_LOG
1866 rxData_t *pRxData = (rxData_t *)hRxData;
1867
1868 WLAN_OS_REPORT(("hCtrlData = 0x%X\n", pRxData->hCtrlData));
1869 WLAN_OS_REPORT(("hMlme = 0x%X\n", pRxData->hMlme));
1870 WLAN_OS_REPORT(("hOs = 0x%X\n", pRxData->hOs));
1871 WLAN_OS_REPORT(("hReport = 0x%X\n", pRxData->hReport));
1872 WLAN_OS_REPORT(("hRsn = 0x%X\n", pRxData->hRsn));
1873 WLAN_OS_REPORT(("hSiteMgr = 0x%X\n", pRxData->hSiteMgr));
1874
1875 WLAN_OS_REPORT(("hCtrlData = 0x%X\n", pRxData->hCtrlData));
1876 WLAN_OS_REPORT(("hMlme = 0x%X\n", pRxData->hMlme));
1877 WLAN_OS_REPORT(("hOs = 0x%X\n", pRxData->hOs));
1878 WLAN_OS_REPORT(("hReport = 0x%X\n", pRxData->hReport));
1879 WLAN_OS_REPORT(("hRsn = 0x%X\n", pRxData->hRsn));
1880 WLAN_OS_REPORT(("hSiteMgr = 0x%X\n", pRxData->hSiteMgr));
1881
1882 WLAN_OS_REPORT(("rxDataPortStatus = %d\n", pRxData->rxDataPortStatus));
1883 WLAN_OS_REPORT(("rxDataExcludeUnencrypted = %d\n", pRxData->rxDataExcludeUnencrypted));
1884 WLAN_OS_REPORT(("rxDataEapolDestination = %d\n", pRxData->rxDataEapolDestination));
1885 #endif
1886 }
1887
1888
rxData_startRxThroughputTimer(TI_HANDLE hRxData)1889 void rxData_startRxThroughputTimer (TI_HANDLE hRxData)
1890 {
1891 rxData_t *pRxData = (rxData_t *)hRxData;
1892
1893 if (!pRxData->rxThroughputTimerEnable)
1894 {
1895 /* reset throughput counter */
1896 pRxData->rxDataCounters.LastSecBytesRecv = 0;
1897 pRxData->rxThroughputTimerEnable = TI_TRUE;
1898
1899 /* start 1 sec throughput timer */
1900 tmr_StartTimer (pRxData->hThroughputTimer,
1901 rxData_printRxThroughput,
1902 (TI_HANDLE)pRxData,
1903 1000,
1904 TI_TRUE);
1905 }
1906 }
1907
1908
rxData_stopRxThroughputTimer(TI_HANDLE hRxData)1909 void rxData_stopRxThroughputTimer (TI_HANDLE hRxData)
1910 {
1911
1912 rxData_t *pRxData = (rxData_t *)hRxData;
1913
1914 if (pRxData->rxThroughputTimerEnable)
1915 {
1916 tmr_StopTimer (pRxData->hThroughputTimer);
1917 pRxData->rxThroughputTimerEnable = TI_FALSE;
1918 }
1919 }
1920
1921
rxData_printRxThroughput(TI_HANDLE hRxData,TI_BOOL bTwdInitOccured)1922 static void rxData_printRxThroughput (TI_HANDLE hRxData, TI_BOOL bTwdInitOccured)
1923 {
1924 rxData_t *pRxData = (rxData_t *)hRxData;
1925
1926 WLAN_OS_REPORT (("\n"));
1927 WLAN_OS_REPORT (("-------------- Rx Throughput Statistics ---------------\n"));
1928 WLAN_OS_REPORT (("Throughput = %d KBits/sec\n", pRxData->rxDataCounters.LastSecBytesRecv * 8 / 1024));
1929
1930 /* reset throughput counter */
1931 pRxData->rxDataCounters.LastSecBytesRecv = 0;
1932 }
1933
rxData_printRxDataFilter(TI_HANDLE hRxData)1934 void rxData_printRxDataFilter (TI_HANDLE hRxData)
1935 {
1936 TI_UINT32 index;
1937 rxData_t *pRxData = (rxData_t *)hRxData;
1938
1939 for (index=0; index<MAX_DATA_FILTERS; index++)
1940 {
1941 if (pRxData->isFilterSet[index])
1942 {
1943 WLAN_OS_REPORT (("index=%d, pattern & mask\n", index));
1944 report_PrintDump(pRxData->filterRequests[index].pattern, pRxData->filterRequests[index].patternLength);
1945 report_PrintDump(pRxData->filterRequests[index].mask, pRxData->filterRequests[index].maskLength);
1946 }
1947 else
1948 {
1949 WLAN_OS_REPORT (("No Filter defined for index-%d\n", index));
1950 }
1951 }
1952 }
1953
1954 #endif /*TI_DBG*/
1955
1956 /****************************************************************************
1957 * rxData_SetReAuthInProgress()
1958 ****************************************************************************
1959 * DESCRIPTION: Sets the ReAuth flag value
1960 *
1961 * INPUTS: hRxData - the object
1962 * value - value to set the flag to
1963 *
1964 * OUTPUT: None
1965 *
1966 * RETURNS: OK or NOK
1967 ****************************************************************************/
rxData_SetReAuthInProgress(TI_HANDLE hRxData,TI_BOOL value)1968 void rxData_SetReAuthInProgress(TI_HANDLE hRxData, TI_BOOL value)
1969 {
1970 rxData_t *pRxData = (rxData_t *)hRxData;
1971
1972 TRACE1(pRxData->hReport, REPORT_SEVERITY_INFORMATION , "Set ReAuth flag to %d\n", value);
1973
1974 pRxData->reAuthInProgress = value;
1975 }
1976
1977 /****************************************************************************
1978 * rxData_IsReAuthInProgress()
1979 ****************************************************************************
1980 * DESCRIPTION: Returns the ReAuth flag value
1981 *
1982 * INPUTS: hRxData - the object
1983 *
1984 * OUTPUT: None
1985 *
1986 * RETURNS: ReAuth flag value
1987 ****************************************************************************/
rxData_IsReAuthInProgress(TI_HANDLE hRxData)1988 TI_BOOL rxData_IsReAuthInProgress(TI_HANDLE hRxData)
1989 {
1990 rxData_t *pRxData = (rxData_t *)hRxData;
1991 return pRxData->reAuthInProgress;
1992 }
1993
1994 /****************************************************************************
1995 * rxData_StartReAuthActiveTimer *
1996 *****************************************************************************
1997 * DESCRIPTION: this function starts the ReAuthActive timer
1998 *
1999 * INPUTS: hRxData - the object
2000 *
2001 * OUTPUT: None
2002 *
2003 * RETURNS: None
2004 ***************************************************************************/
rxData_StartReAuthActiveTimer(TI_HANDLE hRxData)2005 static void rxData_StartReAuthActiveTimer(TI_HANDLE hRxData)
2006 {
2007 rxData_t *pRxData = (rxData_t *)hRxData;
2008 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION , "Start ReAuth Active Timer\n");
2009 tmr_StartTimer (pRxData->reAuthActiveTimer,
2010 reAuthTimeout,
2011 (TI_HANDLE)pRxData,
2012 pRxData->reAuthActiveTimeout,
2013 TI_FALSE);
2014 }
2015
2016 /****************************************************************************
2017 * rxData_StopReAuthActiveTimer *
2018 *****************************************************************************
2019 * DESCRIPTION: this function stops the ReAuthActive timer
2020 *
2021 * INPUTS: hRxData - the object
2022 *
2023 * OUTPUT: None
2024 *
2025 * RETURNS: None
2026 ***************************************************************************/
rxData_StopReAuthActiveTimer(TI_HANDLE hRxData)2027 void rxData_StopReAuthActiveTimer(TI_HANDLE hRxData)
2028 {
2029 rxData_t *pRxData = (rxData_t *)hRxData;
2030 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION , "Stop ReAuth Active Timer\n");
2031 tmr_StopTimer (pRxData->reAuthActiveTimer);
2032 }
2033
2034 /****************************************************************************
2035 * reAuthTimeout *
2036 *****************************************************************************
2037 * DESCRIPTION: this function ia called when the ReAuthActive timer elapses
2038 * It resets the Reauth flag and restore the PS state.
2039 * It also sends RE_AUTH_TERMINATED event to upper layer.
2040 *
2041 * INPUTS: hRxData - the object
2042 *
2043 * OUTPUT: None
2044 *
2045 * RETURNS: None
2046 ***************************************************************************/
reAuthTimeout(TI_HANDLE hRxData,TI_BOOL bTwdInitOccured)2047 static void reAuthTimeout(TI_HANDLE hRxData, TI_BOOL bTwdInitOccured)
2048 {
2049 rxData_t *pRxData = (rxData_t *)hRxData;
2050
2051 TRACE0(pRxData->hReport, REPORT_SEVERITY_INFORMATION , "ReAuth Active Timeout\n");
2052 rxData_SetReAuthInProgress(pRxData, TI_FALSE);
2053 rxData_ReauthDisablePriority(pRxData);
2054 EvHandlerSendEvent(pRxData->hEvHandler, IPC_EVENT_RE_AUTH_TERMINATED, NULL, 0);
2055 }
2056
rxData_ReauthEnablePriority(TI_HANDLE hRxData)2057 void rxData_ReauthEnablePriority(TI_HANDLE hRxData)
2058 {
2059 rxData_t *pRxData = (rxData_t *)hRxData;
2060 paramInfo_t param;
2061
2062 param.paramType = POWER_MGR_ENABLE_PRIORITY;
2063 param.content.powerMngPriority = POWER_MANAGER_REAUTH_PRIORITY;
2064 powerMgr_setParam(pRxData->hPowerMgr,¶m);
2065 }
2066
rxData_ReauthDisablePriority(TI_HANDLE hRxData)2067 void rxData_ReauthDisablePriority(TI_HANDLE hRxData)
2068 {
2069 rxData_t *pRxData = (rxData_t *)hRxData;
2070 paramInfo_t param;
2071
2072 param.paramType = POWER_MGR_DISABLE_PRIORITY;
2073 param.content.powerMngPriority = POWER_MANAGER_REAUTH_PRIORITY;
2074 powerMgr_setParam(pRxData->hPowerMgr,¶m);
2075 }
2076