2 *************************************************************************
4 * 5F., No.36, Taiyuan St., Jhubei City,
8 * (c) Copyright 2002-2007, Ralink Technology, Inc.
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. *
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. *
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. *
25 *************************************************************************
31 Handle association related requests either from WSTA or from local MLME
35 -------- ---------- ----------------------------------------------
36 Jan Lee 2006 created for rt2860
39 #include "../rt_config.h"
43 static VOID ReservedAction(
45 IN MLME_QUEUE_ELEM *Elem);
48 ==========================================================================
50 association state machine init, including state transition and timer init
52 S - pointer to the association state machine
54 The state machine looks like the following
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 ==========================================================================
64 VOID ActionStateMachineInit(
67 OUT STATE_MACHINE_FUNC Trans[])
69 StateMachineInit(S, (STATE_MACHINE_FUNC *)Trans, MAX_ACT_STATE, MAX_ACT_MSG, (STATE_MACHINE_FUNC)Drop, ACT_IDLE, ACT_MACHINE_BASE);
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);
74 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_DLS_CATE, (STATE_MACHINE_FUNC)ReservedAction);
76 #ifdef DOT11_N_SUPPORT
77 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_BA_CATE, (STATE_MACHINE_FUNC)PeerBAAction);
78 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_HT_CATE, (STATE_MACHINE_FUNC)PeerHTAction);
79 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_ADD_BA_CATE, (STATE_MACHINE_FUNC)MlmeADDBAAction);
80 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_ORI_DELBA_CATE, (STATE_MACHINE_FUNC)MlmeDELBAAction);
81 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_REC_DELBA_CATE, (STATE_MACHINE_FUNC)MlmeDELBAAction);
82 #endif // DOT11_N_SUPPORT //
84 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_PUBLIC_CATE, (STATE_MACHINE_FUNC)PeerPublicAction);
85 StateMachineSetAction(S, ACT_IDLE, MT2_PEER_RM_CATE, (STATE_MACHINE_FUNC)PeerRMAction);
87 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_QOS_CATE, (STATE_MACHINE_FUNC)MlmeQOSAction);
88 StateMachineSetAction(S, ACT_IDLE, MT2_MLME_DLS_CATE, (STATE_MACHINE_FUNC)MlmeDLSAction);
89 StateMachineSetAction(S, ACT_IDLE, MT2_ACT_INVALID, (STATE_MACHINE_FUNC)MlmeInvalidAction);
92 #ifdef DOT11_N_SUPPORT
95 IN MLME_QUEUE_ELEM *Elem)
98 MLME_ADDBA_REQ_STRUCT *pInfo;
100 PUCHAR pOutBuffer = NULL;
103 FRAME_ADDBA_REQ Frame;
105 BA_ORI_ENTRY *pBAEntry = NULL;
107 pInfo = (MLME_ADDBA_REQ_STRUCT *)Elem->Msg;
108 NdisZeroMemory(&Frame, sizeof(FRAME_ADDBA_REQ));
110 if(MlmeAddBAReqSanity(pAd, Elem->Msg, Elem->MsgLen, Addr))
112 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
113 if(NStatus != NDIS_STATUS_SUCCESS)
115 DBGPRINT(RT_DEBUG_TRACE,("BA - MlmeADDBAAction() allocate memory failed \n"));
119 Idx = pAd->MacTab.Content[pInfo->Wcid].BAOriWcidArray[pInfo->TID];
122 MlmeFreeMemory(pAd, pOutBuffer);
123 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeADDBAAction() can't find BAOriEntry \n"));
128 pBAEntry =&pAd->BATable.BAOriEntry[Idx];
131 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
134 ActHeaderInit(pAd, &Frame.Hdr, pInfo->pAddr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
136 ActHeaderInit(pAd, &Frame.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pInfo->pAddr);
139 Frame.Category = CATEGORY_BA;
140 Frame.Action = ADDBA_REQ;
141 Frame.BaParm.AMSDUSupported = 0;
142 Frame.BaParm.BAPolicy = IMMED_BA;
143 Frame.BaParm.TID = pInfo->TID;
144 Frame.BaParm.BufSize = pInfo->BaBufSize;
145 Frame.Token = pInfo->Token;
146 Frame.TimeOutValue = pInfo->TimeOutValue;
147 Frame.BaStartSeq.field.FragNum = 0;
148 Frame.BaStartSeq.field.StartSeq = pAd->MacTab.Content[pInfo->Wcid].TxSeq[pInfo->TID];
150 *(USHORT *)(&Frame.BaParm) = cpu2le16(*(USHORT *)(&Frame.BaParm));
151 Frame.TimeOutValue = cpu2le16(Frame.TimeOutValue);
152 Frame.BaStartSeq.word = cpu2le16(Frame.BaStartSeq.word);
154 MakeOutgoingFrame(pOutBuffer, &FrameLen,
155 sizeof(FRAME_ADDBA_REQ), &Frame,
157 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
158 MlmeFreeMemory(pAd, pOutBuffer);
160 DBGPRINT(RT_DEBUG_TRACE, ("BA - Send ADDBA request. StartSeq = %x, FrameLen = %ld. BufSize = %d\n", Frame.BaStartSeq.field.StartSeq, FrameLen, Frame.BaParm.BufSize));
165 ==========================================================================
167 send DELBA and delete BaEntry if any
169 Elem - MLME message MLME_DELBA_REQ_STRUCT
171 IRQL = DISPATCH_LEVEL
173 ==========================================================================
175 VOID MlmeDELBAAction(
176 IN PRTMP_ADAPTER pAd,
177 IN MLME_QUEUE_ELEM *Elem)
179 MLME_DELBA_REQ_STRUCT *pInfo;
180 PUCHAR pOutBuffer = NULL;
181 PUCHAR pOutBuffer2 = NULL;
184 FRAME_DELBA_REQ Frame;
188 pInfo = (MLME_DELBA_REQ_STRUCT *)Elem->Msg;
189 // must send back DELBA
190 NdisZeroMemory(&Frame, sizeof(FRAME_DELBA_REQ));
191 DBGPRINT(RT_DEBUG_TRACE, ("==> MlmeDELBAAction(), Initiator(%d) \n", pInfo->Initiator));
193 if(MlmeDelBAReqSanity(pAd, Elem->Msg, Elem->MsgLen))
195 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
196 if(NStatus != NDIS_STATUS_SUCCESS)
198 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeDELBAAction() allocate memory failed 1. \n"));
202 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer2); //Get an unused nonpaged memory
203 if(NStatus != NDIS_STATUS_SUCCESS)
205 MlmeFreeMemory(pAd, pOutBuffer);
206 DBGPRINT(RT_DEBUG_ERROR, ("BA - MlmeDELBAAction() allocate memory failed 2. \n"));
210 // SEND BAR (Send BAR to refresh peer reordering buffer.)
211 Idx = pAd->MacTab.Content[pInfo->Wcid].BAOriWcidArray[pInfo->TID];
213 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
214 BarHeaderInit(pAd, &FrameBar, pAd->MacTab.Content[pInfo->Wcid].Addr, pAd->CurrentAddress);
216 FrameBar.StartingSeq.field.FragNum = 0; // make sure sequence not clear in DEL funciton.
217 FrameBar.StartingSeq.field.StartSeq = pAd->MacTab.Content[pInfo->Wcid].TxSeq[pInfo->TID]; // make sure sequence not clear in DEL funciton.
218 FrameBar.BarControl.TID = pInfo->TID; // make sure sequence not clear in DEL funciton.
219 FrameBar.BarControl.ACKPolicy = IMMED_BA; // make sure sequence not clear in DEL funciton.
220 FrameBar.BarControl.Compressed = 1; // make sure sequence not clear in DEL funciton.
221 FrameBar.BarControl.MTID = 0; // make sure sequence not clear in DEL funciton.
223 MakeOutgoingFrame(pOutBuffer2, &FrameLen,
224 sizeof(FRAME_BAR), &FrameBar,
226 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer2, FrameLen);
227 MlmeFreeMemory(pAd, pOutBuffer2);
228 DBGPRINT(RT_DEBUG_TRACE,("BA - MlmeDELBAAction() . Send BAR to refresh peer reordering buffer \n"));
233 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
236 ActHeaderInit(pAd, &Frame.Hdr, pAd->MacTab.Content[pInfo->Wcid].Addr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
238 ActHeaderInit(pAd, &Frame.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pAd->MacTab.Content[pInfo->Wcid].Addr);
241 Frame.Category = CATEGORY_BA;
242 Frame.Action = DELBA;
243 Frame.DelbaParm.Initiator = pInfo->Initiator;
244 Frame.DelbaParm.TID = pInfo->TID;
245 Frame.ReasonCode = 39; // Time Out
246 *(USHORT *)(&Frame.DelbaParm) = cpu2le16(*(USHORT *)(&Frame.DelbaParm));
247 Frame.ReasonCode = cpu2le16(Frame.ReasonCode);
249 MakeOutgoingFrame(pOutBuffer, &FrameLen,
250 sizeof(FRAME_DELBA_REQ), &Frame,
252 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
253 MlmeFreeMemory(pAd, pOutBuffer);
254 DBGPRINT(RT_DEBUG_TRACE, ("BA - MlmeDELBAAction() . 3 DELBA sent. Initiator(%d)\n", pInfo->Initiator));
257 #endif // DOT11_N_SUPPORT //
260 IN PRTMP_ADAPTER pAd,
261 IN MLME_QUEUE_ELEM *Elem)
266 IN PRTMP_ADAPTER pAd,
267 IN MLME_QUEUE_ELEM *Elem)
271 VOID MlmeInvalidAction(
272 IN PRTMP_ADAPTER pAd,
273 IN MLME_QUEUE_ELEM *Elem)
275 //PUCHAR pOutBuffer = NULL;
276 //Return the receiving frame except the MSB of category filed set to 1. 7.3.1.11
280 IN PRTMP_ADAPTER pAd,
281 IN MLME_QUEUE_ELEM *Elem)
285 #ifdef DOT11_N_SUPPORT
287 IN PRTMP_ADAPTER pAd,
288 IN MLME_QUEUE_ELEM *Elem)
290 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
295 PeerAddBAReqAction(pAd,Elem);
298 PeerAddBARspAction(pAd,Elem);
301 PeerDelBAAction(pAd,Elem);
305 #endif // DOT11_N_SUPPORT //
307 VOID PeerPublicAction(
308 IN PRTMP_ADAPTER pAd,
309 IN MLME_QUEUE_ELEM *Elem)
311 if (Elem->Wcid >= MAX_LEN_OF_MAC_TABLE)
316 static VOID ReservedAction(
317 IN PRTMP_ADAPTER pAd,
318 IN MLME_QUEUE_ELEM *Elem)
322 if (Elem->MsgLen <= LENGTH_802_11)
327 Category = Elem->Msg[LENGTH_802_11];
328 DBGPRINT(RT_DEBUG_TRACE,("Rcv reserved category(%d) Action Frame\n", Category));
329 hex_dump("Reserved Action Frame", &Elem->Msg[0], Elem->MsgLen);
333 IN PRTMP_ADAPTER pAd,
334 IN MLME_QUEUE_ELEM *Elem)
340 #ifdef DOT11_N_SUPPORT
341 static VOID respond_ht_information_exchange_action(
342 IN PRTMP_ADAPTER pAd,
343 IN MLME_QUEUE_ELEM *Elem)
345 PUCHAR pOutBuffer = NULL;
348 FRAME_HT_INFO HTINFOframe, *pFrame;
352 // 2. Always send back ADDBA Response
353 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
355 if (NStatus != NDIS_STATUS_SUCCESS)
357 DBGPRINT(RT_DEBUG_TRACE,("ACTION - respond_ht_information_exchange_action() allocate memory failed \n"));
362 pFrame = (FRAME_HT_INFO *) &Elem->Msg[0];
363 pAddr = pFrame->Hdr.Addr2;
365 NdisZeroMemory(&HTINFOframe, sizeof(FRAME_HT_INFO));
366 // 2-1. Prepare ADDBA Response frame.
367 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
370 ActHeaderInit(pAd, &HTINFOframe.Hdr, pAddr, pAd->CurrentAddress, pAd->CommonCfg.Bssid);
372 ActHeaderInit(pAd, &HTINFOframe.Hdr, pAd->CommonCfg.Bssid, pAd->CurrentAddress, pAddr);
375 HTINFOframe.Category = CATEGORY_HT;
376 HTINFOframe.Action = HT_INFO_EXCHANGE;
377 HTINFOframe.HT_Info.Request = 0;
378 HTINFOframe.HT_Info.Forty_MHz_Intolerant = pAd->CommonCfg.HtCapability.HtCapInfo.Forty_Mhz_Intolerant;
379 HTINFOframe.HT_Info.STA_Channel_Width = pAd->CommonCfg.AddHTInfo.AddHtInfo.RecomWidth;
381 MakeOutgoingFrame(pOutBuffer, &FrameLen,
382 sizeof(FRAME_HT_INFO), &HTINFOframe,
385 MiniportMMRequest(pAd, QID_AC_BE, pOutBuffer, FrameLen);
386 MlmeFreeMemory(pAd, pOutBuffer);
390 IN PRTMP_ADAPTER pAd,
391 IN MLME_QUEUE_ELEM *Elem)
393 UCHAR Action = Elem->Msg[LENGTH_802_11+1];
395 if (Elem->Wcid >= MAX_LEN_OF_MAC_TABLE)
400 case NOTIFY_BW_ACTION:
401 DBGPRINT(RT_DEBUG_TRACE,("ACTION - HT Notify Channel bandwidth action----> \n"));
403 if(pAd->StaActive.SupportedPhyInfo.bHtEnable == FALSE)
405 // Note, this is to patch DIR-1353 AP. When the AP set to Wep, it will use legacy mode. But AP still keeps
406 // sending BW_Notify Action frame, and cause us to linkup and linkdown.
407 // In legacy mode, don't need to parse HT action frame.
408 DBGPRINT(RT_DEBUG_TRACE,("ACTION -Ignore HT Notify Channel BW when link as legacy mode. BW = %d---> \n",
409 Elem->Msg[LENGTH_802_11+2] ));
413 if (Elem->Msg[LENGTH_802_11+2] == 0) // 7.4.8.2. if value is 1, keep the same as supported channel bandwidth.
414 pAd->MacTab.Content[Elem->Wcid].HTPhyMode.field.BW = 0;
419 DBGPRINT(RT_DEBUG_TRACE,("ACTION - SMPS action----> \n"));
420 if (((Elem->Msg[LENGTH_802_11+2]&0x1) == 0))
422 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_ENABLE;
424 else if (((Elem->Msg[LENGTH_802_11+2]&0x2) == 0))
426 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_STATIC;
430 pAd->MacTab.Content[Elem->Wcid].MmpsMode = MMPS_DYNAMIC;
433 DBGPRINT(RT_DEBUG_TRACE,("Aid(%d) MIMO PS = %d\n", Elem->Wcid, pAd->MacTab.Content[Elem->Wcid].MmpsMode));
434 // rt2860c : add something for smps change.
439 case MIMO_CHA_MEASURE_ACTION:
441 case HT_INFO_EXCHANGE:
443 HT_INFORMATION_OCTET *pHT_info;
445 pHT_info = (HT_INFORMATION_OCTET *) &Elem->Msg[LENGTH_802_11+2];
447 DBGPRINT(RT_DEBUG_TRACE,("ACTION - HT Information Exchange action----> \n"));
448 if (pHT_info->Request)
450 respond_ht_information_exchange_action(pAd, Elem);
459 ==========================================================================
461 Retry sending ADDBA Reqest.
463 IRQL = DISPATCH_LEVEL
466 p8023Header: if this is already 802.3 format, p8023Header is NULL
468 Return : TRUE if put into rx reordering buffer, shouldn't indicaterxhere.
469 FALSE , then continue indicaterx at this moment.
470 ==========================================================================
472 VOID ORIBATimerTimeout(
473 IN PRTMP_ADAPTER pAd)
475 MAC_TABLE_ENTRY *pEntry;
479 total = pAd->MacTab.Size * NUM_OF_TID;
481 for (i = 1; ((i <MAX_LEN_OF_BA_ORI_TABLE) && (total > 0)) ; i++)
483 if (pAd->BATable.BAOriEntry[i].ORI_BA_Status == Originator_Done)
485 pEntry = &pAd->MacTab.Content[pAd->BATable.BAOriEntry[i].Wcid];
486 TID = pAd->BATable.BAOriEntry[i].TID;
488 ASSERT(pAd->BATable.BAOriEntry[i].Wcid < MAX_LEN_OF_MAC_TABLE);
496 IN PRTMP_ADAPTER pAd,
497 IN MAC_TABLE_ENTRY *pEntry)
502 PUCHAR pOutBuffer = NULL;
506 BA_ORI_ENTRY *pBAEntry;
508 for (i = 0; i <NUM_OF_TID; i++)
510 idx = pEntry->BAOriWcidArray[i];
515 pBAEntry = &pAd->BATable.BAOriEntry[idx];
517 if (pBAEntry->ORI_BA_Status == Originator_Done)
521 ASSERT(pBAEntry->Wcid < MAX_LEN_OF_MAC_TABLE);
523 NStatus = MlmeAllocateMemory(pAd, &pOutBuffer); //Get an unused nonpaged memory
524 if(NStatus != NDIS_STATUS_SUCCESS)
526 DBGPRINT(RT_DEBUG_ERROR,("BA - MlmeADDBAAction() allocate memory failed \n"));
530 Sequence = pEntry->TxSeq[TID];
532 IF_DEV_CONFIG_OPMODE_ON_STA(pAd)
533 BarHeaderInit(pAd, &FrameBar, pEntry->Addr, pAd->CurrentAddress);
535 FrameBar.StartingSeq.field.FragNum = 0; // make sure sequence not clear in DEL function.
536 FrameBar.StartingSeq.field.StartSeq = Sequence; // make sure sequence not clear in DEL funciton.
537 FrameBar.BarControl.TID = TID; // make sure sequence not clear in DEL funciton.
539 MakeOutgoingFrame(pOutBuffer, &FrameLen,
540 sizeof(FRAME_BAR), &FrameBar,
542 if (1) // Now we always send BAR.
544 MiniportMMRequest(pAd, 0, pOutBuffer, FrameLen);
546 MlmeFreeMemory(pAd, pOutBuffer);
550 #endif // DOT11_N_SUPPORT //
553 IN PRTMP_ADAPTER pAd,
554 IN OUT PHEADER_802_11 pHdr80211,
559 NdisZeroMemory(pHdr80211, sizeof(HEADER_802_11));
560 pHdr80211->FC.Type = BTYPE_MGMT;
561 pHdr80211->FC.SubType = SUBTYPE_ACTION;
563 COPY_MAC_ADDR(pHdr80211->Addr1, Addr1);
564 COPY_MAC_ADDR(pHdr80211->Addr2, Addr2);
565 COPY_MAC_ADDR(pHdr80211->Addr3, Addr3);
569 IN PRTMP_ADAPTER pAd,
570 IN OUT PFRAME_BAR pCntlBar,
574 NdisZeroMemory(pCntlBar, sizeof(FRAME_BAR));
575 pCntlBar->FC.Type = BTYPE_CNTL;
576 pCntlBar->FC.SubType = SUBTYPE_BLOCK_ACK_REQ;
577 pCntlBar->BarControl.MTID = 0;
578 pCntlBar->BarControl.Compressed = 1;
579 pCntlBar->BarControl.ACKPolicy = 0;
582 pCntlBar->Duration = 16 + RTMPCalcDuration(pAd, RATE_1, sizeof(FRAME_BA));
584 COPY_MAC_ADDR(pCntlBar->Addr1, pDA);
585 COPY_MAC_ADDR(pCntlBar->Addr2, pSA);
590 ==========================================================================
592 Insert Category and action code into the action frame.
595 1. frame buffer pointer.
597 3. category code of the frame.
598 4. action code of the frame.
601 ==========================================================================
604 IN PRTMP_ADAPTER pAd,
605 OUT PUCHAR pFrameBuf,
606 OUT PULONG pFrameLen,
612 MakeOutgoingFrame( pFrameBuf, &TempLen,
617 *pFrameLen = *pFrameLen + TempLen;