| /* |
| * Copyright (c) 2012-2017 The Linux Foundation. All rights reserved. |
| * |
| * Previously licensed under the ISC license by Qualcomm Atheros, Inc. |
| * |
| * |
| * Permission to use, copy, modify, and/or distribute this software for |
| * any purpose with or without fee is hereby granted, provided that the |
| * above copyright notice and this permission notice appear in all |
| * copies. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL |
| * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE |
| * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL |
| * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR |
| * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER |
| * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR |
| * PERFORMANCE OF THIS SOFTWARE. |
| */ |
| |
| /* |
| * This file was originally distributed by Qualcomm Atheros, Inc. |
| * under proprietary terms before Copyright ownership was assigned |
| * to the Linux Foundation. |
| */ |
| |
| |
| /* |
| * This file limIbssPeerMgmt.cc contains the utility functions |
| * LIM uses to maintain peers in IBSS. |
| * Author: Chandra Modumudi |
| * Date: 03/12/04 |
| * History:- |
| * Date Modified by Modification Information |
| * -------------------------------------------------------------------- |
| */ |
| #include "palTypes.h" |
| #include "aniGlobal.h" |
| #include "sirCommon.h" |
| #include "wni_cfg.h" |
| #include "limUtils.h" |
| #include "limAssocUtils.h" |
| #include "limStaHashApi.h" |
| #include "schApi.h" // schSetFixedBeaconFields for IBSS coalesce |
| #include "limSecurityUtils.h" |
| #include "limSendMessages.h" |
| #ifdef WLAN_FEATURE_VOWIFI_11R |
| #include "limFTDefs.h" |
| #endif |
| #include "limSession.h" |
| #include "limIbssPeerMgmt.h" |
| |
| |
| /** |
| * ibss_peer_find |
| * |
| *FUNCTION: |
| * This function is called while adding a context at |
| * DPH for a peer in IBSS. |
| * If peer is found in the list, capabilities from the |
| * returned BSS description are used at DPH node. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param macAddr - MAC address of the peer |
| * |
| * @return Pointer to peer node if found, else NULL |
| */ |
| |
| static tLimIbssPeerNode * |
| ibss_peer_find( |
| tpAniSirGlobal pMac, |
| tSirMacAddr macAddr) |
| { |
| tLimIbssPeerNode *pTempNode = pMac->lim.gLimIbssPeerList; |
| |
| while (pTempNode != NULL) |
| { |
| if (vos_mem_compare((tANI_U8 *) macAddr, |
| (tANI_U8 *) &pTempNode->peerMacAddr, |
| sizeof(tSirMacAddr))) |
| break; |
| pTempNode = pTempNode->next; |
| } |
| return pTempNode; |
| } /*** end ibss_peer_find() ***/ |
| |
| /** |
| * ibss_peer_add |
| * |
| *FUNCTION: |
| * This is called on a STA in IBSS upon receiving Beacon/ |
| * Probe Response from a peer. |
| * |
| *LOGIC: |
| * Node is always added to the front of the list |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @param pPeerNode - Pointer to peer node to be added to the list. |
| * |
| * @return None |
| */ |
| |
| static tSirRetStatus |
| ibss_peer_add(tpAniSirGlobal pMac, tLimIbssPeerNode *pPeerNode) |
| { |
| #ifdef ANI_SIR_IBSS_PEER_CACHING |
| tANI_U32 numIbssPeers = (2 * pMac->lim.maxStation); |
| |
| if (pMac->lim.gLimNumIbssPeers >= numIbssPeers) |
| { |
| /** |
| * Reached max number of peers to be maintained. |
| * Delete last entry & add new entry at the beginning. |
| */ |
| tLimIbssPeerNode *pTemp, *pPrev; |
| pTemp = pPrev = pMac->lim.gLimIbssPeerList; |
| while (pTemp->next != NULL) |
| { |
| pPrev = pTemp; |
| pTemp = pTemp->next; |
| } |
| if(pTemp->beacon) |
| { |
| vos_mem_free(pTemp->beacon); |
| } |
| |
| vos_mem_free(pTemp); |
| pPrev->next = NULL; |
| } |
| else |
| #endif |
| pMac->lim.gLimNumIbssPeers++; |
| |
| pPeerNode->next = pMac->lim.gLimIbssPeerList; |
| pMac->lim.gLimIbssPeerList = pPeerNode; |
| |
| return eSIR_SUCCESS; |
| |
| } /*** end limAddIbssPeerToList() ***/ |
| |
| /** |
| * ibss_peer_collect |
| * |
| *FUNCTION: |
| * This is called to collect IBSS peer information |
| * from received Beacon/Probe Response frame from it. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @param pBeacon - Parsed Beacon Frame structure |
| * @param pBD - Pointer to received BD |
| * @param pPeer - Pointer to IBSS peer node |
| * |
| * @return None |
| */ |
| |
| static void |
| ibss_peer_collect( |
| tpAniSirGlobal pMac, |
| tpSchBeaconStruct pBeacon, |
| tpSirMacMgmtHdr pHdr, |
| tLimIbssPeerNode *pPeer, |
| tpPESession psessionEntry) |
| { |
| vos_mem_copy(pPeer->peerMacAddr, pHdr->sa, sizeof(tSirMacAddr)); |
| |
| pPeer->capabilityInfo = pBeacon->capabilityInfo; |
| pPeer->extendedRatesPresent = pBeacon->extendedRatesPresent; |
| pPeer->edcaPresent = pBeacon->edcaPresent; |
| pPeer->wmeEdcaPresent = pBeacon->wmeEdcaPresent; |
| pPeer->wmeInfoPresent = pBeacon->wmeInfoPresent; |
| |
| if (pBeacon->IBSSParams.present) |
| { |
| pPeer->atimIePresent = pBeacon->IBSSParams.present; |
| pPeer->peerAtimWindowLength = pBeacon->IBSSParams.atim; |
| } |
| |
| if(IS_DOT11_MODE_HT(psessionEntry->dot11mode) && |
| (pBeacon->HTCaps.present)) |
| { |
| pPeer->htCapable = pBeacon->HTCaps.present; |
| vos_mem_copy((tANI_U8 *)pPeer->supportedMCSSet, |
| (tANI_U8 *)pBeacon->HTCaps.supportedMCSSet, |
| sizeof(pPeer->supportedMCSSet)); |
| pPeer->htGreenfield = (tANI_U8)pBeacon->HTCaps.greenField; |
| pPeer->htSupportedChannelWidthSet = ( tANI_U8 ) pBeacon->HTCaps.supportedChannelWidthSet; |
| pPeer->htMIMOPSState = (tSirMacHTMIMOPowerSaveState)pBeacon->HTCaps.mimoPowerSave; |
| pPeer->htMaxAmsduLength = ( tANI_U8 ) pBeacon->HTCaps.maximalAMSDUsize; |
| pPeer->htAMpduDensity = pBeacon->HTCaps.mpduDensity; |
| pPeer->htDsssCckRate40MHzSupport = (tANI_U8)pBeacon->HTCaps.dsssCckMode40MHz; |
| pPeer->htShortGI20Mhz = (tANI_U8)pBeacon->HTCaps.shortGI20MHz; |
| pPeer->htShortGI40Mhz = (tANI_U8)pBeacon->HTCaps.shortGI40MHz; |
| pPeer->htMaxRxAMpduFactor = pBeacon->HTCaps.maxRxAMPDUFactor; |
| pPeer->htSecondaryChannelOffset = pBeacon->HTInfo.secondaryChannelOffset; |
| pPeer->htLdpcCapable = (tANI_U8)pBeacon->HTCaps.advCodingCap; |
| } |
| |
| /* Collect peer VHT capabilities based on the received beacon from the peer */ |
| #ifdef WLAN_FEATURE_11AC |
| if ( pBeacon->VHTCaps.present ) |
| { |
| pPeer->vhtSupportedChannelWidthSet = pBeacon->VHTOperation.chanWidth; |
| pPeer->vhtCapable = pBeacon->VHTCaps.present; |
| |
| // Collect VHT capabilities from beacon |
| vos_mem_copy((tANI_U8 *) &pPeer->VHTCaps, |
| (tANI_U8 *) &pBeacon->VHTCaps, |
| sizeof(tDot11fIEVHTCaps)); |
| } |
| #endif |
| pPeer->erpIePresent = pBeacon->erpPresent; |
| |
| vos_mem_copy((tANI_U8 *) &pPeer->supportedRates, |
| (tANI_U8 *) &pBeacon->supportedRates, |
| pBeacon->supportedRates.numRates + 1); |
| if (pPeer->extendedRatesPresent) |
| vos_mem_copy((tANI_U8 *) &pPeer->extendedRates, |
| (tANI_U8 *) &pBeacon->extendedRates, |
| pBeacon->extendedRates.numRates + 1); |
| else |
| pPeer->extendedRates.numRates = 0; |
| |
| pPeer->next = NULL; |
| } /*** end ibss_peer_collect() ***/ |
| |
| // handle change in peer qos/wme capabilities |
| static void |
| ibss_sta_caps_update( |
| tpAniSirGlobal pMac, |
| tLimIbssPeerNode *pPeerNode, |
| tpPESession psessionEntry) |
| { |
| tANI_U16 peerIdx; |
| tpDphHashNode pStaDs; |
| |
| pPeerNode->beaconHBCount++; //Update beacon count. |
| |
| // if the peer node exists, update its qos capabilities |
| if ((pStaDs = dphLookupHashEntry(pMac, pPeerNode->peerMacAddr, &peerIdx, &psessionEntry->dph.dphHashTable)) == NULL) |
| return; |
| |
| |
| //Update HT Capabilities |
| if(IS_DOT11_MODE_HT(psessionEntry->dot11mode)) |
| { |
| pStaDs->mlmStaContext.htCapability = pPeerNode->htCapable; |
| if (pPeerNode->htCapable) |
| { |
| pStaDs->htGreenfield = pPeerNode->htGreenfield; |
| pStaDs->htSupportedChannelWidthSet = |
| pPeerNode->htSupportedChannelWidthSet; |
| pStaDs->htSecondaryChannelOffset = |
| pPeerNode->htSecondaryChannelOffset; |
| pStaDs->htMIMOPSState = pPeerNode->htMIMOPSState; |
| pStaDs->htMaxAmsduLength = pPeerNode->htMaxAmsduLength; |
| pStaDs->htAMpduDensity = pPeerNode->htAMpduDensity; |
| pStaDs->htDsssCckRate40MHzSupport = pPeerNode->htDsssCckRate40MHzSupport; |
| pStaDs->htShortGI20Mhz = pPeerNode->htShortGI20Mhz; |
| pStaDs->htShortGI40Mhz = pPeerNode->htShortGI40Mhz; |
| pStaDs->htMaxRxAMpduFactor = pPeerNode->htMaxRxAMpduFactor; |
| // In the future, may need to check for "delayedBA" |
| // For now, it is IMMEDIATE BA only on ALL TID's |
| pStaDs->baPolicyFlag = 0xFF; |
| pStaDs->htLdpcCapable = pPeerNode->htLdpcCapable; |
| } |
| } |
| #ifdef WLAN_FEATURE_11AC |
| if ( IS_DOT11_MODE_VHT(psessionEntry->dot11mode) ) |
| { |
| pStaDs->mlmStaContext.vhtCapability = pPeerNode->vhtCapable; |
| if ( pPeerNode->vhtCapable ) |
| { |
| pStaDs->vhtSupportedChannelWidthSet = pPeerNode->vhtSupportedChannelWidthSet; |
| |
| // If in 11AC mode and if session requires 11AC mode, consider peer's |
| // max AMPDU length factor |
| pStaDs->htMaxRxAMpduFactor = pPeerNode->VHTCaps.maxAMPDULenExp; |
| pStaDs->vhtLdpcCapable = (tANI_U8)pPeerNode->VHTCaps.ldpcCodingCap; |
| } |
| } |
| #endif |
| |
| // peer is 11e capable but is not 11e enabled yet |
| // some STA's when joining Airgo IBSS, assert qos capability even when |
| // they don't suport qos. however, they do not include the edca parameter |
| // set. so let's check for edcaParam in addition to the qos capability |
| if (pPeerNode->capabilityInfo.qos && (psessionEntry->limQosEnabled) && pPeerNode->edcaPresent) |
| { |
| pStaDs->qosMode = 1; |
| pStaDs->wmeEnabled = 0; |
| if (! pStaDs->lleEnabled) |
| { |
| pStaDs->lleEnabled = 1; |
| //dphSetACM(pMac, pStaDs); |
| } |
| return; |
| } |
| // peer is not 11e capable now but was 11e enabled earlier |
| else if (pStaDs->lleEnabled) |
| { |
| pStaDs->qosMode = 0; |
| pStaDs->lleEnabled = 0; |
| } |
| |
| // peer is wme capable but is not wme enabled yet |
| if (pPeerNode->wmeInfoPresent && psessionEntry->limWmeEnabled) |
| { |
| pStaDs->qosMode = 1; |
| pStaDs->lleEnabled = 0; |
| if (! pStaDs->wmeEnabled) |
| { |
| pStaDs->wmeEnabled = 1; |
| } |
| return; |
| } |
| /* When the peer device supports EDCA parameters, then we were not |
| considering. Added this code when we saw that one of the Peer Device |
| was advertising WMM param where we were not honouring that. CR# 210756 |
| */ |
| if (pPeerNode->wmeEdcaPresent && psessionEntry->limWmeEnabled) { |
| pStaDs->qosMode = 1; |
| pStaDs->lleEnabled = 0; |
| if (! pStaDs->wmeEnabled) { |
| pStaDs->wmeEnabled = 1; |
| } |
| return; |
| } |
| |
| // peer is not wme capable now but was wme enabled earlier |
| else if (pStaDs->wmeEnabled) |
| { |
| pStaDs->qosMode = 0; |
| pStaDs->wmeEnabled = 0; |
| } |
| |
| } |
| |
| static void |
| ibss_sta_rates_update( |
| tpAniSirGlobal pMac, |
| tpDphHashNode pStaDs, |
| tLimIbssPeerNode *pPeer, |
| tpPESession psessionEntry) |
| { |
| #ifdef WLAN_FEATURE_11AC |
| limPopulateMatchingRateSet(pMac, pStaDs, &pPeer->supportedRates, |
| &pPeer->extendedRates, pPeer->supportedMCSSet, |
| psessionEntry, &pPeer->VHTCaps); |
| #else |
| // Populate supported rateset |
| limPopulateMatchingRateSet(pMac, pStaDs, &pPeer->supportedRates, |
| &pPeer->extendedRates, pPeer->supportedMCSSet, |
| psessionEntry); |
| #endif |
| |
| pStaDs->mlmStaContext.capabilityInfo = pPeer->capabilityInfo; |
| } /*** end ibss_sta_info_update() ***/ |
| |
| /** |
| * ibss_sta_info_update |
| * |
| *FUNCTION: |
| * This is called to program SW context for peer in IBSS. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @param pStaDs - Pointer to DPH node |
| * @param pPeer - Pointer to IBSS peer node |
| * |
| * @return None |
| */ |
| |
| static void |
| ibss_sta_info_update( |
| tpAniSirGlobal pMac, |
| tpDphHashNode pStaDs, |
| tLimIbssPeerNode *pPeer, |
| tpPESession psessionEntry) |
| { |
| pStaDs->staType = STA_ENTRY_PEER; |
| ibss_sta_caps_update(pMac, pPeer,psessionEntry); |
| ibss_sta_rates_update(pMac, pStaDs, pPeer,psessionEntry); |
| } /*** end ibss_sta_info_update() ***/ |
| |
| static void |
| ibss_coalesce_free( |
| tpAniSirGlobal pMac) |
| { |
| if (pMac->lim.ibssInfo.pHdr != NULL) |
| vos_mem_free(pMac->lim.ibssInfo.pHdr); |
| if (pMac->lim.ibssInfo.pBeacon != NULL) |
| vos_mem_free(pMac->lim.ibssInfo.pBeacon); |
| |
| pMac->lim.ibssInfo.pHdr = NULL; |
| pMac->lim.ibssInfo.pBeacon = NULL; |
| } |
| |
| /* |
| * save the beacon params for use when adding the bss |
| */ |
| static void |
| ibss_coalesce_save( |
| tpAniSirGlobal pMac, |
| tpSirMacMgmtHdr pHdr, |
| tpSchBeaconStruct pBeacon) |
| { |
| // get rid of any saved info |
| ibss_coalesce_free(pMac); |
| |
| pMac->lim.ibssInfo.pHdr = vos_mem_malloc(sizeof(*pHdr)); |
| if (NULL == pMac->lim.ibssInfo.pHdr) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("ibbs-save: Failed malloc pHdr"));) |
| return; |
| } |
| pMac->lim.ibssInfo.pBeacon = vos_mem_malloc(sizeof(*pBeacon)); |
| if (NULL == pMac->lim.ibssInfo.pBeacon) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("ibbs-save: Failed malloc pBeacon"));) |
| ibss_coalesce_free(pMac); |
| return; |
| } |
| |
| vos_mem_copy(pMac->lim.ibssInfo.pHdr, pHdr, sizeof(*pHdr)); |
| vos_mem_copy(pMac->lim.ibssInfo.pBeacon, pBeacon, sizeof(*pBeacon)); |
| } |
| |
| /* |
| * tries to add a new entry to dph hash node |
| * if necessary, an existing entry is eliminated |
| */ |
| static tSirRetStatus |
| ibss_dph_entry_add( |
| tpAniSirGlobal pMac, |
| tSirMacAddr peerAddr, |
| tpDphHashNode *ppSta, |
| tpPESession psessionEntry) |
| { |
| tANI_U16 peerIdx; |
| tpDphHashNode pStaDs; |
| |
| *ppSta = NULL; |
| |
| pStaDs = dphLookupHashEntry(pMac, peerAddr, &peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs != NULL) |
| { |
| /* Trying to add context for already existing STA in IBSS */ |
| PELOGE(limLog(pMac, LOGE, FL("STA exists already "));) |
| limPrintMacAddr(pMac, peerAddr, LOGE); |
| return eSIR_FAILURE; |
| } |
| |
| /** |
| * Assign an AID, delete context existing with that |
| * AID and then add an entry to hash table maintained |
| * by DPH module. |
| */ |
| peerIdx = limAssignPeerIdx(pMac, psessionEntry); |
| |
| pStaDs = dphGetHashEntry(pMac, peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs) |
| { |
| (void) limDelSta(pMac, pStaDs, false /*asynchronous*/,psessionEntry); |
| limDeleteDphHashEntry(pMac, pStaDs->staAddr, peerIdx,psessionEntry); |
| } |
| |
| pStaDs = dphAddHashEntry(pMac, peerAddr, peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs == NULL) |
| { |
| // Could not add hash table entry |
| PELOGE(limLog(pMac, LOGE, FL("could not add hash entry at DPH for peerIdx/aid=%d MACaddr:"), peerIdx);) |
| limPrintMacAddr(pMac, peerAddr, LOGE); |
| return eSIR_FAILURE; |
| } |
| |
| *ppSta = pStaDs; |
| return eSIR_SUCCESS; |
| } |
| |
| // send a status change notification |
| static void |
| ibss_status_chg_notify( |
| tpAniSirGlobal pMac, |
| tSirMacAddr peerAddr, |
| tANI_U16 staIndex, |
| tANI_U8 ucastSig, |
| tANI_U8 bcastSig, |
| tANI_U16 status, |
| tANI_U8 sessionId) |
| { |
| |
| tLimIbssPeerNode *peerNode; |
| tANI_U8 *beacon = NULL; |
| tANI_U16 bcnLen = 0; |
| |
| |
| peerNode = ibss_peer_find(pMac,peerAddr); |
| if(peerNode != NULL) |
| { |
| if(peerNode->beacon == NULL) peerNode->beaconLen = 0; |
| beacon = peerNode->beacon; |
| bcnLen = peerNode->beaconLen; |
| peerNode->beacon = NULL; |
| peerNode->beaconLen = 0; |
| } |
| |
| limSendSmeIBSSPeerInd(pMac,peerAddr, staIndex, ucastSig, bcastSig, |
| beacon, bcnLen, status, sessionId); |
| |
| if(beacon != NULL) |
| { |
| vos_mem_free(beacon); |
| } |
| } |
| |
| |
| static void |
| ibss_bss_add( |
| tpAniSirGlobal pMac, |
| tpPESession psessionEntry) |
| { |
| tLimMlmStartReq mlmStartReq; |
| tANI_U32 cfg; |
| tpSirMacMgmtHdr pHdr = (tpSirMacMgmtHdr) pMac->lim.ibssInfo.pHdr; |
| tpSchBeaconStruct pBeacon = (tpSchBeaconStruct) pMac->lim.ibssInfo.pBeacon; |
| tANI_U8 numExtRates = 0; |
| |
| if ((pHdr == NULL) || (pBeacon == NULL)) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("Unable to add BSS (no cached BSS info)"));) |
| return; |
| } |
| |
| vos_mem_copy(psessionEntry->bssId, pHdr->bssId, |
| sizeof(tSirMacAddr)); |
| |
| sirCopyMacAddr(pHdr->bssId,psessionEntry->bssId); |
| |
| /* Copy beacon interval from sessionTable */ |
| cfg = psessionEntry->beaconParams.beaconInterval; |
| if (cfg != pBeacon->beaconInterval) |
| psessionEntry->beaconParams.beaconInterval = pBeacon->beaconInterval; |
| |
| /* This function ibss_bss_add (and hence the below code) is only called during ibss coalescing. We need to |
| * adapt to peer's capability with respect to short slot time. Changes have been made to limApplyConfiguration() |
| * so that the IBSS doesnt blindly start with short slot = 1. If IBSS start is part of coalescing then it will adapt |
| * to peer's short slot using code below. |
| */ |
| /* If cfg is already set to current peer's capability then no need to set it again */ |
| if (psessionEntry->shortSlotTimeSupported != pBeacon->capabilityInfo.shortSlotTime) |
| { |
| psessionEntry->shortSlotTimeSupported = pBeacon->capabilityInfo.shortSlotTime; |
| } |
| vos_mem_copy((tANI_U8 *) &psessionEntry->pLimStartBssReq->operationalRateSet, |
| (tANI_U8 *) &pBeacon->supportedRates, |
| pBeacon->supportedRates.numRates); |
| |
| /** |
| * WNI_CFG_EXTENDED_OPERATIONAL_RATE_SET CFG needs to be reset, when |
| * there is no extended rate IE present in beacon. This is especially important when |
| * supportedRateSet IE contains all the extended rates as well and STA decides to coalesce. |
| * In this IBSS coalescing scenario LIM will tear down the BSS and Add a new one. So LIM needs to |
| * reset this CFG, just in case CSR originally had set this CFG when IBSS was started from the local profile. |
| * If IBSS was started by CSR from the BssDescription, then it would reset this CFG before StartBss is issued. |
| * The idea is that the count of OpRateSet and ExtendedOpRateSet rates should not be more than 12. |
| */ |
| |
| if(pBeacon->extendedRatesPresent) |
| numExtRates = pBeacon->extendedRates.numRates; |
| if (cfgSetStr(pMac, WNI_CFG_EXTENDED_OPERATIONAL_RATE_SET, |
| (tANI_U8 *) &pBeacon->extendedRates.rate, numExtRates) != eSIR_SUCCESS) |
| { |
| limLog(pMac, LOGP, FL("could not update ExtendedOperRateset at CFG")); |
| return; |
| } |
| |
| |
| /* |
| * Each IBSS node will advertise its own HT Capabilities instead of adapting to the Peer's capabilities |
| * If we don't do this then IBSS may not go back to full capabilities when the STA with lower capabilities |
| * leaves the IBSS. e.g. when non-CB STA joins an IBSS and then leaves, the IBSS will be stuck at non-CB mode |
| * even though all the nodes are capable of doing CB. |
| * so it is decided to leave the self HT capabilties intact. This may change if some issues are found in interop. |
| */ |
| vos_mem_set((void *) &mlmStartReq, sizeof(mlmStartReq), 0); |
| |
| vos_mem_copy(mlmStartReq.bssId, pHdr->bssId, sizeof(tSirMacAddr)); |
| mlmStartReq.rateSet.numRates = psessionEntry->pLimStartBssReq->operationalRateSet.numRates; |
| vos_mem_copy(&mlmStartReq.rateSet.rate[0], |
| &psessionEntry->pLimStartBssReq->operationalRateSet.rate[0], |
| mlmStartReq.rateSet.numRates); |
| mlmStartReq.bssType = eSIR_IBSS_MODE; |
| mlmStartReq.beaconPeriod = pBeacon->beaconInterval; |
| mlmStartReq.nwType = psessionEntry->pLimStartBssReq->nwType; //psessionEntry->nwType is also OK???? |
| mlmStartReq.htCapable = psessionEntry->htCapability; |
| mlmStartReq.htOperMode = pMac->lim.gHTOperMode; |
| mlmStartReq.dualCTSProtection = pMac->lim.gHTDualCTSProtection; |
| mlmStartReq.txChannelWidthSet = psessionEntry->htRecommendedTxWidthSet; |
| |
| /* reading the channel num from session Table */ |
| mlmStartReq.channelNumber = psessionEntry->currentOperChannel; |
| |
| mlmStartReq.cbMode = psessionEntry->pLimStartBssReq->cbMode; |
| |
| // Copy the SSID for RxP filtering based on SSID. |
| vos_mem_copy((tANI_U8 *) &mlmStartReq.ssId, |
| (tANI_U8 *) &psessionEntry->pLimStartBssReq->ssId, |
| psessionEntry->pLimStartBssReq->ssId.length + 1); |
| |
| PELOG1(limLog(pMac, LOG1, FL("invoking ADD_BSS as part of coalescing!"));) |
| if (limMlmAddBss(pMac, &mlmStartReq,psessionEntry) != eSIR_SME_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("AddBss failure"));) |
| return; |
| } |
| |
| // Update fields in Beacon |
| if (schSetFixedBeaconFields(pMac,psessionEntry) != eSIR_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("*** Unable to set fixed Beacon fields ***"));) |
| return; |
| } |
| |
| } |
| |
| |
| |
| /* delete the current BSS */ |
| static void |
| ibss_bss_delete( |
| tpAniSirGlobal pMac, |
| tpPESession psessionEntry) |
| { |
| tSirRetStatus status; |
| PELOGW(limLog(pMac, LOGW, FL("Initiating IBSS Delete BSS"));) |
| if (psessionEntry->limMlmState != eLIM_MLM_BSS_STARTED_STATE) |
| { |
| limLog(pMac, LOGW, FL("Incorrect LIM MLM state for delBss (%d)"), |
| psessionEntry->limMlmState); |
| return; |
| } |
| status = limDelBss(pMac, NULL, psessionEntry->bssIdx, psessionEntry); |
| if (status != eSIR_SUCCESS) |
| PELOGE(limLog(pMac, LOGE, FL("delBss failed for bss %d"), psessionEntry->bssIdx);) |
| } |
| |
| /** |
| * limIbssInit |
| * |
| *FUNCTION: |
| * This function is called while starting an IBSS |
| * to initialize list used to maintain IBSS peers. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @return None |
| */ |
| |
| void |
| limIbssInit( |
| tpAniSirGlobal pMac) |
| { |
| pMac->lim.gLimIbssCoalescingHappened = 0; |
| pMac->lim.gLimIbssPeerList = NULL; |
| pMac->lim.gLimNumIbssPeers = 0; |
| |
| // ibss info - params for which ibss to join while coalescing |
| vos_mem_set(&pMac->lim.ibssInfo, sizeof(tAniSirLimIbss), 0); |
| } /*** end limIbssInit() ***/ |
| |
| /** |
| * limIbssDeleteAllPeers |
| * |
| *FUNCTION: |
| * This function is called to delete all peers. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @return None |
| */ |
| |
| void limIbssDeleteAllPeers( tpAniSirGlobal pMac ,tpPESession psessionEntry) |
| { |
| tLimIbssPeerNode *pCurrNode, *pTempNode; |
| tpDphHashNode pStaDs; |
| tANI_U16 peerIdx; |
| |
| pCurrNode = pTempNode = pMac->lim.gLimIbssPeerList; |
| |
| while (pCurrNode != NULL) |
| { |
| if (!pMac->lim.gLimNumIbssPeers) |
| { |
| limLog(pMac, LOGP, |
| FL("Number of peers in the list is zero and node present")); |
| return; |
| } |
| /* Delete the dph entry for the station |
| * Since it is called to remove all peers, just delete from dph, |
| * no need to do any beacon related params i.e., dont call limDeleteDphHashEntry |
| */ |
| pStaDs = dphLookupHashEntry(pMac, pCurrNode->peerMacAddr, &peerIdx, &psessionEntry->dph.dphHashTable); |
| if( pStaDs ) |
| { |
| |
| ibss_status_chg_notify( pMac, pCurrNode->peerMacAddr, pStaDs->staIndex, |
| pStaDs->ucUcastSig, pStaDs->ucBcastSig, |
| eWNI_SME_IBSS_PEER_DEPARTED_IND, psessionEntry->smeSessionId ); |
| limReleasePeerIdx(pMac, peerIdx, psessionEntry); |
| dphDeleteHashEntry(pMac, pStaDs->staAddr, peerIdx, &psessionEntry->dph.dphHashTable); |
| } |
| |
| pTempNode = pCurrNode->next; |
| |
| /* TODO :Sessionize this code */ |
| /* Fix CR 227642: PeerList should point to the next node since the current node is being |
| * freed in the next line. In ibss_peerfind in ibss_status_chg_notify above, we use this |
| * peer list to find the next peer. So this list needs to be updated with the no of peers left |
| * after each iteration in this while loop since one by one peers are deleted (freed) in this |
| * loop causing the lim.gLimIbssPeerList to point to some freed memory. |
| */ |
| pMac->lim.gLimIbssPeerList = pTempNode; |
| |
| if(pCurrNode->beacon) |
| { |
| vos_mem_free(pCurrNode->beacon); |
| } |
| vos_mem_free(pCurrNode); |
| if (pMac->lim.gLimNumIbssPeers > 0) // be paranoid |
| pMac->lim.gLimNumIbssPeers--; |
| pCurrNode = pTempNode; |
| } |
| |
| if (pMac->lim.gLimNumIbssPeers) |
| limLog(pMac, LOGP, FL("Number of peers[%d] in the list is non-zero"), |
| pMac->lim.gLimNumIbssPeers); |
| |
| pMac->lim.gLimNumIbssPeers = 0; |
| pMac->lim.gLimIbssPeerList = NULL; |
| |
| } |
| /** |
| * limIbssDelete |
| * |
| *FUNCTION: |
| * This function is called while tearing down an IBSS. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @return None |
| */ |
| |
| void |
| limIbssDelete( |
| tpAniSirGlobal pMac,tpPESession psessionEntry) |
| { |
| limIbssDeleteAllPeers(pMac,psessionEntry); |
| |
| ibss_coalesce_free(pMac); |
| } /*** end limIbssDelete() ***/ |
| |
| /** ------------------------------------------------------------- |
| \fn limIbssSetProtection |
| \brief Decides all the protection related information. |
| \ |
| \param tpAniSirGlobal pMac |
| \param tSirMacAddr peerMacAddr |
| \param tpUpdateBeaconParams pBeaconParams |
| \return None |
| -------------------------------------------------------------*/ |
| static void |
| limIbssSetProtection(tpAniSirGlobal pMac, tANI_U8 enable, tpUpdateBeaconParams pBeaconParams, tpPESession psessionEntry) |
| { |
| |
| if(!pMac->lim.cfgProtection.fromllb) |
| { |
| PELOG1(limLog(pMac, LOG1, FL("protection from 11b is disabled"));) |
| return; |
| } |
| |
| if (enable) |
| { |
| psessionEntry->gLim11bParams.protectionEnabled = true; |
| if(false == psessionEntry->beaconParams.llbCoexist/*pMac->lim.llbCoexist*/) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("=> IBSS: Enable Protection "));) |
| pBeaconParams->llbCoexist = psessionEntry->beaconParams.llbCoexist = true; |
| pBeaconParams->paramChangeBitmap |= PARAM_llBCOEXIST_CHANGED; |
| } |
| } |
| else if (true == psessionEntry->beaconParams.llbCoexist/*pMac->lim.llbCoexist*/) |
| { |
| psessionEntry->gLim11bParams.protectionEnabled = false; |
| PELOGE(limLog(pMac, LOGE, FL("===> IBSS: Disable protection "));) |
| pBeaconParams->llbCoexist = psessionEntry->beaconParams.llbCoexist = false; |
| pBeaconParams->paramChangeBitmap |= PARAM_llBCOEXIST_CHANGED; |
| } |
| return; |
| } |
| |
| |
| /** ------------------------------------------------------------- |
| \fn limIbssUpdateProtectionParams |
| \brief Decides all the protection related information. |
| \ |
| \param tpAniSirGlobal pMac |
| \param tSirMacAddr peerMacAddr |
| \param tpUpdateBeaconParams pBeaconParams |
| \return None |
| -------------------------------------------------------------*/ |
| static void |
| limIbssUpdateProtectionParams(tpAniSirGlobal pMac, |
| tSirMacAddr peerMacAddr, tLimProtStaCacheType protStaCacheType, |
| tpPESession psessionEntry) |
| { |
| tANI_U32 i; |
| |
| PELOG1(limLog(pMac,LOG1, FL("A STA is associated:")); |
| limLog(pMac,LOG1, FL("Addr : ")); |
| limPrintMacAddr(pMac, peerMacAddr, LOG1);) |
| |
| for (i=0; i<LIM_PROT_STA_CACHE_SIZE; i++) |
| { |
| if (pMac->lim.protStaCache[i].active) |
| { |
| PELOG1(limLog(pMac, LOG1, FL("Addr: "));) |
| PELOG1(limPrintMacAddr(pMac, pMac->lim.protStaCache[i].addr, LOG1);) |
| |
| if (vos_mem_compare(pMac->lim.protStaCache[i].addr, |
| peerMacAddr, sizeof(tSirMacAddr))) |
| { |
| PELOG1(limLog(pMac, LOG1, FL("matching cache entry at %d already active."), i);) |
| return; |
| } |
| } |
| } |
| |
| for (i=0; i<LIM_PROT_STA_CACHE_SIZE; i++) |
| { |
| if (!pMac->lim.protStaCache[i].active) |
| break; |
| } |
| |
| if (i >= LIM_PROT_STA_CACHE_SIZE) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("No space in ProtStaCache"));) |
| return; |
| } |
| |
| vos_mem_copy(pMac->lim.protStaCache[i].addr, |
| peerMacAddr, |
| sizeof(tSirMacAddr)); |
| |
| pMac->lim.protStaCache[i].protStaCacheType = protStaCacheType; |
| pMac->lim.protStaCache[i].active = true; |
| if(eLIM_PROT_STA_CACHE_TYPE_llB == protStaCacheType) |
| { |
| psessionEntry->gLim11bParams.numSta++; |
| } |
| else if(eLIM_PROT_STA_CACHE_TYPE_llG == protStaCacheType) |
| { |
| psessionEntry->gLim11gParams.numSta++; |
| } |
| } |
| |
| |
| /** ------------------------------------------------------------- |
| \fn limIbssDecideProtection |
| \brief Decides all the protection related information. |
| \ |
| \param tpAniSirGlobal pMac |
| \param tSirMacAddr peerMacAddr |
| \param tpUpdateBeaconParams pBeaconParams |
| \return None |
| -------------------------------------------------------------*/ |
| static void |
| limIbssDecideProtection(tpAniSirGlobal pMac, tpDphHashNode pStaDs, tpUpdateBeaconParams pBeaconParams, tpPESession psessionEntry) |
| { |
| tSirRFBand rfBand = SIR_BAND_UNKNOWN; |
| tANI_U32 phyMode; |
| tLimProtStaCacheType protStaCacheType = eLIM_PROT_STA_CACHE_TYPE_INVALID; |
| |
| pBeaconParams->paramChangeBitmap = 0; |
| |
| if(NULL == pStaDs) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("pStaDs is NULL"));) |
| return; |
| } |
| |
| limGetRfBand(pMac, &rfBand, psessionEntry); |
| if(SIR_BAND_2_4_GHZ== rfBand) |
| { |
| limGetPhyMode(pMac, &phyMode, psessionEntry); |
| |
| //We are 11G or 11n. Check if we need protection from 11b Stations. |
| if ((phyMode == WNI_CFG_PHY_MODE_11G) || (psessionEntry->htCapability)) |
| { |
| /* As we found in the past, it is possible that a 11n STA sends |
| * Beacon with HT IE but not ERP IE. So the absense of ERP IE |
| * in the Beacon is not enough to conclude that STA is 11b. |
| */ |
| if ((pStaDs->erpEnabled == eHAL_CLEAR) && |
| (!pStaDs->mlmStaContext.htCapability)) |
| { |
| protStaCacheType = eLIM_PROT_STA_CACHE_TYPE_llB; |
| PELOGE(limLog(pMac, LOGE, FL("Enable protection from 11B"));) |
| limIbssSetProtection(pMac, true, pBeaconParams,psessionEntry); |
| } |
| } |
| } |
| limIbssUpdateProtectionParams(pMac, pStaDs->staAddr, protStaCacheType, psessionEntry); |
| return; |
| } |
| |
| /** |
| * limIbssPeerFind() |
| * |
| *FUNCTION: |
| * This function is called while adding a context at |
| * DPH for a peer in IBSS. |
| * If peer is found in the list, capabilities from the |
| * returned BSS description are used at DPH node. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param macAddr - MAC address of the peer |
| * |
| * @return Pointer to peer node if found, else NULL |
| */ |
| tLimIbssPeerNode* limIbssPeerFind(tpAniSirGlobal pMac, tSirMacAddr macAddr) |
| { |
| return ibss_peer_find(pMac, macAddr); |
| } |
| |
| /** |
| * limIbssStaAdd() |
| * |
| *FUNCTION: |
| * This function is called to add an STA context in IBSS role |
| * whenever a data frame is received from/for a STA that failed |
| * hash lookup at DPH. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * NA |
| * |
| *NOTE: |
| * NA |
| * |
| * @param pMac Pointer to Global MAC structure |
| * @param peerAdddr MAC address of the peer being added |
| * @return retCode Indicates success or failure return code |
| * @return |
| */ |
| |
| tSirRetStatus |
| limIbssStaAdd( |
| tpAniSirGlobal pMac, |
| void *pBody, |
| tpPESession psessionEntry) |
| { |
| tSirRetStatus retCode = eSIR_SUCCESS; |
| tpDphHashNode pStaDs; |
| tLimIbssPeerNode *pPeerNode; |
| tLimMlmStates prevState; |
| tSirMacAddr *pPeerAddr = (tSirMacAddr *) pBody; |
| tUpdateBeaconParams beaconParams; |
| |
| vos_mem_set((tANI_U8 *) &beaconParams, sizeof(tUpdateBeaconParams), 0); |
| |
| if (pBody == 0) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("Invalid IBSS AddSta"));) |
| return eSIR_FAILURE; |
| } |
| |
| PELOGE(limLog(pMac, LOGE, FL("Rx Add-Ibss-Sta for MAC:"));) |
| limPrintMacAddr(pMac, *pPeerAddr, LOGE); |
| |
| pPeerNode = ibss_peer_find(pMac, *pPeerAddr); |
| if (NULL != pPeerNode) |
| { |
| retCode = ibss_dph_entry_add(pMac, *pPeerAddr, &pStaDs, psessionEntry); |
| if (eSIR_SUCCESS == retCode) |
| { |
| prevState = pStaDs->mlmStaContext.mlmState; |
| pStaDs->erpEnabled = pPeerNode->erpIePresent; |
| |
| ibss_sta_info_update(pMac, pStaDs, pPeerNode, psessionEntry); |
| PELOGW(limLog(pMac, LOGW, FL("initiating ADD STA for the IBSS peer."));) |
| retCode = limAddSta(pMac, pStaDs, false, psessionEntry); |
| if (retCode != eSIR_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("ibss-sta-add failed (reason %x)"), |
| retCode);) |
| limPrintMacAddr(pMac, *pPeerAddr, LOGE); |
| pStaDs->mlmStaContext.mlmState = prevState; |
| dphDeleteHashEntry(pMac, pStaDs->staAddr, pStaDs->assocId, |
| &psessionEntry->dph.dphHashTable); |
| } |
| else |
| { |
| if(pMac->lim.gLimProtectionControl != WNI_CFG_FORCE_POLICY_PROTECTION_DISABLE) |
| limIbssDecideProtection(pMac, pStaDs, &beaconParams , psessionEntry); |
| |
| if(beaconParams.paramChangeBitmap) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("---> Update Beacon Params "));) |
| schSetFixedBeaconFields(pMac, psessionEntry); |
| beaconParams.bssIdx = psessionEntry->bssIdx; |
| limSendBeaconParams(pMac, &beaconParams, psessionEntry ); |
| } |
| } |
| } |
| else |
| { |
| PELOGE(limLog(pMac, LOGE, FL("hashTblAdd failed (reason %x)"), retCode);) |
| limPrintMacAddr(pMac, *pPeerAddr, LOGE); |
| } |
| } |
| else |
| { |
| retCode = eSIR_FAILURE; |
| } |
| |
| return retCode; |
| } |
| |
| static void |
| __limIbssSearchAndDeletePeer(tpAniSirGlobal pMac, |
| tpPESession psessionEntry, |
| tSirMacAddr macAddr) |
| { |
| tLimIbssPeerNode *pTempNode, *pPrevNode; |
| tLimIbssPeerNode *pTempNextNode = NULL; |
| tpDphHashNode pStaDs=NULL; |
| tANI_U16 peerIdx=0; |
| tANI_U16 staIndex=0; |
| tANI_U8 ucUcastSig; |
| tANI_U8 ucBcastSig; |
| |
| pPrevNode = pTempNode = pMac->lim.gLimIbssPeerList; |
| |
| limLog(pMac, LOG1, FL("PEER ADDR :" MAC_ADDRESS_STR), |
| MAC_ADDR_ARRAY(macAddr)); |
| |
| /* Compare Peer */ |
| while (NULL != pTempNode) { |
| pTempNextNode = pTempNode->next; |
| |
| /* Delete the STA with MAC address */ |
| if (vos_mem_compare((tANI_U8 *) macAddr, |
| (tANI_U8 *) &pTempNode->peerMacAddr, |
| sizeof(tSirMacAddr))) { |
| pStaDs = dphLookupHashEntry(pMac, macAddr, |
| &peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs) { |
| staIndex = pStaDs->staIndex; |
| ucUcastSig = pStaDs->ucUcastSig; |
| ucBcastSig = pStaDs->ucBcastSig; |
| /** |
| * Send DEL STA only if ADD STA |
| * was success i.e staid is Valid. |
| */ |
| if (HAL_STA_INVALID_IDX != staIndex) |
| limDelSta(pMac, pStaDs, |
| false /*asynchronous*/, |
| psessionEntry); |
| limDeleteDphHashEntry(pMac, |
| pStaDs->staAddr, |
| peerIdx, psessionEntry); |
| limReleasePeerIdx(pMac, peerIdx, psessionEntry); |
| /** |
| * Send indication to upper layers only if ADD |
| * STA was success i.e staid is Valid. |
| */ |
| if (HAL_STA_INVALID_IDX != staIndex) |
| ibss_status_chg_notify(pMac, macAddr, |
| staIndex, |
| ucUcastSig, ucBcastSig, |
| eWNI_SME_IBSS_PEER_DEPARTED_IND, |
| psessionEntry->smeSessionId); |
| if (pTempNode == pMac->lim.gLimIbssPeerList) { |
| pMac->lim.gLimIbssPeerList = |
| pTempNode->next; |
| pPrevNode = pMac->lim.gLimIbssPeerList; |
| } else |
| pPrevNode->next = pTempNode->next; |
| if (pTempNode->beacon) |
| vos_mem_free(pTempNode->beacon); |
| vos_mem_free(pTempNode); |
| pMac->lim.gLimNumIbssPeers--; |
| |
| pTempNode = pTempNextNode; |
| break; |
| } |
| } |
| pPrevNode = pTempNode; |
| pTempNode = pTempNextNode; |
| } |
| /* |
| * if it is the last peer walking out, we better |
| * we set IBSS state to inactive. |
| */ |
| if (0 == pMac->lim.gLimNumIbssPeers) { |
| VOS_TRACE(VOS_MODULE_ID_PE, VOS_TRACE_LEVEL_INFO, |
| "Last STA from IBSS walked out"); |
| psessionEntry->limIbssActive = false; |
| } |
| } |
| |
| /* handle the response from HAL for an ADD STA request */ |
| tSirRetStatus |
| limIbssAddStaRsp( |
| tpAniSirGlobal pMac, |
| void *msg,tpPESession psessionEntry) |
| { |
| tpDphHashNode pStaDs; |
| tANI_U16 peerIdx; |
| tpAddStaParams pAddStaParams = (tpAddStaParams) msg; |
| |
| SET_LIM_PROCESS_DEFD_MESGS(pMac, true); |
| if (pAddStaParams == NULL) |
| { |
| limLog(pMac, LOGE, FL("IBSS: ADD_STA_RSP with no body!")); |
| return eSIR_FAILURE; |
| } |
| |
| pStaDs = |
| dphLookupHashEntry(pMac, |
| pAddStaParams->staMac, &peerIdx, |
| &psessionEntry->dph.dphHashTable); |
| if (pStaDs == NULL) |
| { |
| limLog(pMac, LOGE, |
| FL("IBSS: ADD_STA_RSP for unknown MAC addr " MAC_ADDRESS_STR), |
| MAC_ADDR_ARRAY(pAddStaParams->staMac)); |
| vos_mem_free(pAddStaParams); |
| return eSIR_FAILURE; |
| } |
| |
| if (pAddStaParams->status != eHAL_STATUS_SUCCESS) |
| { |
| limLog(pMac, LOGE, |
| FL("IBSS: ADD_STA_RSP error (%x) for MAC " MAC_ADDRESS_STR), |
| pAddStaParams->status, |
| MAC_ADDR_ARRAY(pAddStaParams->staMac)); |
| __limIbssSearchAndDeletePeer(pMac, |
| psessionEntry, pAddStaParams->staMac); |
| vos_mem_free(pAddStaParams); |
| return eSIR_FAILURE; |
| } |
| |
| pStaDs->bssId = pAddStaParams->bssIdx; |
| pStaDs->staIndex = pAddStaParams->staIdx; |
| pStaDs->ucUcastSig = pAddStaParams->ucUcastSig; |
| pStaDs->ucBcastSig = pAddStaParams->ucBcastSig; |
| pStaDs->valid = 1; |
| pStaDs->mlmStaContext.mlmState = eLIM_MLM_LINK_ESTABLISHED_STATE; |
| |
| PELOGW(limLog(pMac, LOGW, FL("IBSS: sending IBSS_NEW_PEER msg to SME!"));) |
| |
| ibss_status_chg_notify(pMac, pAddStaParams->staMac, pStaDs->staIndex, |
| pStaDs->ucUcastSig, pStaDs->ucBcastSig, |
| eWNI_SME_IBSS_NEW_PEER_IND, |
| psessionEntry->smeSessionId); |
| |
| vos_mem_free(pAddStaParams); |
| |
| return eSIR_SUCCESS; |
| } |
| |
| |
| |
| void limIbssDelBssRspWhenCoalescing(tpAniSirGlobal pMac, void *msg,tpPESession psessionEntry) |
| { |
| tpDeleteBssParams pDelBss = (tpDeleteBssParams) msg; |
| |
| PELOGW(limLog(pMac, LOGW, FL("IBSS: DEL_BSS_RSP Rcvd during coalescing!"));) |
| |
| if (pDelBss == NULL) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("IBSS: DEL_BSS_RSP(coalesce) with no body!"));) |
| goto end; |
| } |
| |
| if (pDelBss->status != eHAL_STATUS_SUCCESS) |
| { |
| limLog(pMac, LOGE, FL("IBSS: DEL_BSS_RSP(coalesce) error (%x) Bss %d "), |
| pDelBss->status, pDelBss->bssIdx); |
| goto end; |
| } |
| //Delete peer entries. |
| limIbssDeleteAllPeers(pMac,psessionEntry); |
| |
| /* add the new bss */ |
| ibss_bss_add(pMac,psessionEntry); |
| |
| end: |
| if(pDelBss != NULL) |
| vos_mem_free(pDelBss); |
| } |
| |
| |
| |
| void limIbssAddBssRspWhenCoalescing(tpAniSirGlobal pMac, void *msg, tpPESession pSessionEntry) |
| { |
| tANI_U8 infoLen; |
| tSirSmeNewBssInfo newBssInfo; |
| |
| tpAddBssParams pAddBss = (tpAddBssParams) msg; |
| |
| tpSirMacMgmtHdr pHdr = (tpSirMacMgmtHdr) pMac->lim.ibssInfo.pHdr; |
| tpSchBeaconStruct pBeacon = (tpSchBeaconStruct) pMac->lim.ibssInfo.pBeacon; |
| |
| if ((pHdr == NULL) || (pBeacon == NULL)) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("Unable to handle AddBssRspWhenCoalescing (no cached BSS info)"));) |
| goto end; |
| } |
| |
| // Inform Host of IBSS coalescing |
| infoLen = sizeof(tSirMacAddr) + sizeof(tSirMacChanNum) + |
| sizeof(tANI_U8) + pBeacon->ssId.length + 1; |
| |
| vos_mem_set((void *) &newBssInfo, sizeof(newBssInfo), 0); |
| vos_mem_copy(newBssInfo.bssId, pHdr->bssId, sizeof(tSirMacAddr)); |
| newBssInfo.channelNumber = (tSirMacChanNum) pAddBss->currentOperChannel; |
| vos_mem_copy((tANI_U8 *) &newBssInfo.ssId, |
| (tANI_U8 *) &pBeacon->ssId, pBeacon->ssId.length + 1); |
| |
| PELOGW(limLog(pMac, LOGW, FL("Sending JOINED_NEW_BSS notification to SME."));) |
| |
| limSendSmeWmStatusChangeNtf(pMac, eSIR_SME_JOINED_NEW_BSS, |
| (tANI_U32 *) &newBssInfo, |
| infoLen,pSessionEntry->smeSessionId); |
| { |
| //Configure beacon and send beacons to HAL |
| limSendBeaconInd(pMac, pSessionEntry); |
| } |
| |
| end: |
| ibss_coalesce_free(pMac); |
| } |
| |
| |
| |
| void |
| limIbssDelBssRsp( |
| tpAniSirGlobal pMac, |
| void *msg,tpPESession psessionEntry) |
| { |
| tSirResultCodes rc = eSIR_SME_SUCCESS; |
| tpDeleteBssParams pDelBss = (tpDeleteBssParams) msg; |
| tSirMacAddr nullBssid = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; |
| |
| |
| SET_LIM_PROCESS_DEFD_MESGS(pMac, true); |
| if (pDelBss == NULL) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("IBSS: DEL_BSS_RSP with no body!"));) |
| rc = eSIR_SME_REFUSED; |
| goto end; |
| } |
| |
| if((psessionEntry = peFindSessionBySessionId(pMac,pDelBss->sessionId))==NULL) |
| { |
| limLog(pMac, LOGP,FL("Session Does not exist for given sessionID")); |
| goto end; |
| } |
| |
| |
| /* |
| * If delBss was issued as part of IBSS Coalescing, gLimIbssCoalescingHappened flag will be true. |
| * BSS has to be added again in this scenario, so this case needs to be handled separately. |
| * If delBss was issued as a result of trigger from SME_STOP_BSS Request, then limSme state changes to |
| * 'IDLE' and gLimIbssCoalescingHappened flag will be false. In this case STOP BSS RSP has to be sent to SME. |
| */ |
| if(true == pMac->lim.gLimIbssCoalescingHappened) |
| { |
| |
| limIbssDelBssRspWhenCoalescing(pMac,msg,psessionEntry); |
| return; |
| } |
| |
| |
| |
| if (pDelBss->status != eHAL_STATUS_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("IBSS: DEL_BSS_RSP error (%x) Bss %d "), |
| pDelBss->status, pDelBss->bssIdx);) |
| rc = eSIR_SME_STOP_BSS_FAILURE; |
| goto end; |
| } |
| |
| |
| |
| if(limSetLinkState(pMac, eSIR_LINK_IDLE_STATE, nullBssid, |
| psessionEntry->selfMacAddr, NULL, NULL) != eSIR_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("IBSS: DEL_BSS_RSP setLinkState failed"));) |
| rc = eSIR_SME_REFUSED; |
| goto end; |
| } |
| |
| limIbssDelete(pMac,psessionEntry); |
| |
| dphHashTableClassInit(pMac, &psessionEntry->dph.dphHashTable); |
| limDeletePreAuthList(pMac); |
| |
| psessionEntry->limMlmState = eLIM_MLM_IDLE_STATE; |
| |
| MTRACE(macTrace(pMac, TRACE_CODE_MLM_STATE, psessionEntry->peSessionId, psessionEntry->limMlmState)); |
| |
| psessionEntry->limSystemRole = eLIM_STA_ROLE; |
| |
| /* Change the short slot operating mode to Default (which is 1 for now) so that when IBSS starts next time with Libra |
| * as originator, it picks up the default. This enables us to remove hard coding of short slot = 1 from limApplyConfiguration |
| */ |
| psessionEntry->shortSlotTimeSupported = WNI_CFG_SHORT_SLOT_TIME_STADEF; |
| |
| end: |
| if(pDelBss != NULL) |
| vos_mem_free(pDelBss); |
| /* Delete PE session once BSS is deleted */ |
| if (NULL != psessionEntry) { |
| limSendSmeRsp(pMac, eWNI_SME_STOP_BSS_RSP, rc,psessionEntry->smeSessionId,psessionEntry->transactionId); |
| peDeleteSession(pMac, psessionEntry); |
| psessionEntry = NULL; |
| } |
| } |
| |
| /** |
| * limIbssCoalesce() |
| * |
| *FUNCTION: |
| * This function is called upon receiving Beacon/Probe Response |
| * while operating in IBSS mode. |
| * |
| *LOGIC: |
| * |
| *ASSUMPTIONS: |
| * |
| *NOTE: |
| * |
| * @param pMac - Pointer to Global MAC structure |
| * @param pBeacon - Parsed Beacon Frame structure |
| * @param pBD - Pointer to received BD |
| * |
| * @return Status whether to process or ignore received Beacon Frame |
| */ |
| |
| tSirRetStatus |
| limIbssCoalesce( |
| tpAniSirGlobal pMac, |
| tpSirMacMgmtHdr pHdr, |
| tpSchBeaconStruct pBeacon, |
| tANI_U8 *pIEs, |
| tANI_U32 ieLen, |
| tANI_U16 fTsfLater, |
| tpPESession psessionEntry) |
| { |
| tANI_U16 peerIdx; |
| tSirMacAddr currentBssId; |
| tLimIbssPeerNode *pPeerNode; |
| tpDphHashNode pStaDs; |
| tUpdateBeaconParams beaconParams; |
| |
| vos_mem_set((tANI_U8 *)&beaconParams, sizeof(tUpdateBeaconParams), 0); |
| |
| sirCopyMacAddr(currentBssId,psessionEntry->bssId); |
| |
| limLog(pMac, LOG1, FL("Current BSSID :" MAC_ADDRESS_STR " Received BSSID :" MAC_ADDRESS_STR ), |
| MAC_ADDR_ARRAY(currentBssId), MAC_ADDR_ARRAY(pHdr->bssId)); |
| |
| /* Check for IBSS Coalescing only if Beacon is from different BSS */ |
| if ( !vos_mem_compare(currentBssId, pHdr->bssId, sizeof( tSirMacAddr )) |
| && psessionEntry->isCoalesingInIBSSAllowed) |
| { |
| /* |
| * If STA entry is already available in the LIM hash table, then it is |
| * possible that the peer may have left and rejoined within the heartbeat |
| * timeout. In the offloaded case with 32 peers, the HB timeout is whopping |
| * 128 seconds. In that case, the FW will not let any frames come in until |
| * atleast the last sequence number is received before the peer is left |
| * Hence, if the coalescing peer is already there in the peer list and if |
| * the BSSID matches then, invoke delSta() to cleanup the entries. We will |
| * let the peer coalesce when we receive next beacon from the peer |
| */ |
| pPeerNode = ibss_peer_find(pMac, pHdr->sa); |
| if (NULL != pPeerNode) |
| { |
| __limIbssSearchAndDeletePeer (pMac, psessionEntry, pHdr->sa); |
| PELOGW(limLog(pMac, LOGW, |
| FL("** Peer attempting to reconnect before HB timeout, deleted **"));) |
| return eSIR_LIM_IGNORE_BEACON; |
| } |
| |
| if (! fTsfLater) // No Coalescing happened. |
| { |
| PELOGW(limLog(pMac, LOGW, FL("No Coalescing happened"));) |
| return eSIR_LIM_IGNORE_BEACON; |
| } |
| /* |
| * IBSS Coalescing happened. |
| * save the received beacon, and delete the current BSS. The rest of the |
| * processing will be done in the delBss response processing |
| */ |
| pMac->lim.gLimIbssCoalescingHappened = true; |
| PELOGW(limLog(pMac, LOGW, FL("IBSS Coalescing happened"));) |
| ibss_coalesce_save(pMac, pHdr, pBeacon); |
| limLog(pMac, LOGW, FL("Delete BSSID :" MAC_ADDRESS_STR ), |
| MAC_ADDR_ARRAY(currentBssId)); |
| ibss_bss_delete(pMac,psessionEntry); |
| return eSIR_SUCCESS; |
| } |
| else |
| { |
| if (!vos_mem_compare(currentBssId, pHdr->bssId, sizeof( tSirMacAddr ))) |
| return eSIR_LIM_IGNORE_BEACON; |
| } |
| |
| |
| // STA in IBSS mode and SSID matches with ours |
| pPeerNode = ibss_peer_find(pMac, pHdr->sa); |
| if (pPeerNode == NULL) |
| { |
| /* Peer not in the list - Collect BSS description & add to the list */ |
| tANI_U32 frameLen; |
| tSirRetStatus retCode; |
| |
| /* |
| * Limit the Max number of IBSS Peers allowed as the max |
| * number of STA's allowed |
| * pMac->lim.gLimNumIbssPeers will be increamented after exiting |
| * this function. so we will add additional 1 to compare against |
| * pMac->lim.gLimIbssStaLimit |
| */ |
| if ((pMac->lim.gLimNumIbssPeers+1) >= pMac->lim.gLimIbssStaLimit) |
| { |
| /*Print every 100th time */ |
| if (pMac->lim.gLimIbssRetryCnt % 100 == 0) |
| { |
| PELOGE(limLog(pMac, LOG1, FL("**** MAX STA LIMIT HAS REACHED ****"));) |
| } |
| pMac->lim.gLimIbssRetryCnt++; |
| return eSIR_LIM_MAX_STA_REACHED_ERROR; |
| } |
| PELOGW(limLog(pMac, LOGW, FL("IBSS Peer node does not exist, adding it***"));) |
| frameLen = sizeof(tLimIbssPeerNode) + ieLen - sizeof(tANI_U32); |
| |
| pPeerNode = vos_mem_malloc((tANI_U16)frameLen); |
| if (NULL == pPeerNode) |
| { |
| limLog(pMac, LOGP, FL("alloc fail (%d bytes) storing IBSS peer info"), |
| frameLen); |
| return eSIR_MEM_ALLOC_FAILED; |
| } |
| |
| /* Initialize all peer node properties to 0 */ |
| vos_mem_zero(pPeerNode, frameLen); |
| |
| pPeerNode->beacon = NULL; |
| pPeerNode->beaconLen = 0; |
| |
| ibss_peer_collect(pMac, pBeacon, pHdr, pPeerNode,psessionEntry); |
| pPeerNode->beacon = vos_mem_malloc(ieLen); |
| if (NULL == pPeerNode->beacon) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("Unable to allocate memory to store beacon"));) |
| } |
| else |
| { |
| vos_mem_copy(pPeerNode->beacon, pIEs, ieLen); |
| pPeerNode->beaconLen = (tANI_U16)ieLen; |
| } |
| ibss_peer_add(pMac, pPeerNode); |
| |
| pStaDs = dphLookupHashEntry(pMac, pPeerNode->peerMacAddr, &peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs != NULL) |
| { |
| /// DPH node already exists for the peer |
| PELOGW(limLog(pMac, LOGW, FL("DPH Node present for just learned peer"));) |
| PELOG1(limPrintMacAddr(pMac, pPeerNode->peerMacAddr, LOG1);) |
| ibss_sta_info_update(pMac, pStaDs, pPeerNode,psessionEntry); |
| return eSIR_SUCCESS; |
| } |
| retCode = limIbssStaAdd(pMac, pPeerNode->peerMacAddr,psessionEntry); |
| if (retCode != eSIR_SUCCESS) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("lim-ibss-sta-add failed (reason %x)"), retCode);) |
| limPrintMacAddr(pMac, pPeerNode->peerMacAddr, LOGE); |
| return retCode; |
| } |
| |
| // Decide protection mode |
| pStaDs = dphLookupHashEntry(pMac, pPeerNode->peerMacAddr, &peerIdx, &psessionEntry->dph.dphHashTable); |
| if(pMac->lim.gLimProtectionControl != WNI_CFG_FORCE_POLICY_PROTECTION_DISABLE) |
| limIbssDecideProtection(pMac, pStaDs, &beaconParams, psessionEntry); |
| |
| if(beaconParams.paramChangeBitmap) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("beaconParams.paramChangeBitmap=1 ---> Update Beacon Params "));) |
| schSetFixedBeaconFields(pMac, psessionEntry); |
| beaconParams.bssIdx = psessionEntry->bssIdx; |
| limSendBeaconParams(pMac, &beaconParams, psessionEntry ); |
| } |
| } |
| else |
| ibss_sta_caps_update(pMac, pPeerNode,psessionEntry); |
| |
| if (psessionEntry->limSmeState != eLIM_SME_NORMAL_STATE) |
| return eSIR_SUCCESS; |
| |
| // Received Beacon from same IBSS we're |
| // currently part of. Inform Roaming algorithm |
| // if not already that IBSS is active. |
| if (psessionEntry->limIbssActive == false) |
| { |
| limResetHBPktCount(psessionEntry); |
| PELOGW(limLog(pMac, LOGW, FL("Partner joined our IBSS, Sending IBSS_ACTIVE Notification to SME"));) |
| psessionEntry->limIbssActive = true; |
| limSendSmeWmStatusChangeNtf(pMac, eSIR_SME_IBSS_ACTIVE, NULL, 0, psessionEntry->smeSessionId); |
| limHeartBeatDeactivateAndChangeTimer(pMac, psessionEntry); |
| MTRACE(macTrace(pMac, TRACE_CODE_TIMER_ACTIVATE, psessionEntry->peSessionId, eLIM_HEART_BEAT_TIMER)); |
| if (limActivateHearBeatTimer(pMac, psessionEntry) != TX_SUCCESS) |
| limLog(pMac, LOGP, FL("could not activate Heartbeat timer")); |
| } |
| |
| return eSIR_SUCCESS; |
| } /*** end limHandleIBSScoalescing() ***/ |
| |
| |
| void limIbssHeartBeatHandle(tpAniSirGlobal pMac,tpPESession psessionEntry) |
| { |
| tLimIbssPeerNode *pTempNode, *pPrevNode; |
| tLimIbssPeerNode *pTempNextNode = NULL; |
| tANI_U16 peerIdx=0; |
| tpDphHashNode pStaDs=0; |
| tANI_U32 threshold=0; |
| tANI_U16 staIndex=0; |
| tANI_U8 ucUcastSig=0; |
| tANI_U8 ucBcastSig=0; |
| |
| /** MLM BSS is started and if PE in scanmode then MLM state will be waiting for probe resp. |
| * If Heart beat timeout triggers during this corner case then we need to reactivate HeartBeat timer |
| */ |
| if(psessionEntry->limMlmState != eLIM_MLM_BSS_STARTED_STATE) { |
| /****** |
| * Note: Use this code once you have converted all |
| * limReactivateHeartBeatTimer() calls to |
| * limReactivateTimer() calls. |
| * |
| ******/ |
| limReactivateHeartBeatTimer(pMac, psessionEntry); |
| return; |
| } |
| /** If LinkMonitor is Disabled */ |
| if(!pMac->sys.gSysEnableLinkMonitorMode) |
| return; |
| |
| pPrevNode = pTempNode = pMac->lim.gLimIbssPeerList; |
| threshold = (pMac->lim.gLimNumIbssPeers / 4 ) + 1; |
| |
| /** Monitor the HeartBeat with the Individual PEERS in the IBSS */ |
| while (pTempNode != NULL) |
| { |
| pTempNextNode = pTempNode->next; |
| if(pTempNode->beaconHBCount) //There was a beacon for this peer during heart beat. |
| { |
| pTempNode->beaconHBCount = 0; |
| pTempNode->heartbeatFailure = 0; |
| } |
| else //There wasnt any beacon received during heartbeat timer. |
| { |
| pTempNode->heartbeatFailure++; |
| PELOGE(limLog(pMac, LOGE, FL("Heartbeat fail = %d thres = %d"), pTempNode->heartbeatFailure, pMac->lim.gLimNumIbssPeers);) |
| if(pTempNode->heartbeatFailure >= threshold ) |
| { |
| //Remove this entry from the list. |
| pStaDs = dphLookupHashEntry(pMac, pTempNode->peerMacAddr, &peerIdx, &psessionEntry->dph.dphHashTable); |
| if (pStaDs) |
| { |
| staIndex = pStaDs->staIndex; |
| ucUcastSig = pStaDs->ucUcastSig; |
| ucBcastSig = pStaDs->ucBcastSig; |
| |
| (void) limDelSta(pMac, pStaDs, false /*asynchronous*/,psessionEntry); |
| limDeleteDphHashEntry(pMac, pStaDs->staAddr, peerIdx,psessionEntry); |
| limReleasePeerIdx(pMac, peerIdx, psessionEntry); |
| //Send indication. |
| ibss_status_chg_notify( pMac, pTempNode->peerMacAddr, staIndex, |
| ucUcastSig, ucBcastSig, |
| eWNI_SME_IBSS_PEER_DEPARTED_IND, |
| psessionEntry->smeSessionId ); |
| } |
| if(pTempNode == pMac->lim.gLimIbssPeerList) |
| { |
| pMac->lim.gLimIbssPeerList = pTempNode->next; |
| pPrevNode = pMac->lim.gLimIbssPeerList; |
| } |
| else |
| pPrevNode->next = pTempNode->next; |
| |
| if (pTempNode->beacon) |
| vos_mem_free(pTempNode->beacon); |
| vos_mem_free(pTempNode); |
| pMac->lim.gLimNumIbssPeers--; |
| |
| pTempNode = pTempNextNode; //Since we deleted current node, prevNode remains same. |
| continue; |
| } |
| } |
| |
| pPrevNode = pTempNode; |
| pTempNode = pTempNextNode; |
| } |
| |
| /** General IBSS Activity Monitor, check if in IBSS Mode we are received any Beacons */ |
| if(pMac->lim.gLimNumIbssPeers) |
| { |
| if(psessionEntry->LimRxedBeaconCntDuringHB < MAX_NO_BEACONS_PER_HEART_BEAT_INTERVAL) |
| pMac->lim.gLimHeartBeatBeaconStats[psessionEntry->LimRxedBeaconCntDuringHB]++; |
| else |
| pMac->lim.gLimHeartBeatBeaconStats[0]++; |
| |
| limReactivateHeartBeatTimer(pMac, psessionEntry); |
| |
| // Reset number of beacons received |
| limResetHBPktCount(psessionEntry); |
| return; |
| } |
| else |
| { |
| |
| PELOGW(limLog(pMac, LOGW, FL("Heartbeat Failure"));) |
| pMac->lim.gLimHBfailureCntInLinkEstState++; |
| |
| if (psessionEntry->limIbssActive == true) |
| { |
| // We don't receive Beacon frames from any |
| // other STA in IBSS. Announce IBSS inactive |
| // to Roaming algorithm |
| PELOGW(limLog(pMac, LOGW, FL("Alone in IBSS"));) |
| psessionEntry->limIbssActive = false; |
| |
| limSendSmeWmStatusChangeNtf(pMac, eSIR_SME_IBSS_INACTIVE, |
| NULL, 0, psessionEntry->smeSessionId); |
| } |
| } |
| } |
| |
| |
| /** ------------------------------------------------------------- |
| \fn limIbssDecideProtectionOnDelete |
| \brief Decides all the protection related information. |
| \ |
| \param tpAniSirGlobal pMac |
| \param tSirMacAddr peerMacAddr |
| \param tpUpdateBeaconParams pBeaconParams |
| \return None |
| -------------------------------------------------------------*/ |
| void |
| limIbssDecideProtectionOnDelete(tpAniSirGlobal pMac, |
| tpDphHashNode pStaDs, tpUpdateBeaconParams pBeaconParams, tpPESession psessionEntry) |
| { |
| tANI_U32 phyMode; |
| tHalBitVal erpEnabled = eHAL_CLEAR; |
| tSirRFBand rfBand = SIR_BAND_UNKNOWN; |
| tANI_U32 i; |
| |
| if(NULL == pStaDs) |
| return; |
| |
| limGetRfBand(pMac, &rfBand, psessionEntry); |
| if(SIR_BAND_2_4_GHZ == rfBand) |
| { |
| limGetPhyMode(pMac, &phyMode, psessionEntry); |
| erpEnabled = pStaDs->erpEnabled; |
| //we are HT or 11G and 11B station is getting deleted. |
| if ( ((phyMode == WNI_CFG_PHY_MODE_11G) || psessionEntry->htCapability) |
| && (erpEnabled == eHAL_CLEAR)) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("(%d) A legacy STA is disassociated. Addr is "), |
| psessionEntry->gLim11bParams.numSta); |
| limPrintMacAddr(pMac, pStaDs->staAddr, LOGE);) |
| if (psessionEntry->gLim11bParams.numSta > 0) |
| { |
| for (i=0; i<LIM_PROT_STA_CACHE_SIZE; i++) |
| { |
| if (pMac->lim.protStaCache[i].active) |
| { |
| if (vos_mem_compare(pMac->lim.protStaCache[i].addr, |
| pStaDs->staAddr, sizeof(tSirMacAddr))) |
| { |
| psessionEntry->gLim11bParams.numSta--; |
| pMac->lim.protStaCache[i].active = false; |
| break; |
| } |
| } |
| } |
| } |
| |
| if (psessionEntry->gLim11bParams.numSta == 0) |
| { |
| PELOGE(limLog(pMac, LOGE, FL("No more 11B STA exists. Disable protection. "));) |
| limIbssSetProtection(pMac, false, pBeaconParams,psessionEntry); |
| } |
| } |
| } |
| } |
| |
| /** ----------------------------------------------------------------- |
| \fn __limIbssPeerInactivityHandler |
| \brief Internal function. Deletes FW indicated peer which is inactive |
| \ |
| \param tpAniSirGlobal pMac |
| \param tpPESession psessionEntry |
| \param tpSirIbssPeerInactivityInd peerInactivityInd |
| \return None |
| -----------------------------------------------------------------*/ |
| static void |
| __limIbssPeerInactivityHandler(tpAniSirGlobal pMac, |
| tpPESession psessionEntry, |
| tpSirIbssPeerInactivityInd peerInactivityInd) |
| { |
| if(psessionEntry->limMlmState != eLIM_MLM_BSS_STARTED_STATE) |
| { |
| limReactivateHeartBeatTimer(pMac, psessionEntry); |
| return; |
| } |
| |
| /* delete the peer for which heartbeat is observed */ |
| __limIbssSearchAndDeletePeer (pMac, psessionEntry, peerInactivityInd->peerAddr); |
| |
| } |
| |
| /** ------------------------------------------------------------- |
| \fn limProcessIbssPeerInactivity |
| \brief Peer inactivity message handler |
| \ |
| \param tpAniSirGlobal pMac |
| \param void* buf |
| \return None |
| -------------------------------------------------------------*/ |
| void |
| limProcessIbssPeerInactivity(tpAniSirGlobal pMac, void *buf) |
| { |
| /* |
| * --------------- HEARTBEAT OFFLOAD CASE ------------------ |
| * This message handler is executed when the firmware identifies |
| * inactivity from one or more peer devices. We will come here |
| * for every inactive peer device |
| */ |
| tANI_U8 i; |
| |
| tSirIbssPeerInactivityInd *peerInactivityInd = |
| (tSirIbssPeerInactivityInd *) buf; |
| |
| /* |
| * If IBSS is not started or heartbeat offload is not enabled |
| * we should not handle this request |
| */ |
| if (eLIM_STA_IN_IBSS_ROLE != pMac->lim.gLimSystemRole && |
| !IS_IBSS_HEARTBEAT_OFFLOAD_FEATURE_ENABLE) |
| { |
| return; |
| } |
| |
| /** If LinkMonitor is Disabled */ |
| if (!pMac->sys.gSysEnableLinkMonitorMode) |
| { |
| return; |
| } |
| |
| for (i = 0; i < pMac->lim.maxBssId; i++) |
| { |
| if (VOS_TRUE == pMac->lim.gpSession[i].valid && |
| eSIR_IBSS_MODE == pMac->lim.gpSession[i].bssType) |
| { |
| __limIbssPeerInactivityHandler(pMac, |
| &pMac->lim.gpSession[i], |
| peerInactivityInd); |
| break; |
| } |
| } |
| } |