Staging: add rt2870 wireless driver
[deliverable/linux.git] / drivers / staging / rt2870 / common / action.c
1 /*
2 *************************************************************************
3 * Ralink Tech Inc.
4 * 5F., No.36, Taiyuan St., Jhubei City,
5 * Hsinchu County 302,
6 * Taiwan, R.O.C.
7 *
8 * (c) Copyright 2002-2007, Ralink Technology, Inc.
9 *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
24 * *
25 *************************************************************************
26
27 Module Name:
28 action.c
29
30 Abstract:
31 Handle association related requests either from WSTA or from local MLME
32
33 Revision History:
34 Who When What
35 -------- ---------- ----------------------------------------------
36 Jan Lee 2006 created for rt2860
37 */
38
39 #include "../rt_config.h"
40 #include "action.h"
41
42
43 static VOID ReservedAction(
44 IN PRTMP_ADAPTER pAd,
45 IN MLME_QUEUE_ELEM *Elem);
46
47 /*
48 ==========================================================================
49 Description:
50 association state machine init, including state transition and timer init
51 Parameters:
52 S - pointer to the association state machine
53 Note:
54 The state machine looks like the following
55
56 ASSOC_IDLE
57 MT2_MLME_DISASSOC_REQ mlme_disassoc_req_action
58 MT2_PEER_DISASSOC_REQ peer_disassoc_action
59 MT2_PEER_ASSOC_REQ drop
60 MT2_PEER_REASSOC_REQ drop
61 MT2_CLS3ERR cls3err_action
62 ==========================================================================
63 */
64 VOID ActionStateMachineInit(
65 IN PRTMP_ADAPTER pAd,
66 IN STATE_MACHINE *S,
67 OUT STATE_MACHINE_FUNC Trans[])
68 {
69 StateMachineInit(S, (STATE_MACHINE_FUNC *)Trans, MAX_ACT_STATE, MAX_ACT_MSG, (STATE_MACHINE_FUNC)Drop, ACT_IDLE, ACT_MACHINE_BASE);
70
71 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_SPECTRUM_CATE, (STATE_MACHINE_FUNC)PeerSpectrumAction);
72 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_QOS_CATE, (STATE_MACHINE_FUNC)PeerQOSAction);
73
74 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_DLS_CATE, (STATE_MACHINE_FUNC)ReservedAction);
75 #ifdef QOS_DLS_SUPPORT
76 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_DLS_CATE, (STATE_MACHINE_FUNC)PeerDLSAction);
77 #endif // QOS_DLS_SUPPORT //
78
79 #ifdef DOT11_N_SUPPORT
80 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_BA_CATE, (STATE_MACHINE_FUNC)PeerBAAction);
81 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_HT_CATE, (STATE_MACHINE_FUNC)PeerHTAction);
82 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_ADD_BA_CATE, (STATE_MACHINE_FUNC)MlmeADDBAAction);
83 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_ORI_DELBA_CATE, (STATE_MACHINE_FUNC)MlmeDELBAAction);
84 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_REC_DELBA_CATE, (STATE_MACHINE_FUNC)MlmeDELBAAction);
85 #endif // DOT11_N_SUPPORT //
86
87 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_PUBLIC_CATE, (STATE_MACHINE_FUNC)PeerPublicAction);
88 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_RM_CATE, (STATE_MACHINE_FUNC)PeerRMAction);
89
90 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_QOS_CATE, (STATE_MACHINE_FUNC)MlmeQOSAction);
91 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_DLS_CATE, (STATE_MACHINE_FUNC)MlmeDLSAction);
92 StateMachineSetAction(S, ACT_IDLE, MT2_ACT_INVALID, (STATE_MACHINE_FUNC)MlmeInvalidAction);
93 }
94
95 #ifdef DOT11_N_SUPPORT
96 VOID MlmeADDBAAction(
97 IN PRTMP_ADAPTER pAd,
98 IN MLME_QUEUE_ELEM *Elem)
99
100 {
101 MLME_ADDBA_REQ_STRUCT *pInfo;
102 UCHAR Addr[6];
103 PUCHAR pOutBuffer = NULL;
104 NDIS_STATUS NStatus;
105 ULONG Idx;
106 FRAME_ADDBA_REQ Frame;
107 ULONG FrameLen;
108 BA_ORI_ENTRY *pBAEntry = NULL;
109
110 pInfo = (MLME_ADDBA_REQ_STRUCT *)Elem->Msg;
111 NdisZeroMemory(&Frame, sizeof(FRAME_ADDBA_REQ));
112
113 if(MlmeAddBAReqSanity(pAd, Elem->Msg, Elem->MsgLen, Addr))
114 {
115 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
116 if(NStatus != NDIS_STATUS_SUCCESS)
117 {
118 DBGPRINT(RT_DEBUG_TRACE,("BA - MlmeADDBAAction() allocate memory failed \n"));
119 return;
120 }
121 // 1. find entry
122 Idx = pAd->MacTab.Content[pInfo->Wcid].BAOriWcidArray[pInfo->TID];
123 if (Idx == 0)
124 {
125 MlmeFreeMemory(pAd, pOutBuffer);
126 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeADDBAAction() can't find BAOriEntry \n"));
127 return;
128 }
129 else
130 {
131 pBAEntry =&pAd->BATable.BAOriEntry[Idx];
132 }
133
134 #ifdef CONFIG_STA_SUPPORT
135 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
136 {
137 if (ADHOC_ON(pAd))
138 ActHeaderInit(pAd, &Frame.Hdr, pInfo->pAddr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
139 else
140 #ifdef QOS_DLS_SUPPORT
141 if (pAd->MacTab.Content[pInfo->Wcid].ValidAsDls)
142 ActHeaderInit(pAd, &Frame.Hdr, pInfo->pAddr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
143 else
144 #endif // QOS_DLS_SUPPORT //
145 ActHeaderInit(pAd, &Frame.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pInfo->pAddr);
146
147 }
148 #endif // CONFIG_STA_SUPPORT //
149
150 Frame.Category = CATEGORY_BA;
151 Frame.Action = ADDBA_REQ;
152 Frame.BaParm.AMSDUSupported = 0;
153 Frame.BaParm.BAPolicy = IMMED_BA;
154 Frame.BaParm.TID = pInfo->TID;
155 Frame.BaParm.BufSize = pInfo->BaBufSize;
156 Frame.Token = pInfo->Token;
157 Frame.TimeOutValue = pInfo->TimeOutValue;
158 Frame.BaStartSeq.field.FragNum = 0;
159 Frame.BaStartSeq.field.StartSeq = pAd->MacTab.Content[pInfo->Wcid].TxSeq[pInfo->TID];
160
161 *(USHORT *)(&Frame.BaParm) = cpu2le16(*(USHORT *)(&Frame.BaParm));
162 Frame.TimeOutValue = cpu2le16(Frame.TimeOutValue);
163 Frame.BaStartSeq.word = cpu2le16(Frame.BaStartSeq.word);
164
165 MakeOutgoingFrame(pOutBuffer, &FrameLen,
166 sizeof(FRAME_ADDBA_REQ), &Frame,
167 END_OF_ARGS);
168 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
169 MlmeFreeMemory(pAd, pOutBuffer);
170
171 DBGPRINT(RT_DEBUG_TRACE, ("BA - Send ADDBA request. StartSeq = %x, FrameLen = %ld. BufSize = %d\n", Frame.BaStartSeq.field.StartSeq, FrameLen, Frame.BaParm.BufSize));
172 }
173 }
174
175 /*
176 ==========================================================================
177 Description:
178 send DELBA and delete BaEntry if any
179 Parametrs:
180 Elem - MLME message MLME_DELBA_REQ_STRUCT
181
182 IRQL = DISPATCH_LEVEL
183
184 ==========================================================================
185 */
186 VOID MlmeDELBAAction(
187 IN PRTMP_ADAPTER pAd,
188 IN MLME_QUEUE_ELEM *Elem)
189 {
190 MLME_DELBA_REQ_STRUCT *pInfo;
191 PUCHAR pOutBuffer = NULL;
192 PUCHAR pOutBuffer2 = NULL;
193 NDIS_STATUS NStatus;
194 ULONG Idx;
195 FRAME_DELBA_REQ Frame;
196 ULONG FrameLen;
197 FRAME_BAR FrameBar;
198
199 pInfo = (MLME_DELBA_REQ_STRUCT *)Elem->Msg;
200 // must send back DELBA
201 NdisZeroMemory(&Frame, sizeof(FRAME_DELBA_REQ));
202 DBGPRINT(RT_DEBUG_TRACE, ("==> MlmeDELBAAction(), Initiator(%d) \n", pInfo->Initiator));
203
204 if(MlmeDelBAReqSanity(pAd, Elem->Msg, Elem->MsgLen))
205 {
206 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
207 if(NStatus != NDIS_STATUS_SUCCESS)
208 {
209 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeDELBAAction() allocate memory failed 1. \n"));
210 return;
211 }
212
213 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer2); //Get an unused nonpaged memory
214 if(NStatus != NDIS_STATUS_SUCCESS)
215 {
216 MlmeFreeMemory(pAd, pOutBuffer);
217 DBGPRINT(RT_DEBUG_ERROR, ("BA - MlmeDELBAAction() allocate memory failed 2. \n"));
218 return;
219 }
220
221 // SEND BAR (Send BAR to refresh peer reordering buffer.)
222 Idx = pAd->MacTab.Content[pInfo->Wcid].BAOriWcidArray[pInfo->TID];
223
224 #ifdef CONFIG_STA_SUPPORT
225 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
226 BarHeaderInit(pAd, &FrameBar, pAd->MacTab.Content[pInfo->Wcid].Addr, pAd->CurrentAddress);
227 #endif // CONFIG_STA_SUPPORT //
228
229 FrameBar.StartingSeq.field.FragNum = 0; // make sure sequence not clear in DEL funciton.
230 FrameBar.StartingSeq.field.StartSeq = pAd->MacTab.Content[pInfo->Wcid].TxSeq[pInfo->TID]; // make sure sequence not clear in DEL funciton.
231 FrameBar.BarControl.TID = pInfo->TID; // make sure sequence not clear in DEL funciton.
232 FrameBar.BarControl.ACKPolicy = IMMED_BA; // make sure sequence not clear in DEL funciton.
233 FrameBar.BarControl.Compressed = 1; // make sure sequence not clear in DEL funciton.
234 FrameBar.BarControl.MTID = 0; // make sure sequence not clear in DEL funciton.
235
236 MakeOutgoingFrame(pOutBuffer2, &FrameLen,
237 sizeof(FRAME_BAR), &FrameBar,
238 END_OF_ARGS);
239 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer2, FrameLen);
240 MlmeFreeMemory(pAd, pOutBuffer2);
241 DBGPRINT(RT_DEBUG_TRACE,("BA - MlmeDELBAAction() . Send BAR to refresh peer reordering buffer \n"));
242
243 // SEND DELBA FRAME
244 FrameLen = 0;
245 #ifdef CONFIG_STA_SUPPORT
246 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
247 {
248 if (ADHOC_ON(pAd))
249 ActHeaderInit(pAd, &Frame.Hdr, pAd->MacTab.Content[pInfo->Wcid].Addr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
250 else
251 #ifdef QOS_DLS_SUPPORT
252 if (pAd->MacTab.Content[pInfo->Wcid].ValidAsDls)
253 ActHeaderInit(pAd, &Frame.Hdr, pAd->MacTab.Content[pInfo->Wcid].Addr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
254 else
255 #endif // QOS_DLS_SUPPORT //
256 ActHeaderInit(pAd, &Frame.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pAd->MacTab.Content[pInfo->Wcid].Addr);
257 }
258 #endif // CONFIG_STA_SUPPORT //
259 Frame.Category = CATEGORY_BA;
260 Frame.Action = DELBA;
261 Frame.DelbaParm.Initiator = pInfo->Initiator;
262 Frame.DelbaParm.TID = pInfo->TID;
263 Frame.ReasonCode = 39; // Time Out
264 *(USHORT *)(&Frame.DelbaParm) = cpu2le16(*(USHORT *)(&Frame.DelbaParm));
265 Frame.ReasonCode = cpu2le16(Frame.ReasonCode);
266
267 MakeOutgoingFrame(pOutBuffer, &FrameLen,
268 sizeof(FRAME_DELBA_REQ), &Frame,
269 END_OF_ARGS);
270 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
271 MlmeFreeMemory(pAd, pOutBuffer);
272 DBGPRINT(RT_DEBUG_TRACE, ("BA - MlmeDELBAAction() . 3 DELBA sent. Initiator(%d)\n", pInfo->Initiator));
273 }
274 }
275 #endif // DOT11_N_SUPPORT //
276
277 VOID MlmeQOSAction(
278 IN PRTMP_ADAPTER pAd,
279 IN MLME_QUEUE_ELEM *Elem)
280 {
281 }
282
283 VOID MlmeDLSAction(
284 IN PRTMP_ADAPTER pAd,
285 IN MLME_QUEUE_ELEM *Elem)
286 {
287 }
288
289 VOID MlmeInvalidAction(
290 IN PRTMP_ADAPTER pAd,
291 IN MLME_QUEUE_ELEM *Elem)
292 {
293 //PUCHAR pOutBuffer = NULL;
294 //Return the receiving frame except the MSB of category filed set to 1. 7.3.1.11
295 }
296
297 VOID PeerQOSAction(
298 IN PRTMP_ADAPTER pAd,
299 IN MLME_QUEUE_ELEM *Elem)
300 {
301 }
302
303 #ifdef QOS_DLS_SUPPORT
304 VOID PeerDLSAction(
305 IN PRTMP_ADAPTER pAd,
306 IN MLME_QUEUE_ELEM *Elem)
307 {
308 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
309
310 switch(Action)
311 {
312 case ACTION_DLS_REQUEST:
313 #ifdef CONFIG_STA_SUPPORT
314 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
315 PeerDlsReqAction(pAd, Elem);
316 #endif // CONFIG_STA_SUPPORT //
317 break;
318
319 case ACTION_DLS_RESPONSE:
320 #ifdef CONFIG_STA_SUPPORT
321 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
322 PeerDlsRspAction(pAd, Elem);
323 #endif // CONFIG_STA_SUPPORT //
324 break;
325
326 case ACTION_DLS_TEARDOWN:
327 #ifdef CONFIG_STA_SUPPORT
328 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
329 PeerDlsTearDownAction(pAd, Elem);
330 #endif // CONFIG_STA_SUPPORT //
331 break;
332 }
333 }
334 #endif // QOS_DLS_SUPPORT //
335
336 #ifdef DOT11_N_SUPPORT
337 VOID PeerBAAction(
338 IN PRTMP_ADAPTER pAd,
339 IN MLME_QUEUE_ELEM *Elem)
340 {
341 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
342
343 switch(Action)
344 {
345 case ADDBA_REQ:
346 PeerAddBAReqAction(pAd,Elem);
347 break;
348 case ADDBA_RESP:
349 PeerAddBARspAction(pAd,Elem);
350 break;
351 case DELBA:
352 PeerDelBAAction(pAd,Elem);
353 break;
354 }
355 }
356
357
358 #ifdef DOT11N_DRAFT3
359
360 #ifdef CONFIG_STA_SUPPORT
361 VOID StaPublicAction(
362 IN PRTMP_ADAPTER pAd,
363 IN UCHAR Bss2040Coexist)
364 {
365 BSS_2040_COEXIST_IE BssCoexist;
366 MLME_SCAN_REQ_STRUCT ScanReq;
367
368 BssCoexist.word = Bss2040Coexist;
369 // AP asks Station to return a 20/40 BSS Coexistence mgmt frame. So we first starts a scan, then send back 20/40 BSS Coexistence mgmt frame
370 if ((BssCoexist.field.InfoReq == 1) && (OPSTATUS_TEST_FLAG(pAd, fOP_STATUS_SCAN_2040)))
371 {
372 // Clear record first. After scan , will update those bit and send back to transmiter.
373 pAd->CommonCfg.BSSCoexist2040.field.InfoReq = 1;
374 pAd->CommonCfg.BSSCoexist2040.field.Intolerant40 = 0;
375 pAd->CommonCfg.BSSCoexist2040.field.BSS20WidthReq = 0;
376 // Fill out stuff for scan request
377 ScanParmFill(pAd, &ScanReq, ZeroSsid, 0, BSS_ANY, SCAN_2040_BSS_COEXIST);
378 MlmeEnqueue(pAd, SYNC_STATE_MACHINE, MT2_MLME_SCAN_REQ, sizeof(MLME_SCAN_REQ_STRUCT), &ScanReq);
379 pAd->Mlme.CntlMachine.CurrState = CNTL_WAIT_OID_LIST_SCAN;
380 }
381 }
382
383
384 /*
385 Description : Build Intolerant Channel Rerpot from Trigger event table.
386 return : how many bytes copied.
387 */
388 ULONG BuildIntolerantChannelRep(
389 IN PRTMP_ADAPTER pAd,
390 IN PUCHAR pDest)
391 {
392 ULONG FrameLen = 0;
393 ULONG ReadOffset = 0;
394 UCHAR i;
395 UCHAR LastRegClass = 0xff;
396 PUCHAR pLen;
397
398 for ( i = 0;i < MAX_TRIGGER_EVENT;i++)
399 {
400 if (pAd->CommonCfg.TriggerEventTab.EventA[i].bValid == TRUE)
401 {
402 if (pAd->CommonCfg.TriggerEventTab.EventA[i].RegClass == LastRegClass)
403 {
404 *(pDest + ReadOffset) = (UCHAR)pAd->CommonCfg.TriggerEventTab.EventA[i].Channel;
405 *pLen++;
406 ReadOffset++;
407 FrameLen++;
408 }
409 else
410 {
411 *(pDest + ReadOffset) = IE_2040_BSS_INTOLERANT_REPORT; // IE
412 *(pDest + ReadOffset + 1) = 2; // Len = RegClass byte + channel byte.
413 pLen = pDest + ReadOffset + 1;
414 LastRegClass = pAd->CommonCfg.TriggerEventTab.EventA[i].RegClass;
415 *(pDest + ReadOffset + 2) = LastRegClass; // Len = RegClass byte + channel byte.
416 *(pDest + ReadOffset + 3) = (UCHAR)pAd->CommonCfg.TriggerEventTab.EventA[i].Channel;
417 FrameLen += 4;
418 ReadOffset += 4;
419 }
420
421 }
422 }
423 return FrameLen;
424 }
425
426
427 /*
428 Description : Send 20/40 BSS Coexistence Action frame If one trigger event is triggered.
429 */
430 VOID Send2040CoexistAction(
431 IN PRTMP_ADAPTER pAd,
432 IN UCHAR Wcid,
433 IN BOOLEAN bAddIntolerantCha)
434 {
435 PUCHAR pOutBuffer = NULL;
436 NDIS_STATUS NStatus;
437 FRAME_ACTION_HDR Frame;
438 ULONG FrameLen;
439 ULONG IntolerantChaRepLen;
440
441 IntolerantChaRepLen = 0;
442 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
443 if(NStatus != NDIS_STATUS_SUCCESS)
444 {
445 DBGPRINT(RT_DEBUG_ERROR,("ACT - Send2040CoexistAction() allocate memory failed \n"));
446 return;
447 }
448 ActHeaderInit(pAd, &Frame.Hdr, pAd->MacTab.Content[Wcid].Addr, pAd->CommonCfg.Bssid);
449 Frame.Category = CATEGORY_PUBLIC;
450 Frame.Action = ACTION_BSS_2040_COEXIST;
451
452 MakeOutgoingFrame(pOutBuffer, &FrameLen,
453 sizeof(FRAME_ACTION_HDR), &Frame,
454 END_OF_ARGS);
455
456 *(pOutBuffer + FrameLen) = pAd->CommonCfg.BSSCoexist2040.word;
457 FrameLen++;
458
459 if (bAddIntolerantCha == TRUE)
460 IntolerantChaRepLen = BuildIntolerantChannelRep(pAd, pOutBuffer + FrameLen);
461
462 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen + IntolerantChaRepLen);
463 DBGPRINT(RT_DEBUG_ERROR,("ACT - Send2040CoexistAction( BSSCoexist2040 = 0x%x ) \n", pAd->CommonCfg.BSSCoexist2040.word));
464
465 }
466
467
468 /*
469 ==========================================================================
470 Description:
471 After scan, Update 20/40 BSS Coexistence IE and send out.
472 According to 802.11n D3.03 11.14.10
473
474 Parameters:
475 ==========================================================================
476 */
477 VOID Update2040CoexistFrameAndNotify(
478 IN PRTMP_ADAPTER pAd,
479 IN UCHAR Wcid,
480 IN BOOLEAN bAddIntolerantCha)
481 {
482 BSS_2040_COEXIST_IE OldValue;
483
484 OldValue.word = pAd->CommonCfg.BSSCoexist2040.word;
485 if ((pAd->CommonCfg.TriggerEventTab.EventANo > 0) || (pAd->CommonCfg.TriggerEventTab.EventBCountDown > 0))
486 pAd->CommonCfg.BSSCoexist2040.field.BSS20WidthReq = 1;
487
488 // Need to check !!!!
489 // How STA will set Intolerant40 if implementation dependent. Now we don't set this bit first.!!!!!
490 // So Only check BSS20WidthReq change.
491 if (OldValue.field.BSS20WidthReq != pAd->CommonCfg.BSSCoexist2040.field.BSS20WidthReq)
492 {
493 Send2040CoexistAction(pAd, Wcid, bAddIntolerantCha);
494 }
495 }
496 #endif // CONFIG_STA_SUPPORT //
497
498
499 BOOLEAN ChannelSwitchSanityCheck(
500 IN PRTMP_ADAPTER pAd,
501 IN UCHAR Wcid,
502 IN UCHAR NewChannel,
503 IN UCHAR Secondary)
504 {
505 UCHAR i;
506
507 if (Wcid >= MAX_LEN_OF_MAC_TABLE)
508 return FALSE;
509
510 if ((NewChannel > 7) && (Secondary == 1))
511 return FALSE;
512
513 if ((NewChannel < 5) && (Secondary == 3))
514 return FALSE;
515
516 // 0. Check if new channel is in the channellist.
517 for (i = 0;i < pAd->ChannelListNum;i++)
518 {
519 if (pAd->ChannelList[i].Channel == NewChannel)
520 {
521 break;
522 }
523 }
524
525 if (i == pAd->ChannelListNum)
526 return FALSE;
527
528 return TRUE;
529 }
530
531
532 VOID ChannelSwitchAction(
533 IN PRTMP_ADAPTER pAd,
534 IN UCHAR Wcid,
535 IN UCHAR NewChannel,
536 IN UCHAR Secondary)
537 {
538 UCHAR BBPValue = 0;
539 ULONG MACValue;
540
541 DBGPRINT(RT_DEBUG_TRACE,("SPECTRUM - ChannelSwitchAction(NewChannel = %d , Secondary = %d) \n", NewChannel, Secondary));
542
543 if (ChannelSwitchSanityCheck(pAd, Wcid, NewChannel, Secondary) == FALSE)
544 return;
545
546 // 1. Switches to BW = 20.
547 if (Secondary == 0)
548 {
549 RTMP_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &BBPValue);
550 BBPValue&= (~0x18);
551 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, BBPValue);
552 if (pAd->MACVersion == 0x28600100)
553 {
554 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R69, 0x16);
555 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R70, 0x08);
556 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R73, 0x11);
557 DBGPRINT(RT_DEBUG_TRACE, ("!!!rt2860C !!! \n" ));
558 }
559 pAd->CommonCfg.BBPCurrentBW = BW_20;
560 pAd->CommonCfg.Channel = NewChannel;
561 pAd->CommonCfg.CentralChannel = pAd->CommonCfg.Channel;
562 AsicSwitchChannel(pAd, pAd->CommonCfg.Channel,FALSE);
563 AsicLockChannel(pAd, pAd->CommonCfg.Channel);
564 pAd->MacTab.Content[Wcid].HTPhyMode.field.BW = 0;
565 DBGPRINT(RT_DEBUG_TRACE, ("!!!20MHz !!! \n" ));
566 }
567 // 1. Switches to BW = 40 And Station supports BW = 40.
568 else if (((Secondary == 1) || (Secondary == 3)) && (pAd->CommonCfg.HtCapability.HtCapInfo.ChannelWidth == 1))
569 {
570 pAd->CommonCfg.Channel = NewChannel;
571
572 if (Secondary == 1)
573 {
574 // Secondary above.
575 pAd->CommonCfg.CentralChannel = pAd->CommonCfg.Channel + 2;
576 RTMP_IO_READ32(pAd, TX_BAND_CFG, &MACValue);
577 MACValue &= 0xfe;
578 RTMP_IO_WRITE32(pAd, TX_BAND_CFG, MACValue);
579 RTMP_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &BBPValue);
580 BBPValue&= (~0x18);
581 BBPValue|= (0x10);
582 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, BBPValue);
583 RTMP_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BBPValue);
584 BBPValue&= (~0x20);
585 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BBPValue);
586 DBGPRINT(RT_DEBUG_TRACE, ("!!!40MHz Lower LINK UP !!! Control Channel at Below. Central = %d \n", pAd->CommonCfg.CentralChannel ));
587 }
588 else
589 {
590 // Secondary below.
591 pAd->CommonCfg.CentralChannel = pAd->CommonCfg.Channel - 2;
592 RTMP_IO_READ32(pAd, TX_BAND_CFG, &MACValue);
593 MACValue &= 0xfe;
594 MACValue |= 0x1;
595 RTMP_IO_WRITE32(pAd, TX_BAND_CFG, MACValue);
596 RTMP_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R4, &BBPValue);
597 BBPValue&= (~0x18);
598 BBPValue|= (0x10);
599 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R4, BBPValue);
600 RTMP_BBP_IO_READ8_BY_REG_ID(pAd, BBP_R3, &BBPValue);
601 BBPValue&= (~0x20);
602 BBPValue|= (0x20);
603 RTMP_BBP_IO_WRITE8_BY_REG_ID(pAd, BBP_R3, BBPValue);
604 DBGPRINT(RT_DEBUG_TRACE, ("!!!40MHz Upper LINK UP !!! Control Channel at UpperCentral = %d \n", pAd->CommonCfg.CentralChannel ));
605 }
606 pAd->CommonCfg.BBPCurrentBW = BW_40;
607 AsicSwitchChannel(pAd, pAd->CommonCfg.CentralChannel, FALSE);
608 AsicLockChannel(pAd, pAd->CommonCfg.CentralChannel);
609 pAd->MacTab.Content[Wcid].HTPhyMode.field.BW = 1;
610 }
611 }
612 #endif // DOT11N_DRAFT3 //
613 #endif // DOT11_N_SUPPORT //
614
615 VOID PeerPublicAction(
616 IN PRTMP_ADAPTER pAd,
617 IN MLME_QUEUE_ELEM *Elem)
618 {
619 #ifdef DOT11N_DRAFT3
620 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
621 #endif // DOT11N_DRAFT3 //
622
623 if (Elem->Wcid >= MAX_LEN_OF_MAC_TABLE)
624 return;
625
626 #ifdef DOT11N_DRAFT3
627 switch(Action)
628 {
629 case ACTION_BSS_2040_COEXIST: // Format defined in IEEE 7.4.7a.1 in 11n Draf3.03
630 {
631 //UCHAR BssCoexist;
632 BSS_2040_COEXIST_ELEMENT *pCoexistInfo;
633 BSS_2040_COEXIST_IE *pBssCoexistIe;
634 BSS_2040_INTOLERANT_CH_REPORT *pIntolerantReport = NULL;
635
636 if (Elem->MsgLen <= (LENGTH_802_11 + sizeof(BSS_2040_COEXIST_ELEMENT)) )
637 {
638 DBGPRINT(RT_DEBUG_ERROR, ("ACTION - 20/40 BSS Coexistence Management Frame length too short! len = %ld!\n", Elem->MsgLen));
639 break;
640 }
641 DBGPRINT(RT_DEBUG_TRACE, ("ACTION - 20/40 BSS Coexistence Management action----> \n"));
642 hex_dump("CoexistenceMgmtFrame", Elem->Msg, Elem->MsgLen);
643
644
645 pCoexistInfo = (BSS_2040_COEXIST_ELEMENT *) &Elem->Msg[LENGTH_802_11+2];
646 //hex_dump("CoexistInfo", (PUCHAR)pCoexistInfo, sizeof(BSS_2040_COEXIST_ELEMENT));
647 if (Elem->MsgLen >= (LENGTH_802_11 + sizeof(BSS_2040_COEXIST_ELEMENT) + sizeof(BSS_2040_INTOLERANT_CH_REPORT)))
648 {
649 pIntolerantReport = (BSS_2040_INTOLERANT_CH_REPORT *)((PUCHAR)pCoexistInfo + sizeof(BSS_2040_COEXIST_ELEMENT));
650 }
651 //hex_dump("IntolerantReport ", (PUCHAR)pIntolerantReport, sizeof(BSS_2040_INTOLERANT_CH_REPORT));
652
653 pBssCoexistIe = (BSS_2040_COEXIST_IE *)(&pCoexistInfo->BssCoexistIe);
654
655 #ifdef CONFIG_STA_SUPPORT
656 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
657 {
658 if (INFRA_ON(pAd))
659 {
660 StaPublicAction(pAd, pCoexistInfo);
661 }
662 }
663 #endif // CONFIG_STA_SUPPORT //
664
665 }
666 break;
667 }
668
669 #endif // DOT11N_DRAFT3 //
670
671 }
672
673
674 static VOID ReservedAction(
675 IN PRTMP_ADAPTER pAd,
676 IN MLME_QUEUE_ELEM *Elem)
677 {
678 UCHAR Category;
679
680 if (Elem->MsgLen <= LENGTH_802_11)
681 {
682 return;
683 }
684
685 Category = Elem->Msg[LENGTH_802_11];
686 DBGPRINT(RT_DEBUG_TRACE,("Rcv reserved category(%d) Action Frame\n", Category));
687 hex_dump("Reserved Action Frame", &Elem->Msg[0], Elem->MsgLen);
688 }
689
690 VOID PeerRMAction(
691 IN PRTMP_ADAPTER pAd,
692 IN MLME_QUEUE_ELEM *Elem)
693
694 {
695 return;
696 }
697
698 #ifdef DOT11_N_SUPPORT
699 static VOID respond_ht_information_exchange_action(
700 IN PRTMP_ADAPTER pAd,
701 IN MLME_QUEUE_ELEM *Elem)
702 {
703 PUCHAR pOutBuffer = NULL;
704 NDIS_STATUS NStatus;
705 ULONG FrameLen;
706 FRAME_HT_INFO HTINFOframe, *pFrame;
707 UCHAR *pAddr;
708
709
710 // 2. Always send back ADDBA Response
711 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
712
713 if (NStatus != NDIS_STATUS_SUCCESS)
714 {
715 DBGPRINT(RT_DEBUG_TRACE,("ACTION - respond_ht_information_exchange_action() allocate memory failed \n"));
716 return;
717 }
718
719 // get RA
720 pFrame = (FRAME_HT_INFO *) &Elem->Msg[0];
721 pAddr = pFrame->Hdr.Addr2;
722
723 NdisZeroMemory(&HTINFOframe, sizeof(FRAME_HT_INFO));
724 // 2-1. Prepare ADDBA Response frame.
725 #ifdef CONFIG_STA_SUPPORT
726 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
727 {
728 if (ADHOC_ON(pAd))
729 ActHeaderInit(pAd, &HTINFOframe.Hdr, pAddr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
730 else
731 ActHeaderInit(pAd, &HTINFOframe.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pAddr);
732 }
733 #endif // CONFIG_STA_SUPPORT //
734
735 HTINFOframe.Category = CATEGORY_HT;
736 HTINFOframe.Action = HT_INFO_EXCHANGE;
737 HTINFOframe.HT_Info.Request = 0;
738 HTINFOframe.HT_Info.Forty_MHz_Intolerant = pAd->CommonCfg.HtCapability.HtCapInfo.Forty_Mhz_Intolerant;
739 HTINFOframe.HT_Info.STA_Channel_Width = pAd->CommonCfg.AddHTInfo.AddHtInfo.RecomWidth;
740
741 MakeOutgoingFrame(pOutBuffer, &FrameLen,
742 sizeof(FRAME_HT_INFO), &HTINFOframe,
743 END_OF_ARGS);
744
745 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
746 MlmeFreeMemory(pAd, pOutBuffer);
747 }
748
749
750 #ifdef DOT11N_DRAFT3
751 VOID SendNotifyBWActionFrame(
752 IN PRTMP_ADAPTER pAd,
753 IN UCHAR Wcid,
754 IN UCHAR apidx)
755 {
756 PUCHAR pOutBuffer = NULL;
757 NDIS_STATUS NStatus;
758 FRAME_ACTION_HDR Frame;
759 ULONG FrameLen;
760 PUCHAR pAddr1;
761
762
763 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
764 if(NStatus != NDIS_STATUS_SUCCESS)
765 {
766 DBGPRINT(RT_DEBUG_ERROR,("ACT - SendNotifyBWAction() allocate memory failed \n"));
767 return;
768 }
769
770 if (Wcid == MCAST_WCID)
771 pAddr1 = &BROADCAST_ADDR[0];
772 else
773 pAddr1 = pAd->MacTab.Content[Wcid].Addr;
774 ActHeaderInit(pAd, &Frame.Hdr, pAddr1, pAd->ApCfg.MBSSID[apidx].Bssid, pAd->ApCfg.MBSSID[apidx].Bssid);
775
776 Frame.Category = CATEGORY_HT;
777 Frame.Action = NOTIFY_BW_ACTION;
778
779 MakeOutgoingFrame(pOutBuffer, &FrameLen,
780 sizeof(FRAME_ACTION_HDR), &Frame,
781 END_OF_ARGS);
782
783 *(pOutBuffer + FrameLen) = pAd->CommonCfg.AddHTInfo.AddHtInfo.RecomWidth;
784 FrameLen++;
785
786
787 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
788 DBGPRINT(RT_DEBUG_TRACE,("ACT - SendNotifyBWAction(NotifyBW= %d)!\n", pAd->CommonCfg.AddHTInfo.AddHtInfo.RecomWidth));
789
790 }
791 #endif // DOT11N_DRAFT3 //
792
793
794 VOID PeerHTAction(
795 IN PRTMP_ADAPTER pAd,
796 IN MLME_QUEUE_ELEM *Elem)
797 {
798 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
799
800 if (Elem->Wcid >= MAX_LEN_OF_MAC_TABLE)
801 return;
802
803 switch(Action)
804 {
805 case NOTIFY_BW_ACTION:
806 DBGPRINT(RT_DEBUG_TRACE,("ACTION - HT Notify Channel bandwidth action----> \n"));
807 #ifdef CONFIG_STA_SUPPORT
808 if(pAd->StaActive.SupportedPhyInfo.bHtEnable == FALSE)
809 {
810 // Note, this is to patch DIR-1353 AP. When the AP set to Wep, it will use legacy mode. But AP still keeps
811 // sending BW_Notify Action frame, and cause us to linkup and linkdown.
812 // In legacy mode, don't need to parse HT action frame.
813 DBGPRINT(RT_DEBUG_TRACE,("ACTION -Ignore HT Notify Channel BW when link as legacy mode. BW = %d---> \n",
814 Elem->Msg[LENGTH_802_11+2] ));
815 break;
816 }
817 #endif // CONFIG_STA_SUPPORT //
818
819 if (Elem->Msg[LENGTH_802_11+2] == 0) // 7.4.8.2. if value is 1, keep the same as supported channel bandwidth.
820 pAd->MacTab.Content[Elem->Wcid].HTPhyMode.field.BW = 0;
821
822 break;
823
824 case SMPS_ACTION:
825 // 7.3.1.25
826 DBGPRINT(RT_DEBUG_TRACE,("ACTION - SMPS action----> \n"));
827 if (((Elem->Msg[LENGTH_802_11+2]&0x1) == 0))
828 {
829 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_ENABLE;
830 }
831 else if (((Elem->Msg[LENGTH_802_11+2]&0x2) == 0))
832 {
833 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_STATIC;
834 }
835 else
836 {
837 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_DYNAMIC;
838 }
839
840 DBGPRINT(RT_DEBUG_TRACE,("Aid(%d) MIMO PS = %d\n", Elem->Wcid, pAd->MacTab.Content[Elem->Wcid].MmpsMode));
841 // rt2860c : add something for smps change.
842 break;
843
844 case SETPCO_ACTION:
845 break;
846
847 case MIMO_CHA_MEASURE_ACTION:
848 break;
849
850 case HT_INFO_EXCHANGE:
851 {
852 HT_INFORMATION_OCTET *pHT_info;
853
854 pHT_info = (HT_INFORMATION_OCTET *) &Elem->Msg[LENGTH_802_11+2];
855 // 7.4.8.10
856 DBGPRINT(RT_DEBUG_TRACE,("ACTION - HT Information Exchange action----> \n"));
857 if (pHT_info->Request)
858 {
859 respond_ht_information_exchange_action(pAd, Elem);
860 }
861 }
862 break;
863 }
864 }
865
866
867 /*
868 ==========================================================================
869 Description:
870 Retry sending ADDBA Reqest.
871
872 IRQL = DISPATCH_LEVEL
873
874 Parametrs:
875 p8023Header: if this is already 802.3 format, p8023Header is NULL
876
877 Return : TRUE if put into rx reordering buffer, shouldn't indicaterxhere.
878 FALSE , then continue indicaterx at this moment.
879 ==========================================================================
880 */
881 VOID ORIBATimerTimeout(
882 IN PRTMP_ADAPTER pAd)
883 {
884 MAC_TABLE_ENTRY *pEntry;
885 INT i, total;
886 // FRAME_BAR FrameBar;
887 // ULONG FrameLen;
888 // NDIS_STATUS NStatus;
889 // PUCHAR pOutBuffer = NULL;
890 // USHORT Sequence;
891 UCHAR TID;
892
893 #ifdef RALINK_ATE
894 if (ATE_ON(pAd))
895 return;
896 #endif // RALINK_ATE //
897
898 total = pAd->MacTab.Size * NUM_OF_TID;
899
900 for (i = 1; ((i <MAX_LEN_OF_BA_ORI_TABLE) && (total > 0)) ; i++)
901 {
902 if (pAd->BATable.BAOriEntry[i].ORI_BA_Status == Originator_Done)
903 {
904 pEntry = &pAd->MacTab.Content[pAd->BATable.BAOriEntry[i].Wcid];
905 TID = pAd->BATable.BAOriEntry[i].TID;
906
907 ASSERT(pAd->BATable.BAOriEntry[i].Wcid < MAX_LEN_OF_MAC_TABLE);
908 }
909 total --;
910 }
911 }
912
913
914 VOID SendRefreshBAR(
915 IN PRTMP_ADAPTER pAd,
916 IN MAC_TABLE_ENTRY *pEntry)
917 {
918 FRAME_BAR FrameBar;
919 ULONG FrameLen;
920 NDIS_STATUS NStatus;
921 PUCHAR pOutBuffer = NULL;
922 USHORT Sequence;
923 UCHAR i, TID;
924 USHORT idx;
925 BA_ORI_ENTRY *pBAEntry;
926
927 for (i = 0; i <NUM_OF_TID; i++)
928 {
929 idx = pEntry->BAOriWcidArray[i];
930 if (idx == 0)
931 {
932 continue;
933 }
934 pBAEntry = &pAd->BATable.BAOriEntry[idx];
935
936 if (pBAEntry->ORI_BA_Status == Originator_Done)
937 {
938 TID = pBAEntry->TID;
939
940 ASSERT(pBAEntry->Wcid < MAX_LEN_OF_MAC_TABLE);
941
942 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
943 if(NStatus != NDIS_STATUS_SUCCESS)
944 {
945 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeADDBAAction() allocate memory failed \n"));
946 return;
947 }
948
949 Sequence = pEntry->TxSeq[TID];
950
951
952 #ifdef CONFIG_STA_SUPPORT
953 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
954 BarHeaderInit(pAd, &FrameBar, pEntry->Addr, pAd->CurrentAddress);
955 #endif // CONFIG_STA_SUPPORT //
956
957 FrameBar.StartingSeq.field.FragNum = 0; // make sure sequence not clear in DEL function.
958 FrameBar.StartingSeq.field.StartSeq = Sequence; // make sure sequence not clear in DEL funciton.
959 FrameBar.BarControl.TID = TID; // make sure sequence not clear in DEL funciton.
960
961 MakeOutgoingFrame(pOutBuffer, &FrameLen,
962 sizeof(FRAME_BAR), &FrameBar,
963 END_OF_ARGS);
964 //if (!(CLIENT_STATUS_TEST_FLAG(pEntry, fCLIENT_STATUS_RALINK_CHIPSET)))
965 if (1) // Now we always send BAR.
966 {
967 //MiniportMMRequestUnlock(pAd, 0, pOutBuffer, FrameLen);
968 MiniportMMRequest(pAd, 0, pOutBuffer, FrameLen);
969 }
970 MlmeFreeMemory(pAd, pOutBuffer);
971 }
972 }
973 }
974 #endif // DOT11_N_SUPPORT //
975
976 VOID ActHeaderInit(
977 IN PRTMP_ADAPTER pAd,
978 IN OUT PHEADER_802_11 pHdr80211,
979 IN PUCHAR Addr1,
980 IN PUCHAR Addr2,
981 IN PUCHAR Addr3)
982 {
983 NdisZeroMemory(pHdr80211, sizeof(HEADER_802_11));
984 pHdr80211->FC.Type = BTYPE_MGMT;
985 pHdr80211->FC.SubType = SUBTYPE_ACTION;
986
987 COPY_MAC_ADDR(pHdr80211->Addr1, Addr1);
988 COPY_MAC_ADDR(pHdr80211->Addr2, Addr2);
989 COPY_MAC_ADDR(pHdr80211->Addr3, Addr3);
990 }
991
992 VOID BarHeaderInit(
993 IN PRTMP_ADAPTER pAd,
994 IN OUT PFRAME_BAR pCntlBar,
995 IN PUCHAR pDA,
996 IN PUCHAR pSA)
997 {
998 // USHORT Duration;
999
1000 NdisZeroMemory(pCntlBar, sizeof(FRAME_BAR));
1001 pCntlBar->FC.Type = BTYPE_CNTL;
1002 pCntlBar->FC.SubType = SUBTYPE_BLOCK_ACK_REQ;
1003 pCntlBar->BarControl.MTID = 0;
1004 pCntlBar->BarControl.Compressed = 1;
1005 pCntlBar->BarControl.ACKPolicy = 0;
1006
1007
1008 pCntlBar->Duration = 16 + RTMPCalcDuration(pAd, RATE_1, sizeof(FRAME_BA));
1009
1010 COPY_MAC_ADDR(pCntlBar->Addr1, pDA);
1011 COPY_MAC_ADDR(pCntlBar->Addr2, pSA);
1012 }
1013
1014
1015 /*
1016 ==========================================================================
1017 Description:
1018 Insert Category and action code into the action frame.
1019
1020 Parametrs:
1021 1. frame buffer pointer.
1022 2. frame length.
1023 3. category code of the frame.
1024 4. action code of the frame.
1025
1026 Return : None.
1027 ==========================================================================
1028 */
1029 VOID InsertActField(
1030 IN PRTMP_ADAPTER pAd,
1031 OUT PUCHAR pFrameBuf,
1032 OUT PULONG pFrameLen,
1033 IN UINT8 Category,
1034 IN UINT8 ActCode)
1035 {
1036 ULONG TempLen;
1037
1038 MakeOutgoingFrame( pFrameBuf, &TempLen,
1039 1, &Category,
1040 1, &ActCode,
1041 END_OF_ARGS);
1042
1043 *pFrameLen = *pFrameLen + TempLen;
1044
1045 return;
1046 }
This page took 0.053083 seconds and 5 git commands to generate.