Files
Linux_Drivers/osdrv/extdrv/wireless/mediatek/mt7603/ap/ap.c
forum_service 213c880673 add driver of tp、wiegand-gpio and wireless
Change-Id: Ie3c11d9d85cf1a05042f5690ac711856fe8b1ad7
2023-12-22 09:56:05 +08:00

3078 lines
89 KiB
C

/****************************************************************************
* Ralink Tech Inc.
* 4F, No. 2 Technology 5th Rd.
* Science-based Industrial Park
* Hsin-chu, Taiwan, R.O.C.
* (c) Copyright 2002, Ralink Technology, Inc.
*
* All rights reserved. Ralink's source code is an unpublished work and the
* use of a copyright notice does not imply otherwise. This source code
* contains confidential trade secret material of Ralink Tech. Any attemp
* or participation in deciphering, decoding, reverse engineering or in any
* way altering the source code is stricitly prohibited, unless the prior
* written consent of Ralink Technology, Inc. is obtained.
****************************************************************************
Module Name:
soft_ap.c
Abstract:
Access Point specific routines and MAC table maintenance routines
Revision History:
Who When What
-------- ---------- ----------------------------------------------
John Chang 08-04-2003 created for 11g soft-AP
*/
#include "rt_config.h"
char const *pEventText[EVENT_MAX_EVENT_TYPE] = {
"restart access point",
"successfully associated",
"has disassociated",
"has been aged-out and disassociated" ,
"active countermeasures",
"has disassociated with invalid PSK password"};
UCHAR get_apidx_by_addr(RTMP_ADAPTER *pAd, UCHAR *addr)
{
UCHAR apidx;
for (apidx=0; apidx<pAd->ApCfg.BssidNum; apidx++)
{
if (RTMPEqualMemory(addr, pAd->ApCfg.MBSSID[apidx].wdev.bssid, MAC_ADDR_LEN))
break;
}
return apidx;
}
// TODO: shiang-usw, need to revise this to asic specific functions!
INT set_wdev_if_addr(RTMP_ADAPTER *pAd, struct wifi_dev *wdev, INT opmode)
{
CHAR idx = wdev->func_idx;
if (opmode == OPMODE_AP)
{
COPY_MAC_ADDR(wdev->if_addr, pAd->CurrentAddress);
//+++Add by Carter for MT7603
#if defined(RTMP_MAC) || defined(RLT_MAC)
if (pAd->chipCap.hif_type == HIF_RTMP || pAd->chipCap.hif_type == HIF_RLT)
{
if (pAd->chipCap.MBSSIDMode >= MBSSID_MODE1)
{
UCHAR MacMask = 0;
if ((pAd->ApCfg.BssidNum + MAX_APCLI_NUM + MAX_MESH_NUM) <= 2)
MacMask = 0xFE;
else if ((pAd->ApCfg.BssidNum + MAX_APCLI_NUM + MAX_MESH_NUM) <= 4)
MacMask = 0xFC;
else if ((pAd->ApCfg.BssidNum + MAX_APCLI_NUM + MAX_MESH_NUM) <= 8)
MacMask = 0xF8;
else if ((pAd->ApCfg.BssidNum + MAX_APCLI_NUM + MAX_MESH_NUM) <= 16)
MacMask = 0xF0;
if (idx > 0)
{
wdev->if_addr[0] |= 0x2;
if (pAd->chipCap.MBSSIDMode == MBSSID_MODE1)
{
/*
Refer to HW definition -
Bit1 of MAC address Byte0 is local administration bit
and should be set to 1 in extended multiple BSSIDs'
Bit3~ of MAC address Byte0 is extended multiple BSSID index.
*/
#ifdef ENHANCE_NEW_MBSSID_MODE
wdev->if_addr[0] &= ((MacMask << 2) + 3);
#endif /* ENHANCE_NEW_MBSSID_MODE */
wdev->if_addr[0] += ((wdev->func_idx - 1) << 2);
}
#ifdef ENHANCE_NEW_MBSSID_MODE
else
{
wdev->if_addr[pAd->chipCap.MBSSIDMode - 1] &= (MacMask);
wdev->if_addr[pAd->chipCap.MBSSIDMode - 1] += (idx - 1);
}
#endif /* ENHANCE_NEW_MBSSID_MODE */
}
}
else
wdev->if_addr[5] += idx;
}
#endif /* defined(RTMP_MAC) || defined(RLT_MAC) */
#ifdef MT_MAC
if (pAd->chipCap.hif_type == HIF_MT)
{
//TODO: Carter, Apcli interface and MESH interface shall use HWBSSID1 or HWBSSID2???
UINT32 Value, MacByte = 0;
UCHAR MacMask = 0;
//TODO: shall we make choosing which byte to be selectable???
Value = 0x00000000;
RTMP_IO_READ32(pAd, LPON_BTEIR, &Value);//read BTEIR bit[31:29] for determine to choose which byte to extend BSSID mac address.
Value = Value | (0x2 << 29);//Note: Carter, make default will use byte4 bit[31:28] to extend Mac Address
RTMP_IO_WRITE32(pAd, LPON_BTEIR, Value);
MacByte = Value >> 29;
Value = 0x00000000;
RTMP_IO_READ32(pAd, RMAC_RMACDR, &Value);
Value = Value & 0xfcffffff;/* clear bit[25:24] */
if (pAd->ApCfg.BssidNum <= 2) {
Value &= ~RMACDR_MBSSID_MASK;
Value |= RMACDR_MBSSID(0x0);
MacMask = 0xef;
}
else if (pAd->ApCfg.BssidNum <= 4) {
Value &= ~RMACDR_MBSSID_MASK;
Value |= RMACDR_MBSSID(0x1);
MacMask = 0xcf;
}
else if (pAd->ApCfg.BssidNum <= 8) {
Value &= ~RMACDR_MBSSID_MASK;
Value |= RMACDR_MBSSID(0x2);
MacMask = 0x8f;
}
else if (pAd->ApCfg.BssidNum <= 16) {
Value &= ~RMACDR_MBSSID_MASK;
Value |= RMACDR_MBSSID(0x3);
MacMask = 0x0f;
}
else {
Value &= ~RMACDR_MBSSID_MASK;
Value |= RMACDR_MBSSID(0x3);
MacMask = 0x0f;
}
RTMP_IO_WRITE32(pAd, RMAC_RMACDR, Value);
if (idx > 0)
{
/* MT7603, bit1 in byte0 shall always be b'1 for Multiple BSSID */
wdev->if_addr[0] |= 0x2;
switch (MacByte) {
case 0x1: /* choose bit[23:20]*/
wdev->if_addr[2] = wdev->if_addr[2] & MacMask;//clear high 4 bits,
wdev->if_addr[2] = (wdev->if_addr[2] | (idx << 4));
break;
case 0x2: /* choose bit[31:28]*/
wdev->if_addr[3] = wdev->if_addr[3] & MacMask;//clear high 4 bits,
wdev->if_addr[3] = (wdev->if_addr[3] | (idx << 4));
break;
case 0x3: /* choose bit[39:36]*/
wdev->if_addr[4] = wdev->if_addr[4] & MacMask;//clear high 4 bits,
wdev->if_addr[4] = (wdev->if_addr[4] | (idx << 4));
break;
case 0x4: /* choose bit [47:44]*/
wdev->if_addr[5] = wdev->if_addr[5] & MacMask;//clear high 4 bits,
wdev->if_addr[5] = (wdev->if_addr[5] | (idx << 4));
break;
default: /* choose bit[15:12]*/
wdev->if_addr[1] = wdev->if_addr[1] & MacMask;//clear high 4 bits,
wdev->if_addr[1] = (wdev->if_addr[1] | (idx << 4));
break;
}
}
}
#endif /* MT_MAC */
//---Add by Carter for MT7603
}
return 0;
}
/*
==========================================================================
Description:
Initialize AP specific data especially the NDIS packet pool that's
used for wireless client bridging.
==========================================================================
*/
NDIS_STATUS APInitialize(RTMP_ADAPTER *pAd)
{
NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
INT i;
DBGPRINT(RT_DEBUG_TRACE, ("---> APInitialize\n"));
/* Init Group key update timer, and countermeasures timer */
for (i = 0; i < MAX_MBSSID_NUM(pAd); i++)
RTMPInitTimer(pAd, &pAd->ApCfg.MBSSID[i].REKEYTimer, GET_TIMER_FUNCTION(GREKEYPeriodicExec), pAd, TRUE);
RTMPInitTimer(pAd, &pAd->ApCfg.CounterMeasureTimer, GET_TIMER_FUNCTION(CMTimerExec), pAd, FALSE);
#ifndef BCN_OFFLOAD_SUPPORT
#ifdef RTMP_MAC_USB
RTMPInitTimer(pAd, &pAd->CommonCfg.BeaconUpdateTimer, GET_TIMER_FUNCTION(BeaconUpdateExec), pAd, TRUE);
#endif /* RTMP_MAC_USB */
#endif
#ifdef IDS_SUPPORT
/* Init intrusion detection timer */
RTMPInitTimer(pAd, &pAd->ApCfg.IDSTimer, GET_TIMER_FUNCTION(RTMPIdsPeriodicExec), pAd, FALSE);
pAd->ApCfg.IDSTimerRunning = FALSE;
#endif /* IDS_SUPPORT */
#ifdef WAPI_SUPPORT
/* Init WAPI rekey timer */
RTMPInitWapiRekeyTimerAction(pAd, NULL);
#endif /* WAPI_SUPPORT */
#ifdef IGMP_SNOOP_SUPPORT
MulticastFilterTableInit(pAd, &pAd->pMulticastFilterTable);
#endif /* IGMP_SNOOP_SUPPORT */
#ifdef DOT11V_WNM_SUPPORT
initList(&pAd->DMSEntryList);
#endif /* DOT11V_WNM_SUPPORT */
#ifdef DOT11K_RRM_SUPPORT
RRM_CfgInit(pAd);
#endif /* DOT11K_RRM_SUPPORT */
DBGPRINT(RT_DEBUG_TRACE, ("<--- APInitialize\n"));
return Status;
}
/*
==========================================================================
Description:
Shutdown AP and free AP specific resources
==========================================================================
*/
VOID APShutdown(RTMP_ADAPTER *pAd)
{
DBGPRINT(RT_DEBUG_TRACE, ("---> APShutdown\n"));
#ifdef RT65xx
if (!IS_MT76x0E(pAd) && !IS_MT76x2E(pAd))
#endif /* RT65xx */
MlmeRadioOff(pAd);
#ifdef RTMP_MAC_PCI
APStop(pAd);
#endif /* RTMP_MAC_PCI */
#ifdef IGMP_SNOOP_SUPPORT
MultiCastFilterTableReset(&pAd->pMulticastFilterTable);
#endif /* IGMP_SNOOP_SUPPORT */
#ifdef DOT11V_WNM_SUPPORT
DMSTable_Release(pAd);
#endif /* DOT11V_WNM_SUPPORT */
NdisFreeSpinLock(&pAd->MacTabLock);
#ifdef WDS_SUPPORT
NdisFreeSpinLock(&pAd->WdsTabLock);
#endif /* WDS_SUPPORT */
DBGPRINT(RT_DEBUG_TRACE, ("<--- APShutdown\n"));
}
#ifdef DOT11W_PMF_SUPPORT
static INT ap_pmf_init(RTMP_ADAPTER *pAd, BSS_STRUCT *pMbss, INT idx)
{
struct wifi_dev *wdev = &pMbss->wdev;
/*
IEEE 802.11W/P.10 -
A STA that has associated with Management Frame Protection enabled
shall not use pairwise cipher suite selectors WEP-40, WEP-104,
TKIP, or "Use Group cipher suite".
IEEE 802.11W/P.3 -
IEEE Std 802.11 provides one security protocol, CCMP, for protection
of unicast Robust Management frames.
*/
pMbss->PmfCfg.MFPC = FALSE;
pMbss->PmfCfg.MFPR = FALSE;
pMbss->PmfCfg.PMFSHA256 = FALSE;
if ((wdev->AuthMode == Ndis802_11AuthModeWPA2 ||
wdev->AuthMode == Ndis802_11AuthModeWPA2PSK
#ifdef WPA3_SUPPORT
|| wdev->AuthMode == Ndis802_11AuthModeWPA3SAE
#endif
) &&
(wdev->WepStatus == Ndis802_11AESEnable) &&
(pMbss->PmfCfg.Desired_MFPC))
{
pMbss->PmfCfg.MFPC = TRUE;
pMbss->PmfCfg.MFPR = pMbss->PmfCfg.Desired_MFPR;
/* IGTK default key index as 4 */
pMbss->PmfCfg.IGTK_KeyIdx = 4;
/* Derive IGTK */
PMF_DeriveIGTK(pAd, &pMbss->PmfCfg.IGTK[0][0]);
#ifdef MT_MAC
if ((pAd->chipCap.hif_type == HIF_MT) && (pAd->chipCap.FlgPMFEncrtptMode == PMF_ENCRYPT_MODE_2))
{
CIPHER_KEY CipherKey;
USHORT Wcid;
#if 0
GET_GroupKey_WCID(pAd, Wcid, pMbss->mbss_idx);
#else
GET_PMF_GroupKey_WCID(pAd, Wcid, pMbss->mbss_idx);
#endif
CipherKey.KeyLen = 16;
memcpy(CipherKey.Key, &pMbss->PmfCfg.IGTK[0][0], CipherKey.KeyLen);
CipherKey.CipherAlg = CIPHER_BIP;
CmdProcAddRemoveKey(pAd, 0, pMbss->mbss_idx, 0, Wcid, SHAREDKEYTABLE, &CipherKey, BROADCAST_ADDR);
}
#endif
if ((pMbss->PmfCfg.Desired_PMFSHA256) || (pMbss->PmfCfg.MFPR))
pMbss->PmfCfg.PMFSHA256 = TRUE;
} else if (pMbss->PmfCfg.Desired_MFPC) {
DBGPRINT(RT_DEBUG_ERROR, ("[PMF]%s:: Security is not WPA2/WPA2PSK AES\n", __FUNCTION__));
}
DBGPRINT(RT_DEBUG_ERROR, ("[PMF]%s:: apidx=%d, MFPC=%d, MFPR=%d, SHA256=%d\n",
__FUNCTION__, idx, pMbss->PmfCfg.MFPC,
pMbss->PmfCfg.MFPR, pMbss->PmfCfg.PMFSHA256));
return TRUE;
}
#endif /* DOT11W_PMF_SUPPORT */
static INT ap_security_init(RTMP_ADAPTER *pAd, struct wifi_dev *wdev, INT idx)
{
#ifdef DOT11W_PMF_SUPPORT
ap_pmf_init(pAd, &pAd->ApCfg.MBSSID[idx], idx);
#endif /* DOT11W_PMF_SUPPORT */
/* decide the mixed WPA cipher combination */
if (wdev->WepStatus == Ndis802_11TKIPAESMix)
{
switch ((UCHAR)wdev->AuthMode)
{
/* WPA mode */
case Ndis802_11AuthModeWPA:
case Ndis802_11AuthModeWPAPSK:
wdev->WpaMixPairCipher = WPA_TKIPAES_WPA2_NONE;
break;
/* WPA2 mode */
case Ndis802_11AuthModeWPA2:
case Ndis802_11AuthModeWPA2PSK:
#ifdef WPA3_SUPPORT
case Ndis802_11AuthModeWPA3SAE:
case Ndis802_11AuthModeWPA2PSKWPA3SAE:
#endif
wdev->WpaMixPairCipher = WPA_NONE_WPA2_TKIPAES;
break;
/* WPA and WPA2 both mode */
case Ndis802_11AuthModeWPA1WPA2:
case Ndis802_11AuthModeWPA1PSKWPA2PSK:
/* In WPA-WPA2 and TKIP-AES mixed mode, it shall use the maximum */
/* cipher capability unless users assign the desired setting. */
if (wdev->WpaMixPairCipher == MIX_CIPHER_NOTUSE ||
wdev->WpaMixPairCipher == WPA_TKIPAES_WPA2_NONE ||
wdev->WpaMixPairCipher == WPA_NONE_WPA2_TKIPAES)
wdev->WpaMixPairCipher = WPA_TKIPAES_WPA2_TKIPAES;
break;
}
}
else
wdev->WpaMixPairCipher = MIX_CIPHER_NOTUSE;
/* Generate the corresponding RSNIE */
RTMPMakeRSNIE(pAd, wdev->AuthMode, wdev->WepStatus, (UCHAR)idx);
return TRUE;
}
static INT ap_key_tb_init(RTMP_ADAPTER *pAd)
{
BSS_STRUCT *pMbss;
struct wifi_dev *wdev;
UCHAR Wcid, idx;
INT i;
/*
Initialize security variable per entry,
1. pairwise key table, re-set all WCID entry as NO-security mode.
2. access control port status
*/
/* Init Security variables */
for (idx = 0; idx < pAd->ApCfg.BssidNum; idx++)
{
pMbss = &pAd->ApCfg.MBSSID[idx];
wdev = &pAd->ApCfg.MBSSID[idx].wdev;
Wcid = 0;
wdev->PortSecured = WPA_802_1X_PORT_NOT_SECURED;
if (IS_WPA_CAPABILITY(wdev->AuthMode))
wdev->DefaultKeyId = 1;
/* Get a specific WCID to record this MBSS key attribute */
GET_GroupKey_WCID(pAd, Wcid, idx);
/* When WEP, TKIP or AES is enabled, set group key info to Asic */
if (wdev->WepStatus == Ndis802_11WEPEnabled)
{
UCHAR CipherAlg, key_idx;
for (key_idx=0; key_idx < SHARE_KEY_NUM; key_idx++)
{
CipherAlg = pAd->SharedKey[idx][key_idx].CipherAlg;
if (pAd->SharedKey[idx][key_idx].KeyLen > 0)
{
/* Set key material to Asic */
AsicAddSharedKeyEntry(pAd, idx, key_idx, &pAd->SharedKey[idx][key_idx]);
if (key_idx == wdev->DefaultKeyId)
{
/* Generate 3-bytes IV randomly for software encryption using */
for(i = 0; i < LEN_WEP_TSC; i++)
pAd->SharedKey[idx][key_idx].TxTsc[i] = RandomByte(pAd);
/* Update WCID attribute table and IVEIV table */
RTMPSetWcidSecurityInfo(pAd,
idx,
key_idx,
CipherAlg,
Wcid,
SHAREDKEYTABLE);
#ifdef MT_MAC
if (pAd->chipCap.hif_type == HIF_MT)
CmdProcAddRemoveKey(pAd, 0, idx, key_idx, Wcid, SHAREDKEYTABLE, &pAd->SharedKey[idx][key_idx], BROADCAST_ADDR);
#endif
}
}
}
}
else if ((wdev->WepStatus == Ndis802_11TKIPEnable) ||
(wdev->WepStatus == Ndis802_11AESEnable) ||
(wdev->WepStatus == Ndis802_11TKIPAESMix))
{
/* Generate GMK and GNonce randomly per MBSS */
GenRandom(pAd, wdev->bssid, pMbss->GMK);
GenRandom(pAd, wdev->bssid, pMbss->GNonce);
/* Derive GTK per BSSID */
WpaDeriveGTK(pMbss->GMK,
(UCHAR*)pMbss->GNonce,
wdev->bssid,
pMbss->GTK,
LEN_TKIP_GTK);
/* Install Shared key */
WPAInstallSharedKey(pAd,
(UINT8)wdev->GroupKeyWepStatus,
idx,
wdev->DefaultKeyId,
Wcid,
TRUE,
pMbss->GTK,
LEN_TKIP_GTK);
}
#ifdef WAPI_SUPPORT
else if (pMbss->wdev.WepStatus == Ndis802_11EncryptionSMS4Enabled)
{
INT cnt;
/* Initial the related variables */
pMbss->wdev.DefaultKeyId = 0;
NdisMoveMemory(pMbss->key_announce_flag, AE_BCAST_PN, LEN_WAPI_TSC);
if (IS_HW_WAPI_SUPPORT(pAd))
pMbss->sw_wpi_encrypt = FALSE;
else
pMbss->sw_wpi_encrypt = TRUE;
/* Generate NMK randomly */
for (cnt = 0; cnt < LEN_WAPI_NMK; cnt++)
pMbss->NMK[cnt] = RandomByte(pAd);
/* Count GTK for this BSSID */
RTMPDeriveWapiGTK(pMbss->NMK, pMbss->GTK);
/* Install Shared key */
WAPIInstallSharedKey(pAd,
wdev->GroupKeyWepStatus,
idx,
wdev->DefaultKeyId,
Wcid,
pMbss->GTK);
}
#endif /* WAPI_SUPPORT */
#ifdef DOT1X_SUPPORT
/* Send singal to daemon to indicate driver had restarted */
if ((wdev->AuthMode == Ndis802_11AuthModeWPA) || (wdev->AuthMode == Ndis802_11AuthModeWPA2)
|| (wdev->AuthMode == Ndis802_11AuthModeWPA1WPA2) || (wdev->IEEE8021X == TRUE))
{
;/*bDot1xReload = TRUE; */
}
#endif /* DOT1X_SUPPORT */
DBGPRINT(RT_DEBUG_TRACE, ("### BSS(%d) AuthMode(%d)=%s, WepStatus(%d)=%s, AccessControlList.Policy=%ld\n",
idx, wdev->AuthMode, GetAuthMode(wdev->AuthMode),
wdev->WepStatus, GetEncryptType(wdev->WepStatus),
pMbss->AccessControlList.Policy));
}
#if 0
/*
If We don't comment this code segment,
coverity will complain it due to bDot1xReload is always FALSE
*/
#ifdef DOT1X_SUPPORT
/* Send internal command to DOT1X daemon for reloading configuration */
if (bDot1xReload)
DOT1X_InternalCmdAction(pAd, NULL, DOT1X_RELOAD_CONFIG);
#endif /* DOT1X_SUPPORT */
#endif
return TRUE;
}
#ifdef RELEASE_EXCLUDE
/*@!Release
Reset whole WCID/Key table
WCID Table:
In AP mode, First WCID Table in ASIC will never be used.
To prevent it's 0xff-ff-ff-ff-ff-ff, Write 0 here.
p.s ASIC use all 0xff as termination of WCID table search.
Key Table:
*/
#endif /* RELEASE_EXCLUDE */
static INT ap_hw_tb_init(RTMP_ADAPTER *pAd)
{
INT i;
DBGPRINT(RT_DEBUG_TRACE, ("%s():Reset WCID Table\n", __FUNCTION__));
#ifdef MT7601U
if (IS_MT7601U(pAd))
{
UINT32 MACValue[128 * 2];
UINT32 Index;
for (Index = 0; Index < 128 * 2; Index+=2)
{
MACValue[Index] = 0;
MACValue[Index + 1] = 0;
}
if (pAd->chipOps.BurstWrite)
pAd->chipOps.BurstWrite(pAd, MAC_WCID_BASE, MACValue, 128 * 2);
}
else
#endif /* MT7601U */
AsicDelWcidTab(pAd, WCID_ALL);
DBGPRINT(RT_DEBUG_TRACE, ("%s():Reset Sec Table\n", __FUNCTION__));
for (i=0; i<MAX_LEN_OF_MAC_TABLE; i++)
{
pAd->MacTab.tr_entry[i].PortSecured = WPA_802_1X_PORT_NOT_SECURED;
AsicRemovePairwiseKeyEntry(pAd, (UCHAR)i);
}
return TRUE;
}
/*Nobody uses it currently*/
#if 0
static INT ap_dot11_proto_init(RTMP_ADAPTER *pAd)
{
return TRUE;
}
static INT ap_dot11_cap_init(RTMP_ADAPTER *pAd, struct wifi_dev *wdev, INT idx)
{
return TRUE;
}
#endif
static INT ap_phy_rrm_init(RTMP_ADAPTER *pAd)
{
#ifdef MT_MAC
if (pAd->chipCap.hif_type == HIF_MT)
AsicSetTxStream(pAd, pAd->Antenna.field.TxPath);
else
#endif /* MT_MAC */
ASIC_RLT_SET_TX_STREAM(pAd, OPMODE_AP, TRUE);
AsicSetRxStream(pAd, pAd->Antenna.field.RxPath);
pAd->CommonCfg.CentralChannel = pAd->CommonCfg.Channel;
// TODO: shiang-usw, get from MT7620_MT7610 Single driver, check this!!
N_ChannelCheck(pAd);//correct central channel offset
AsicBBPAdjust(pAd);
#ifdef DOT11_VHT_AC
if (pAd->CommonCfg.BBPCurrentBW == BW_80)
pAd->hw_cfg.cent_ch = pAd->CommonCfg.vht_cent_ch;
else
#endif /* DOT11_VHT_AC */
pAd->hw_cfg.cent_ch = pAd->CommonCfg.CentralChannel;
AsicSwitchChannel(pAd, pAd->hw_cfg.cent_ch, FALSE);
AsicLockChannel(pAd, pAd->hw_cfg.cent_ch);
#ifdef DOT11_VHT_AC
//+++Add by shiang for debug
DBGPRINT(RT_DEBUG_OFF, ("%s(): AP Set CentralFreq at %d(Prim=%d, HT-CentCh=%d, VHT-CentCh=%d, BBP_BW=%d)\n",
__FUNCTION__, pAd->hw_cfg.cent_ch, pAd->CommonCfg.Channel,
pAd->CommonCfg.CentralChannel, pAd->CommonCfg.vht_cent_ch,
pAd->CommonCfg.BBPCurrentBW));
//---Add by shiang for debug
#endif /* DOT11_VHT_AC */
return TRUE;
}
static INT ap_mlme_set_capability(RTMP_ADAPTER *pAd, BSS_STRUCT *pMbss)
{
struct wifi_dev *wdev = &pMbss->wdev;
BOOLEAN SpectrumMgmt = FALSE;
#ifdef A_BAND_SUPPORT
/* Decide the Capability information field */
/* In IEEE Std 802.1h-2003, the spectrum management bit is enabled in the 5 GHz band */
if ((pAd->CommonCfg.Channel > 14) && pAd->CommonCfg.bIEEE80211H == TRUE)
SpectrumMgmt = TRUE;
#endif /* A_BAND_SUPPORT */
pMbss->CapabilityInfo = CAP_GENERATE(1,
0,
(wdev->WepStatus != Ndis802_11EncryptionDisabled),
(pAd->CommonCfg.TxPreamble == Rt802_11PreambleLong ? 0 : 1),
pAd->CommonCfg.bUseShortSlotTime,
SpectrumMgmt);
#ifdef DOT11K_RRM_SUPPORT
if (pMbss->RrmCfg.bDot11kRRMEnable == TRUE)
pMbss->CapabilityInfo |= RRM_CAP_BIT;
#endif /* DOT11K_RRM_SUPPORT */
if (pMbss->wdev.bWmmCapable == TRUE)
{
#ifdef RELEASE_EXCLUDE
/* The IOT issue of Intel 4965 :
If we enable the Bit9(QoS) of "Capability Information field",
it would make the Intel 4965 AGN Chipset didn't send "QoS data" frame
to us even the Assoc Req/Resp have the WMM IE. */
#endif /* RELEASE_EXCLUDE */
/*
In WMM spec v1.1, A WMM-only AP or STA does not set the "QoS"
bit in the capability field of association, beacon and probe
management frames.
*/
/* pMbss->CapabilityInfo |= 0x0200; */
}
#ifdef UAPSD_SUPPORT
if (pMbss->wdev.UapsdInfo.bAPSDCapable == TRUE)
{
/*
QAPs set the APSD subfield to 1 within the Capability
Information field when the MIB attribute
dot11APSDOptionImplemented is true and set it to 0 otherwise.
STAs always set this subfield to 0.
*/
pMbss->CapabilityInfo |= 0x0800;
}
#endif /* UAPSD_SUPPORT */
return TRUE;
}
INT ap_func_init(RTMP_ADAPTER *pAd)
{
#ifdef RTMP_RBUS_SUPPORT
if (pAd->infType == RTMP_DEV_INF_RBUS)
{
#ifdef VIDEO_TURBINE_SUPPORT
VideoTurbineDynamicTune(pAd);
#endif /* VIDEO_TURBINE_SUPPORT */
#ifdef RT3XXX_ANTENNA_DIVERSITY_SUPPORT
RT3XXX_AntDiversity_Init(pAd);
#endif /* RT3XXX_ANTENNA_DIVERSITY_SUPPORT */
}
#endif /* RTMP_RBUS_SUPPORT */
#ifdef MAT_SUPPORT
MATEngineInit(pAd);
#endif /* MAT_SUPPORT */
#ifdef CLIENT_WDS
CliWds_ProxyTabInit(pAd);
#endif /* CLIENT_WDS */
return TRUE;
}
static void update_edca_param(RTMP_ADAPTER *pAd)
{
//TODO:
return ;
}
/*
==========================================================================
Description:
Start AP service. If any vital AP parameter is changed, a STOP-START
sequence is required to disassociate all STAs.
IRQL = DISPATCH_LEVEL.(from SetInformationHandler)
IRQL = PASSIVE_LEVEL. (from InitializeHandler)
Note:
Can't call NdisMIndicateStatus on this routine.
RT61 is a serialized driver on Win2KXP and is a deserialized on Win9X
Serialized callers of NdisMIndicateStatus must run at IRQL = DISPATCH_LEVEL.
==========================================================================
*/
VOID APStartUp(RTMP_ADAPTER *pAd)
{
#if defined(INF_AMAZON_SE) || defined(RTMP_MAC_USB)
UINT32 i;
#endif /* defined(INF_AMAZON_SE) || defined(RTMP_MAC_USB) */
UCHAR idx;
UCHAR phy_mode = pAd->CommonCfg.cfg_wmode;
BOOLEAN bWmmCapable = FALSE;
EDCA_PARM *edca_param;
DBGPRINT(RT_DEBUG_TRACE, ("===> APStartUp\n"));
#ifdef INF_AMAZON_SE
for (i=0;i<NUM_OF_TX_RING;i++)
pAd->BulkOutDataSizeLimit[i]=24576;
#endif /* INF_AMAZON_SE */
AsicDisableSync(pAd);
for (idx = 0; idx < pAd->ApCfg.BssidNum; idx++)
{
BSS_STRUCT *pMbss = &pAd->ApCfg.MBSSID[idx];
struct wifi_dev *wdev = &pMbss->wdev;
UCHAR tr_tb_idx = MAX_LEN_OF_MAC_TABLE + idx;
pMbss->mbss_idx = idx;
pMbss->bcn_buf.bcn_state = BCN_TX_IDLE;
#ifdef BCN_OFFLOAD_SUPPORT
pMbss->updateEventIsTriggered = FALSE;
#endif /* BCN_OFFLOAD_SUPPORT */
if ((pMbss->SsidLen <= 0) || (pMbss->SsidLen > MAX_LEN_OF_SSID))
{
NdisMoveMemory(pMbss->Ssid, "HT_AP", 5);
pMbss->Ssid[5] = '0' + idx;
pMbss->SsidLen = 6;
}
/*
re-copy the MAC to virtual interface to avoid these MAC = all zero,
when re-open the ra0,
i.e. ifconfig ra0 down, ifconfig ra0 up, ifconfig ra0 down, ifconfig up...
*/
wdev->func_idx = idx;
wdev->tr_tb_idx = tr_tb_idx;
//+++Add by Carter for MT7603
wdev->func_dev = (VOID *)pMbss;//FIXME: dirty code for prevent ra1 up crash.
//---Add by Carter for MT7603
set_wdev_if_addr(pAd, wdev, OPMODE_AP);
if (wdev->if_dev)
{
NdisMoveMemory(RTMP_OS_NETDEV_GET_PHYADDR(wdev->if_dev),
wdev->if_addr, MAC_ADDR_LEN);
}
COPY_MAC_ADDR(wdev->bssid, wdev->if_addr);
wdev_init(pAd, wdev, WDEV_TYPE_AP);
if (idx == 0)
mgmt_tb_set_mcast_entry(pAd, MCAST_WCID);
tr_tb_set_mcast_entry(pAd, tr_tb_idx, wdev);
COPY_MAC_ADDR(pAd->CommonCfg.Bssid, pAd->CurrentAddress);
ap_security_init(pAd, wdev, idx);
ap_mlme_set_capability(pAd, pMbss);
#ifdef WSC_V2_SUPPORT
if (pMbss->WscControl.WscV2Info.bEnableWpsV2)
{
/* WPS V2 doesn't support WEP and WPA/WPAPSK-TKIP. */
if ((wdev->WepStatus == Ndis802_11WEPEnabled) ||
(wdev->WepStatus == Ndis802_11TKIPEnable) ||
(pMbss->bHideSsid))
WscOnOff(pAd, idx, TRUE);
else
WscOnOff(pAd, idx, FALSE);
}
#endif /* WSC_V2_SUPPORT */
/* If any BSS is WMM Capable, we need to config HW CRs */
if (pMbss->wdev.bWmmCapable)
bWmmCapable = TRUE;
}
#ifdef DOT11_N_SUPPORT
if (phy_mode != pAd->CommonCfg.PhyMode)
RTMPSetPhyMode(pAd, phy_mode);
SetCommonHT(pAd);
#endif /* DOT11_N_SUPPORT */
if (WMODE_CAP_N(pAd->CommonCfg.PhyMode) || bWmmCapable)
{
/* EDCA parameters used for AP's own transmission */
if (pAd->CommonCfg.APEdcaParm.bValid == FALSE)
set_default_ap_edca_param(pAd);
/* EDCA parameters to be annouced in outgoing BEACON, used by WMM STA */
if (pAd->ApCfg.BssEdcaParm.bValid == FALSE)
set_default_sta_edca_param(pAd);
edca_param = &pAd->CommonCfg.APEdcaParm;
}
else
edca_param = NULL;
AsicSetEdcaParm(pAd, edca_param);
#ifdef DOT11_N_SUPPORT
if (!WMODE_CAP_N(pAd->CommonCfg.PhyMode))
pAd->CommonCfg.HtCapability.HtCapInfo.ChannelWidth = BW_20; /* Patch UI */
if (pAd->chipCap.hif_type == HIF_MT)
{
AsicSetRDG(pAd, pAd->CommonCfg.bRdg);
AsicWtblSetRDG(pAd, pAd->CommonCfg.bRdg);
if (pAd->CommonCfg.bRdg)
AsicUpdateTxOP(pAd, WMM_PARAM_AC_1, 0x80);
else
AsicUpdateTxOP(pAd, WMM_PARAM_AC_1, 0);
}
AsicSetRalinkBurstMode(pAd, pAd->CommonCfg.bRalinkBurstMode);
/*update edca depend on CommCfg*/
update_edca_param(pAd); // Currently do nothing inside...
#ifdef PIGGYBACK_SUPPORT
RTMPSetPiggyBack(pAd, pAd->CommonCfg.bPiggyBackCapable);
#endif /* PIGGYBACK_SUPPORT */
#endif /* DOT11_N_SUPPORT */
AsicSetBssid(pAd, pAd->CurrentAddress, 0x0);
ap_hw_tb_init(pAd);
#if defined(RTMP_MAC) || defined(RLT_MAC)
#ifdef FIFO_EXT_SUPPORT
if ((pAd->chipCap.hif_type == HIF_RTMP) || (pAd->chipCap.hif_type == HIF_RLT))
AsicFifoExtSet(pAd);
#endif /* FIFO_EXT_SUPPORT */
#endif /* defined(RTMP_MAC) || defined(RLT_MAC) */
ap_phy_rrm_init(pAd);
/* Clear BG-Protection flag */
OPSTATUS_CLEAR_FLAG(pAd, fOP_STATUS_BG_PROTECTION_INUSED);
#ifdef DOT11_N_SUPPORT
#ifdef GREENAP_SUPPORT
if (pAd->ApCfg.bGreenAPEnable == TRUE)
{
RTMP_CHIP_ENABLE_AP_MIMOPS(pAd,TRUE);
pAd->ApCfg.GreenAPLevel=GREENAP_WITHOUT_ANY_STAS_CONNECT;
}
#endif /* GREENAP_SUPPORT */
#endif /* DOT11_N_SUPPORT */
MlmeSetTxPreamble(pAd, (USHORT)pAd->CommonCfg.TxPreamble);
for (idx = 0; idx < pAd->ApCfg.BssidNum; idx++)
{
MlmeUpdateTxRates(pAd, FALSE, idx);
#ifdef DOT11_N_SUPPORT
if (WMODE_CAP_N(pAd->CommonCfg.PhyMode))
MlmeUpdateHtTxRates(pAd, idx);
#endif /* DOT11_N_SUPPORT */
}
/* Set the RadarDetect Mode as Normal, bc the APUpdateAllBeaconFram() will refer this parameter. */
pAd->Dot11_H.RDMode = RD_NORMAL_MODE;
/* Disable Protection first. */
AsicUpdateProtect(pAd, 0, (ALLN_SETPROTECT|CCKSETPROTECT|OFDMSETPROTECT), TRUE, FALSE);
APUpdateCapabilityAndErpIe(pAd);
#ifdef DOT11_N_SUPPORT
APUpdateOperationMode(pAd);
#endif /* DOT11_N_SUPPORT */
#ifdef LED_CONTROL_SUPPORT
RTMPSetLED(pAd, LED_LINK_UP);
#endif /* LED_CONTROL_SUPPORT */
ap_key_tb_init(pAd);
ApLogEvent(pAd, pAd->CurrentAddress, EVENT_RESET_ACCESS_POINT);
pAd->Mlme.PeriodicRound = 0;
pAd->Mlme.OneSecPeriodicRound = 0;
pAd->MacTab.MsduLifeTime = 5; /* default 5 seconds */
OPSTATUS_SET_FLAG(pAd, fOP_AP_STATUS_MEDIA_STATE_CONNECTED);
RTMP_IndicateMediaState(pAd, NdisMediaStateConnected);
#ifdef NINTENDO_AP
InitNINTENDO_TABLE(pAd);
#endif /* NINTENDO_AP */
/*
NOTE!!!:
All timer setting shall be set after following flag be cleared
fRTMP_ADAPTER_HALT_IN_PROGRESS
*/
RTMP_CLEAR_FLAG(pAd, fRTMP_ADAPTER_HALT_IN_PROGRESS);
RadarStateCheck(pAd);
#ifdef MT76x0
if (IS_MT76x0(pAd))
{
#ifdef MT76x0_TSSI_CAL_COMPENSATION
if (pAd->chipCap.bInternalTxALC)
{
UINT32 Value = 0;
CHAR target_power = 0;
UCHAR delta_power = 0;
RTMP_IO_READ32(pAd, TX_ALC_CFG_0, &Value);
Value = Value & (~0x3F3F);
if (pAd->hw_cfg.cent_ch > 14)
{
target_power = (pAd->chipCap.tssi_5G_target_power + (0 - pAd->chipCap.efuse_5G_54M_tx_power));
#ifdef DOT11_VHT_AC
if (pAd->CommonCfg.BBPCurrentBW == BW_80)
delta_power = pAd->chipCap.delta_tw_pwr_bw80;
else
#endif /* DOT11_VHT_AC */
delta_power = pAd->chipCap.delta_tw_pwr_bw40_5G;
}
else
{
target_power = (pAd->chipCap.tssi_2G_target_power + (0 - pAd->chipCap.efuse_2G_54M_tx_power));
delta_power = pAd->chipCap.delta_tw_pwr_bw40_2G;
}
DBGPRINT(RT_DEBUG_TRACE, ("%s: target_power = 0x%x, delta_power = 0x%x\n", __FUNCTION__, target_power, delta_power));
/*
EEPROM 0x50 - Power delta for 2.4G HT40
EEPROM 0x51 - Power delta for 5G HT40
EEPROM 0xD3 - Power delta for VHT80
Bit<7>: Enable/disable power delta of this BW
Bit<6>: 0: decrease power, 1: increase power
Bit<5:0>: Each step represents 0.5dB, range from 0 to 4
Increase or decrease 0x13b0<5:0> when bandwidth is changed
*/
if ((pAd->CommonCfg.BBPCurrentBW != BW_20)
&& (delta_power & 0x80))
{
if (delta_power & 0x40)
target_power += (delta_power & 0x3F);
else
target_power -= (delta_power & 0x3F);
}
Value |= target_power;
Value |= (0x2F2F << 16);
RTMP_IO_WRITE32(pAd, TX_ALC_CFG_0, Value);
DBGPRINT_RAW(RT_DEBUG_TRACE,("0x13b0: 0x%08x (%d)\n",
Value, (pAd->chipCap.tssi_5G_target_power + (0 - pAd->chipCap.efuse_5G_54M_tx_power))));
#ifdef SINGLE_SKU_V2
MT76x0_UpdateSkuPwr(pAd, pAd->hw_cfg.cent_ch);
#endif /* SINGLE_SKU_V2 */
}
#endif /* MT76x0_TSSI_CAL_COMPENSATION */
if (RTMP_TEST_FLAG(pAd, fRTMP_ADAPTER_START_UP))
mt76x0_calibration(pAd, pAd->hw_cfg.cent_ch, FALSE, TRUE, FALSE);
}
#endif /* MT76x0 */
#ifdef MT76x2
if (IS_MT76x2(pAd))
mt76x2_calibration(pAd, pAd->hw_cfg.cent_ch);
#endif
#ifdef RTMP_MAC_USB
RTUSBBssBeaconInit(pAd);
#endif /* RTMP_MAC_USB */
/* start sending BEACON out */
APMakeAllBssBeacon(pAd);
APUpdateAllBeaconFrame(pAd);
#ifdef DFS_SUPPORT
if (IS_DOT11_H_RADAR_STATE(pAd, RD_SILENCE_MODE))
NewRadarDetectionStart(pAd);
#endif /* DFS_SUPPORT */
#ifdef CARRIER_DETECTION_SUPPORT
if (pAd->CommonCfg.CarrierDetect.Enable == TRUE)
CarrierDetectionStart(pAd);
#endif /* CARRIER_DETECTION_SUPPORT */
if (pAd->Dot11_H.RDMode == RD_NORMAL_MODE)
AsicEnableBssSync(pAd, pAd->CommonCfg.BeaconPeriod);
/* Pre-tbtt interrupt setting. */
AsicSetPreTbtt(pAd, TRUE);
#ifdef WAPI_SUPPORT
RTMPStartWapiRekeyTimerAction(pAd, NULL);
#endif /* WAPI_SUPPORT */
/*
Set group re-key timer if necessary.
It must be processed after clear flag "fRTMP_ADAPTER_HALT_IN_PROGRESS"
*/
WPA_APSetGroupRekeyAction(pAd);
#ifdef WDS_SUPPORT
/* Prepare WEP key */
WdsPrepareWepKeyFromMainBss(pAd);
/* Add wds key infomation to ASIC */
AsicUpdateWdsRxWCIDTable(pAd);
#endif /* WDS_SUPPORT */
#ifdef IDS_SUPPORT
/* Start IDS timer */
if (pAd->ApCfg.IdsEnable)
{
#ifdef SYSTEM_LOG_SUPPORT
if (pAd->CommonCfg.bWirelessEvent == FALSE)
DBGPRINT(RT_DEBUG_WARN, ("!!! WARNING !!! The WirelessEvent parameter doesn't be enabled \n"));
#endif /* SYSTEM_LOG_SUPPORT */
RTMPIdsStart(pAd);
}
#endif /* IDS_SUPPORT */
#ifdef RTMP_MAC_USB
/* Support multi-BulkIn IRP, pAd->CommonCfg.NumOfBulkInIRP may > 1. */
for(i=0; i<pAd->CommonCfg.NumOfBulkInIRP; i++)
{
RTUSBBulkReceive(pAd);
DBGPRINT(RT_DEBUG_TRACE, ("RTUSBBulkReceive!\n" ));
}
/* Enable RX */
AsicSetMacTxRx(pAd, ASIC_MAC_RX, TRUE);
#endif /* RTMP_MAC_USB */
#ifdef MESH_SUPPORT
if (MESH_ON(pAd))
MeshUp(pAd);
#endif /* MESH_SUPPORT */
#ifdef DOT11R_FT_SUPPORT
FT_Init(pAd);
#endif /* DOT11R_FT_SUPPORT */
#ifdef SMART_ANTENNA
RtmpSAStart(pAd);
#endif /* SMART_ANTENNA */
#ifdef MT7601
if (IS_MT7601(pAd))
{
#ifdef CONFIG_ANDES_SUPPORT
#ifdef DPD_CALIBRATION_SUPPORT
CHIP_CALIBRATION(pAd, DPD_CALIBRATION, pAd->chipCap.CurrentTemperature);
#endif /* DPD_CALIBRATION_SUPPORT */
#endif /* CONFIG_ANDES_SUPPORT */
MT7601_RXDC_CAL(pAd);
}
#endif /* MT7601 */
#ifdef CONFIG_MAC_PCI
RTMP_ASIC_INTERRUPT_ENABLE(pAd);
#endif
DBGPRINT(RT_DEBUG_OFF, ("Main bssid = %02x:%02x:%02x:%02x:%02x:%02x\n",
PRINT_MAC(pAd->ApCfg.MBSSID[BSS0].wdev.bssid)));
DBGPRINT(RT_DEBUG_TRACE, ("<=== APStartUp\n"));
}
/*
==========================================================================
Description:
disassociate all STAs and stop AP service.
Note:
==========================================================================
*/
VOID APStop(RTMP_ADAPTER *pAd)
{
BOOLEAN Cancelled;
INT idx;
BSS_STRUCT *pMbss;
DBGPRINT(RT_DEBUG_TRACE, ("!!! APStop !!!\n"));
#ifdef CONFIG_MAC_PCI
RTMP_ASIC_INTERRUPT_DISABLE(pAd);
#endif
#ifdef DFS_SUPPORT
NewRadarDetectionStop(pAd);
#endif /* DFS_SUPPORT */
#ifdef CONFIG_AP_SUPPORT
#ifdef CARRIER_DETECTION_SUPPORT
if (pAd->CommonCfg.CarrierDetect.Enable == TRUE)
{
/* make sure CarrierDetect wont send CTS */
CarrierDetectionStop(pAd);
}
#endif /* CARRIER_DETECTION_SUPPORT */
#endif /* CONFIG_AP_SUPPORT */
#ifdef MESH_SUPPORT
if (MESH_ON(pAd))
MeshDown(pAd, TRUE);
#endif /* MESH_SUPPORT */
#ifdef WDS_SUPPORT
WdsDown(pAd);
#endif /* WDS_SUPPORT */
#ifdef APCLI_SUPPORT
ApCliIfDown(pAd);
#endif /* APCLI_SUPPORT */
MacTableReset(pAd, 1);
RTMP_SET_FLAG(pAd, fRTMP_ADAPTER_HALT_IN_PROGRESS);
/* Disable pre-tbtt interrupt */
AsicSetPreTbtt(pAd, FALSE);
/* Disable piggyback */
RTMPSetPiggyBack(pAd, FALSE);
AsicUpdateProtect(pAd, 0, (ALLN_SETPROTECT|CCKSETPROTECT|OFDMSETPROTECT), TRUE, FALSE);
if (!RTMP_TEST_FLAG(pAd, fRTMP_ADAPTER_NIC_NOT_EXIST))
{
AsicDisableSync(pAd);
#ifdef LED_CONTROL_SUPPORT
/* Set LED */
RTMPSetLED(pAd, LED_LINK_DOWN);
#endif /* LED_CONTROL_SUPPORT */
}
#ifdef RTMP_MAC_USB
/* For RT2870, we need to clear the beacon sync buffer. */
RTUSBBssBeaconExit(pAd);
#endif /* RTMP_MAC_USB */
#ifdef NINTENDO_AP
NdisFreeSpinLock(&pAd->nindo_ctrl_block.NINTENDO_TABLE_Lock);
#endif /* NINTENDO_AP */
for (idx = 0; idx < MAX_MBSSID_NUM(pAd); idx++)
{
pMbss = &pAd->ApCfg.MBSSID[idx];
if (pMbss->REKEYTimerRunning == TRUE)
{
RTMPCancelTimer(&pMbss->REKEYTimer, &Cancelled);
pMbss->REKEYTimerRunning = FALSE;
}
pMbss->bcn_buf.bcn_state = BCN_TX_IDLE;
}
if (pAd->ApCfg.CMTimerRunning == TRUE)
{
RTMPCancelTimer(&pAd->ApCfg.CounterMeasureTimer, &Cancelled);
pAd->ApCfg.CMTimerRunning = FALSE;
}
#ifdef WAPI_SUPPORT
RTMPCancelWapiRekeyTimerAction(pAd, NULL);
#endif /* WAPI_SUPPORT */
/* */
/* Cancel the Timer, to make sure the timer was not queued. */
/* */
OPSTATUS_CLEAR_FLAG(pAd, fOP_AP_STATUS_MEDIA_STATE_CONNECTED);
RTMP_IndicateMediaState(pAd, NdisMediaStateDisconnected);
#ifdef IDS_SUPPORT
/* if necessary, cancel IDS timer */
RTMPIdsStop(pAd);
#endif /* IDS_SUPPORT */
#ifdef DOT11R_FT_SUPPORT
FT_Release(pAd);
#endif /* DOT11R_FT_SUPPORT */
#ifdef DOT11V_WNM_SUPPORT
DMSTable_Release(pAd);
#endif /* DOT11V_WNM_SUPPORT */
#ifdef SMART_ANTENNA
RtmpSAStop(pAd);
#endif /* SMART_ANTENNA */
#ifdef RTMP_MAC_USB
/* Disable RX */
AsicSetMacTxRx(pAd, ASIC_MAC_RX, FALSE);
/* Polling TX/RX path until packets empty */
MTUsbPollTxRxEmpty(pAd);
#endif /* RTMP_MAC_USB */
}
/*
==========================================================================
Description:
This routine is used to clean up a specified power-saving queue. It's
used whenever a wireless client is deleted.
==========================================================================
*/
VOID APCleanupPsQueue(RTMP_ADAPTER *pAd, QUEUE_HEADER *pQueue)
{
PQUEUE_ENTRY pEntry;
PNDIS_PACKET pPacket;
DBGPRINT(RT_DEBUG_TRACE, ("%s(): (0x%08lx)...\n", __FUNCTION__, (ULONG)pQueue));
while (pQueue->Head)
{
DBGPRINT(RT_DEBUG_TRACE, ("%s():%u...\n", __FUNCTION__, pQueue->Number));
pEntry = RemoveHeadQueue(pQueue);
/*pPacket = CONTAINING_RECORD(pEntry, NDIS_PACKET, MiniportReservedEx); */
pPacket = QUEUE_ENTRY_TO_PACKET(pEntry);
RELEASE_NDIS_PACKET(pAd, pPacket, NDIS_STATUS_FAILURE);
}
}
/*
==========================================================================
Description:
This routine is called by APMlmePeriodicExec() every second to check if
1. any associated client in PSM. If yes, then TX MCAST/BCAST should be
out in DTIM only
2. any client being idle for too long and should be aged-out from MAC table
3. garbage collect PSQ
==========================================================================
*/
VOID MacTableMaintenance(RTMP_ADAPTER *pAd)
{
int i, startWcid;
#ifdef DOT11_N_SUPPORT
ULONG MinimumAMPDUSize = pAd->CommonCfg.DesiredHtPhy.MaxRAmpduFactor; /*Default set minimum AMPDU Size to 2, i.e. 32K */
BOOLEAN bRdgActive;
BOOLEAN bRalinkBurstMode;
#endif /* DOT11_N_SUPPORT */
#ifdef RTMP_MAC_PCI
ULONG IrqFlags = 0;
#endif /* RTMP_MAC_PCI */
UINT fAnyStationPortSecured[HW_BEACON_MAX_NUM];
UINT bss_index;
MAC_TABLE *pMacTable;
UINT32 IdleTimeout;
#if defined(PRE_ANT_SWITCH) || defined(CFO_TRACK)
int lastClient=0;
#endif /* defined(PRE_ANT_SWITCH) || defined(CFO_TRACK) */
CHAR avgRssi;
BSS_STRUCT *pMbss;
#ifdef WFA_VHT_PF
RSSI_SAMPLE *worst_rssi = NULL;
int worst_rssi_sta_idx = 0;
#endif /* WFA_VHT_PF */
#ifdef MT_MAC
BOOLEAN bPreAnyStationInPsm = FALSE;
#endif /* MT_MAC */
#ifdef MT76x0
BOOLEAN bDisableSF = FALSE;
#endif /* MT76x0 */
NdisZeroMemory(fAnyStationPortSecured, sizeof(fAnyStationPortSecured));
pMacTable = &pAd->MacTab;
#ifdef MT_MAC
bPreAnyStationInPsm = pMacTable->fAnyStationInPsm;
#endif /* MT_MAC */
pMacTable->fAnyStationInPsm = FALSE;
pMacTable->fAnyStationBadAtheros = FALSE;
pMacTable->fAnyTxOPForceDisable = FALSE;
pMacTable->fAllStationAsRalink = TRUE;
#ifdef DOT11_N_SUPPORT
pMacTable->fAnyStationNonGF = FALSE;
pMacTable->fAnyStation20Only = FALSE;
pMacTable->fAnyStationIsLegacy = FALSE;
pMacTable->fAnyStationMIMOPSDynamic = FALSE;
#ifdef GREENAP_SUPPORT
/*Support Green AP */
pMacTable->fAnyStationIsHT=FALSE;
#endif /* GREENAP_SUPPORT */
#ifdef DOT11N_DRAFT3
pMacTable->fAnyStaFortyIntolerant = FALSE;
#endif /* DOT11N_DRAFT3 */
pMacTable->fAllStationGainGoodMCS = TRUE;
#endif /* DOT11_N_SUPPORT */
#ifdef WAPI_SUPPORT
pMacTable->fAnyWapiStation = FALSE;
#endif /* WAPI_SUPPORT */
startWcid = 1;
#ifdef RT_CFG80211_P2P_CONCURRENT_DEVICE
/* Skip the Infra Side */
startWcid = 2;
#endif /* RT_CFG80211_P2P_CONCURRENT_DEVICE */
#ifdef SMART_CARRIER_SENSE_SUPPORT
pAd->SCSCtrl.SCSMinRssi = 0; /* (Reset)The minimum RSSI of STA */
#endif /* SMART_CARRIER_SENSE_SUPPORT */
for (i = startWcid; i < MAX_LEN_OF_MAC_TABLE; i++)
{
MAC_TABLE_ENTRY *pEntry = &pMacTable->Content[i];
STA_TR_ENTRY *tr_entry = &pMacTable->tr_entry[i];
BOOLEAN bDisconnectSta = FALSE;
#ifdef APCLI_SUPPORT
if(IS_ENTRY_APCLI(pEntry) && (tr_entry->PortSecured == WPA_802_1X_PORT_SECURED))
{
#ifdef MAC_REPEATER_SUPPORT
if (pEntry->bReptCli)
{
pEntry->ReptCliIdleCount++;
if ((pEntry->bReptEthCli) && (pEntry->ReptCliIdleCount >= MAC_TABLE_AGEOUT_TIME))
{
MlmeEnqueue(pAd, APCLI_CTRL_STATE_MACHINE, APCLI_CTRL_DISCONNECT_REQ, 0, NULL,
(64 + (MAX_EXT_MAC_ADDR_SIZE * pEntry->func_tb_idx) + pEntry->MatchReptCliIdx));
RTMP_MLME_HANDLER(pAd);
RTMPRemoveRepeaterEntry(pAd, pEntry->func_tb_idx, pEntry->MatchReptCliIdx);
continue;
}
}
#endif /* MAC_REPEATER_SUPPORT */
#ifdef MT76x0
if (pEntry->RssiSample.AvgRssi0 > -62)
bDisableSF = TRUE;
#endif /* MT76x0 */
if ((pAd->Mlme.OneSecPeriodicRound % 10) == 8)
{
/* use Null or QoS Null to detect the ACTIVE station*/
BOOLEAN ApclibQosNull = FALSE;
if (CLIENT_STATUS_TEST_FLAG(pEntry, fCLIENT_STATUS_WMM_CAPABLE))
ApclibQosNull = TRUE;
ApCliRTMPSendNullFrame(pAd,pEntry->CurrTxRate, ApclibQosNull, pEntry, PWR_ACTIVE);
continue;
}
}
#endif /* APCLI_SUPPORT */
if (!IS_ENTRY_CLIENT(pEntry))
continue;
#ifdef MT_PS
CheckSkipTX(pAd, pEntry);
#endif /* MT_PS */
if (pEntry->NoDataIdleCount == 0)
pEntry->StationKeepAliveCount = 0;
pEntry->NoDataIdleCount ++;
// TODO: shiang-usw, remove upper setting becasue we need to migrate to tr_entry!
pAd->MacTab.tr_entry[pEntry->wcid].NoDataIdleCount = 0;
pEntry->StaConnectTime ++;
pMbss = &pAd->ApCfg.MBSSID[pEntry->func_tb_idx];
/* 0. STA failed to complete association should be removed to save MAC table space. */
if ((pEntry->Sst != SST_ASSOC) && (pEntry->NoDataIdleCount >= pEntry->AssocDeadLine))
{
DBGPRINT(RT_DEBUG_TRACE,
("%02x:%02x:%02x:%02x:%02x:%02x fail to complete ASSOC in %d sec\n",
PRINT_MAC(pEntry->Addr), MAC_TABLE_ASSOC_TIMEOUT));
#ifdef WSC_AP_SUPPORT
if (NdisEqualMemory(pEntry->Addr, pMbss->WscControl.EntryAddr, MAC_ADDR_LEN))
NdisZeroMemory(pMbss->WscControl.EntryAddr, MAC_ADDR_LEN);
#endif /* WSC_AP_SUPPORT */
MacTableDeleteEntry(pAd, pEntry->wcid, pEntry->Addr);
continue;
}
/*
1. check if there's any associated STA in power-save mode. this affects outgoing
MCAST/BCAST frames should be stored in PSQ till DtimCount=0
*/
if (pEntry->PsMode == PWR_SAVE) {
pMacTable->fAnyStationInPsm = TRUE;
if (pEntry->wdev && pEntry->wdev->wdev_type == WDEV_TYPE_AP) {
pAd->MacTab.tr_entry[pEntry->wdev->tr_tb_idx].PsMode = PWR_SAVE;
#ifdef RELEASE_EXCLUDE
/*
We may just receive a PS poll right after this function so that
it need to check PsDeQWaitCnt at least > 2 to make sure
we have enough time to dequeue a data pkt.
*/
#endif /* RELEASE_EXCLUDE */
if (tr_entry->PsDeQWaitCnt)
{
tr_entry->PsDeQWaitCnt++;
if (tr_entry->PsDeQWaitCnt > 2)
tr_entry->PsDeQWaitCnt = 0;
}
}
}
#ifdef DOT11_N_SUPPORT
if (pEntry->MmpsMode == MMPS_DYNAMIC)
pMacTable->fAnyStationMIMOPSDynamic = TRUE;
if (pEntry->MaxHTPhyMode.field.BW == BW_20)
pMacTable->fAnyStation20Only = TRUE;
if (pEntry->MaxHTPhyMode.field.MODE != MODE_HTGREENFIELD)
pMacTable->fAnyStationNonGF = TRUE;
if ((pEntry->MaxHTPhyMode.field.MODE == MODE_OFDM) || (pEntry->MaxHTPhyMode.field.MODE == MODE_CCK))
pMacTable->fAnyStationIsLegacy = TRUE;
#ifdef GREENAP_SUPPORT
else
pMacTable->fAnyStationIsHT=TRUE;
#endif /* GREENAP_SUPPORT */
#ifdef DOT11N_DRAFT3
if (pEntry->bForty_Mhz_Intolerant)
pMacTable->fAnyStaFortyIntolerant = TRUE;
#endif /* DOT11N_DRAFT3 */
/* Get minimum AMPDU size from STA */
if (MinimumAMPDUSize > pEntry->MaxRAmpduFactor)
MinimumAMPDUSize = pEntry->MaxRAmpduFactor;
#endif /* DOT11_N_SUPPORT */
if (pEntry->bIAmBadAtheros)
{
pMacTable->fAnyStationBadAtheros = TRUE;
#ifdef DOT11_N_SUPPORT
if (pAd->CommonCfg.IOTestParm.bRTSLongProtOn == FALSE)
AsicUpdateProtect(pAd, 8, ALLN_SETPROTECT, FALSE, pMacTable->fAnyStationNonGF);
#endif /* DOT11_N_SUPPORT */
}
/* detect the station alive status */
#ifdef WMM_ACM_SUPPORT
/* WMM ACM: QAP can send any packet to QSTA even no any TSPEC is built */
/* if (ACMP_IsAllACEnabled(pAd) == ACM_RTN_FAIL) */
#endif /* WMM_ACM_SUPPORT */
/* detect the station alive status */
if ((pMbss->StationKeepAliveTime > 0) &&
(pEntry->NoDataIdleCount >= pMbss->StationKeepAliveTime))
{
/*
If no any data success between ap and the station for
StationKeepAliveTime, try to detect whether the station is
still alive.
Note: Just only keepalive station function, no disassociation
function if too many no response.
*/
/*
For example as below:
1. Station in ACTIVE mode,
......
sam> tx ok!
sam> count = 1! ==> 1 second after the Null Frame is acked
sam> count = 2! ==> 2 second after the Null Frame is acked
sam> count = 3!
sam> count = 4!
sam> count = 5!
sam> count = 6!
sam> count = 7!
sam> count = 8!
sam> count = 9!
sam> count = 10!
sam> count = 11!
sam> count = 12!
sam> count = 13!
sam> count = 14!
sam> count = 15! ==> 15 second after the Null Frame is acked
sam> tx ok! ==> (KeepAlive Mechanism) send a Null Frame to
detect the STA life status
sam> count = 1! ==> 1 second after the Null Frame is acked
sam> count = 2!
sam> count = 3!
sam> count = 4!
......
If the station acknowledges the QoS Null Frame,
the NoDataIdleCount will be reset to 0.
2. Station in legacy PS mode,
We will set TIM bit after 15 seconds, the station will send a
PS-Poll frame and we will send a QoS Null frame to it.
If the station acknowledges the QoS Null Frame, the
NoDataIdleCount will be reset to 0.
3. Station in legacy UAPSD mode,
Currently we do not support the keep alive mechanism.
So if your station is in UAPSD mode, the station will be
kicked out after 300 seconds.
Note: the rate of QoS Null frame can not be 1M of 2.4GHz or
6M of 5GHz, or no any statistics count will occur.
*/
if (pEntry->StationKeepAliveCount++ == 0)
{
#ifdef P2P_SUPPORT
/* Modify for P2P test plan 6.1.12, enqueue null frame will influence the test item */
if (pAd->P2pCfg.bSigmaEnabled == FALSE)
{
#endif /* P2P_SUPPORT */
if (pEntry->PsMode == PWR_SAVE)
{
/* use TIM bit to detect the PS station */
WLAN_MR_TIM_BIT_SET(pAd, pEntry->func_tb_idx, pEntry->Aid);
}
else
{
/* use Null or QoS Null to detect the ACTIVE station */
BOOLEAN bQosNull = FALSE;
if (CLIENT_STATUS_TEST_FLAG(pEntry, fCLIENT_STATUS_WMM_CAPABLE))
bQosNull = TRUE;
RtmpEnqueueNullFrame(pAd, pEntry->Addr
, pEntry->CurrTxRate, (UCHAR)pEntry->Aid
, pEntry->func_tb_idx, bQosNull, TRUE, 0);
}
#ifdef P2P_SUPPORT
}
#endif /* P2P_SUPPORT */
}
else
{
if (pEntry->StationKeepAliveCount >= pMbss->StationKeepAliveTime)
pEntry->StationKeepAliveCount = 0;
}
}
/* 2. delete those MAC entry that has been idle for a long time */
#if 0
/*
PwrSaving STA means that a STA is in Power-Saving mode.
Continue counting TxFailCnt for PwrSaving STA for
TxRing Cleaning Action.
*/
if (pEntry->PsMode == PWR_SAVE)
{
pEntry->ContinueTxFailCnt = 0;
}
#endif
if (RTMP_CFG80211_VIF_P2P_GO_ON(pAd))
IdleTimeout = pAd->ApCfg.P2pGcIdleTimeout;
else
IdleTimeout = pAd->ApCfg.StaIdleTimeout;
if (pEntry->NoDataIdleCount >= IdleTimeout) {
bDisconnectSta = TRUE;
DBGPRINT(RT_DEBUG_WARN, ("ageout %02x:%02x:%02x:%02x:%02x:%02x after %d-sec silence\n",
PRINT_MAC(pEntry->Addr), pEntry->StaIdleTimeout));
ApLogEvent(pAd, pEntry->Addr, EVENT_AGED_OUT);
}
else if (pEntry->ContinueTxFailCnt >= pAd->ApCfg.EntryLifeCheck)
{
/*
AP have no way to know that the PwrSaving STA is leaving or not.
So do not disconnect for PwrSaving STA.
*/
if (pEntry->PsMode != PWR_SAVE)
{
bDisconnectSta = TRUE;
DBGPRINT(RT_DEBUG_WARN, ("STA-%02x:%02x:%02x:%02x:%02x:%02x had left (%d %lu)\n",
PRINT_MAC(pEntry->Addr),
pEntry->ContinueTxFailCnt, pAd->ApCfg.EntryLifeCheck));
}
}
if ((pMbss->RssiLowForStaKickOut != 0) &&
( (avgRssi=RTMPAvgRssi(pAd, &pEntry->RssiSample)) < pMbss->RssiLowForStaKickOut))
{
bDisconnectSta = TRUE;
DBGPRINT(RT_DEBUG_WARN, ("Disassoc STA %02x:%02x:%02x:%02x:%02x:%02x , RSSI Kickout Thres[%d]-[%d]\n",
PRINT_MAC(pEntry->Addr), pMbss->RssiLowForStaKickOut,
avgRssi));
}
#ifdef SMART_CARRIER_SENSE_SUPPORT
if (pAd->SCSCtrl.SCSEnable == SCS_ENABLE) {
CHAR tmpRssi = RTMPMinRssi(pAd, pEntry->RssiSample.AvgRssi[0],
pEntry->RssiSample.AvgRssi[1], pEntry->RssiSample.AvgRssi[2]);
if (tmpRssi < pAd->SCSCtrl.SCSMinRssi)
pAd->SCSCtrl.SCSMinRssi = tmpRssi;
}
#endif /* SMART_CARRIER_SENSE_SUPPORT */
if (bDisconnectSta)
{
/* send wireless event - for ageout */
RTMPSendWirelessEvent(pAd, IW_AGEOUT_EVENT_FLAG, pEntry->Addr, 0, 0);
if (pEntry->Sst == SST_ASSOC)
{
PUCHAR pOutBuffer = NULL;
NDIS_STATUS NStatus;
ULONG FrameLen = 0;
HEADER_802_11 DeAuthHdr;
USHORT Reason;
/* send out a DISASSOC request frame */
NStatus = MlmeAllocateMemory(pAd, &pOutBuffer);
if (NStatus != NDIS_STATUS_SUCCESS)
{
DBGPRINT(RT_DEBUG_TRACE, (" MlmeAllocateMemory fail ..\n"));
/*NdisReleaseSpinLock(&pAd->MacTabLock); */
continue;
}
#if 1
Reason = REASON_DEAUTH_STA_LEAVING;
DBGPRINT(RT_DEBUG_WARN, ("Send DEAUTH - Reason = %d frame TO %x %x %x %x %x %x \n",
Reason, PRINT_MAC(pEntry->Addr)));
MgtMacHeaderInit(pAd, &DeAuthHdr, SUBTYPE_DEAUTH, 0, pEntry->Addr,
pMbss->wdev.if_addr,
pMbss->wdev.bssid);
MakeOutgoingFrame(pOutBuffer, &FrameLen,
sizeof(HEADER_802_11), &DeAuthHdr,
2, &Reason,
END_OF_ARGS);
MiniportMMRequest(pAd, 0, pOutBuffer, FrameLen);
MlmeFreeMemory(pAd, pOutBuffer);
#else
Reason = REASON_DISASSOC_INACTIVE;
DBGPRINT(RT_DEBUG_ERROR, ("ASSOC - Send DISASSOC Reason = %d frame TO %x %x %x %x %x %x \n",Reason,
PRINT_MAC(pEntry->Addr)));
MgtMacHeaderInit(pAd, &DisassocHdr, SUBTYPE_DISASSOC, 0, pEntry->Addr, pMbss->Bssid);
MakeOutgoingFrame(pOutBuffer, &FrameLen, sizeof(HEADER_802_11), &DisassocHdr, 2, &Reason, END_OF_ARGS);
MiniportMMRequest(pAd, 0, pOutBuffer, FrameLen);
MlmeFreeMemory(pAd, pOutBuffer);
#endif
#ifdef MAC_REPEATER_SUPPORT
if ((pAd->ApCfg.bMACRepeaterEn == TRUE) && IS_ENTRY_CLIENT(pEntry))
{
UCHAR apCliIdx, CliIdx;
REPEATER_CLIENT_ENTRY *pReptEntry = NULL;
pReptEntry = RTMPLookupRepeaterCliEntry(pAd, TRUE, pEntry->Addr);
if (pReptEntry && (pReptEntry->CliConnectState != 0))
{
apCliIdx = pReptEntry->MatchApCliIdx;
CliIdx = pReptEntry->MatchLinkIdx;
MlmeEnqueue(pAd, APCLI_CTRL_STATE_MACHINE, APCLI_CTRL_DISCONNECT_REQ, 0, NULL,
(64 + MAX_EXT_MAC_ADDR_SIZE*apCliIdx + CliIdx));
RTMP_MLME_HANDLER(pAd);
RTMPRemoveRepeaterEntry(pAd, apCliIdx, CliIdx);
}
}
#endif /* MAC_REPEATER_SUPPORT */
}
MacTableDeleteEntry(pAd, pEntry->wcid, pEntry->Addr);
continue;
}
#ifdef CONFIG_HOTSPOT_R2
if (pEntry->BTMDisassocCount == 1)
{
PUCHAR pOutBuffer = NULL;
NDIS_STATUS NStatus;
ULONG FrameLen = 0;
HEADER_802_11 DisassocHdr;
USHORT Reason;
/* send out a DISASSOC request frame */
NStatus = MlmeAllocateMemory(pAd, &pOutBuffer);
if (NStatus != NDIS_STATUS_SUCCESS)
{
DBGPRINT(RT_DEBUG_TRACE, (" MlmeAllocateMemory fail ..\n"));
/*NdisReleaseSpinLock(&pAd->MacTabLock); */
continue;
}
Reason = REASON_DISASSOC_INACTIVE;
DBGPRINT(RT_DEBUG_ERROR, ("BTM ASSOC - Send DISASSOC Reason = %d frame TO %x %x %x %x %x %x \n",Reason,pEntry->Addr[0],
pEntry->Addr[1],pEntry->Addr[2],pEntry->Addr[3],pEntry->Addr[4],pEntry->Addr[5]));
MgtMacHeaderInit(pAd, &DisassocHdr, SUBTYPE_DISASSOC, 0, pEntry->Addr, pMbss->wdev.if_addr, pMbss->wdev.bssid);
MakeOutgoingFrame(pOutBuffer, &FrameLen, sizeof(HEADER_802_11), &DisassocHdr, 2, &Reason, END_OF_ARGS);
MiniportMMRequest(pAd, MGMT_USE_PS_FLAG, pOutBuffer, FrameLen);
MlmeFreeMemory(pAd, pOutBuffer);
//JERRY
if (!pEntry->IsKeep)
MacTableDeleteEntry(pAd, pEntry->Aid, pEntry->Addr);
continue;
}
if (pEntry->BTMDisassocCount != 0)
pEntry->BTMDisassocCount--;
#endif /* CONFIG_HOTSPOT_R2 */
/* 3. garbage collect the ps_queue if the STA has being idle for a while */
if ((pEntry->PsMode == PWR_SAVE) && (tr_entry->ps_state == APPS_RETRIEVE_DONE || tr_entry->ps_state == APPS_RETRIEVE_IDLE))
{
if (tr_entry->enqCount > 0)
{
tr_entry->PsQIdleCount++;
if (tr_entry->PsQIdleCount > 2)
{
rtmp_tx_swq_exit(pAd, pEntry->wcid);
tr_entry->PsQIdleCount = 0;
WLAN_MR_TIM_BIT_CLEAR(pAd, pEntry->func_tb_idx, pEntry->Aid);
DBGPRINT(RT_DEBUG_TRACE, ("%s():Clear WCID[%d] packets\n",__FUNCTION__, pEntry->wcid));
}
}
}
else
{
tr_entry->PsQIdleCount = 0;
}
#ifdef UAPSD_SUPPORT
UAPSD_QueueMaintenance(pAd, pEntry);
#endif /* UAPSD_SUPPORT */
/* check if this STA is Ralink-chipset */
if (!CLIENT_STATUS_TEST_FLAG(pEntry, fCLIENT_STATUS_RALINK_CHIPSET))
pMacTable->fAllStationAsRalink = FALSE;
/* Check if the port is secured */
if (tr_entry->PortSecured == WPA_802_1X_PORT_SECURED)
fAnyStationPortSecured[pEntry->func_tb_idx]++;
#ifdef DOT11_N_SUPPORT
#ifdef DOT11N_DRAFT3
if ((pEntry->BSS2040CoexistenceMgmtSupport)
&& (pAd->CommonCfg.Bss2040CoexistFlag & BSS_2040_COEXIST_INFO_NOTIFY)
&& (pAd->CommonCfg.bBssCoexEnable == TRUE)
)
{
SendNotifyBWActionFrame(pAd, pEntry->wcid, pEntry->func_tb_idx);
}
#endif /* DOT11N_DRAFT3 */
#endif /* DOT11_N_SUPPORT */
#ifdef WAPI_SUPPORT
if (pEntry->WepStatus == Ndis802_11EncryptionSMS4Enabled)
pMacTable->fAnyWapiStation = TRUE;
#endif /* WAPI_SUPPORT */
#if defined(PRE_ANT_SWITCH) || defined(CFO_TRACK)
lastClient = i;
#endif /* defined(PRE_ANT_SWITCH) || defined(CFO_TRACK) */
/* only apply burst when run in MCS0,1,8,9,16,17, not care about phymode */
if ((pEntry->HTPhyMode.field.MCS != 32) &&
((pEntry->HTPhyMode.field.MCS % 8 == 0) || (pEntry->HTPhyMode.field.MCS % 8 == 1)))
{
pMacTable->fAllStationGainGoodMCS = FALSE;
}
#ifdef WFA_VHT_PF
if (worst_rssi == NULL) {
worst_rssi = &pEntry->RssiSample;
worst_rssi_sta_idx = i;
} else {
if (worst_rssi->AvgRssi[0] > pEntry->RssiSample.AvgRssi[0]) {
worst_rssi = &pEntry->RssiSample;
worst_rssi_sta_idx = i;
}
}
#endif /* WFA_VHT_PF */
#ifdef MT76x0
if (pEntry->RssiSample.AvgRssi0 > -62)
bDisableSF = TRUE;
#endif /* MT76x0 */
}
#ifdef MT76x0
if (IS_MT76x0(pAd) && (pAd->CommonCfg.BBPCurrentBW == BW_80))
{
UINT32 reg_val = 0;
RTMP_IO_READ32(pAd, RXO_R18, &reg_val);
if (bDisableSF && ((reg_val & 0x1) == 0x1))
{
/*
Disable channel smoothing: 0x2948[0] = 0
*/
reg_val &= ~(0x1);
RTMP_IO_WRITE32(pAd, RXO_R18, reg_val);
}
else if ((!bDisableSF) && ((reg_val & 0x1) == 0x0))
{
/*
Enabled channel smoothing: 0x2948[0] = 1
*/
reg_val |= 1;
RTMP_IO_WRITE32(pAd, RXO_R18, reg_val);
}
/*
else
do nothing, we don't need to change RXO_R18
*/
}
#endif /* MT76x0 */
#ifdef MT_MAC
/* If we check that any preview stations are in Psm and no stations are in Psm now. */
/* AP will dequeue all buffer broadcast packets */
if ((pAd->chipCap.hif_type == HIF_MT) && (pMacTable->fAnyStationInPsm == FALSE)) {
UINT apidx = 0;
for (apidx = 0; apidx<pAd->ApCfg.BssidNum; apidx++)
{
BSS_STRUCT *pMbss;
UINT wcid = 0;
STA_TR_ENTRY *tr_entry = NULL;
pMbss = &pAd->ApCfg.MBSSID[apidx];
wcid = pMbss->wdev.tr_tb_idx;
tr_entry = &pAd->MacTab.tr_entry[wcid];
if ((bPreAnyStationInPsm == TRUE) && (tr_entry->tx_queue[QID_AC_BE].Head != NULL)) {
if (tr_entry->tx_queue[QID_AC_BE].Number > MAX_PACKETS_IN_MCAST_PS_QUEUE)
RTMPDeQueuePacket(pAd, FALSE, NUM_OF_TX_RING, wcid, MAX_PACKETS_IN_MCAST_PS_QUEUE);
else
RTMPDeQueuePacket(pAd, FALSE, NUM_OF_TX_RING, wcid, tr_entry->tx_queue[QID_AC_BE].Number);
}
}
}
#endif
#ifdef WFA_VHT_PF
if (worst_rssi != NULL &&
((pAd->Mlme.OneSecPeriodicRound % 10) == 5) &&
(worst_rssi_sta_idx >= 1))
{
CHAR gain = 2;
if (worst_rssi->AvgRssi[0] >= -40)
gain = 1;
else if (worst_rssi->AvgRssi[0] <= -50)
gain = 2;
rt85592_lna_gain_adjust(pAd, gain);
DBGPRINT(RT_DEBUG_TRACE, ("%s():WorstRSSI for STA(%02x:%02x:%02x:%02x:%02x:%02x):%d,%d,%d, Set Gain as %s\n",
__FUNCTION__,
PRINT_MAC(pMacTable->Content[worst_rssi_sta_idx].Addr),
worst_rssi->AvgRssi[0], worst_rssi->AvgRssi[1], worst_rssi->AvgRssi[2],
(gain == 2 ? "Mid" : "Low")));
}
#endif /* WFA_VHT_PF */
#ifdef PRE_ANT_SWITCH
#if defined (RT2883) || defined (RT3883)
if ((pMacTable->Size == 1) &&
(lastClient >= 1) && (lastClient < MAX_LEN_OF_MAC_TABLE)
#ifdef SMART_ANTENNA
&& !pAd->smartAntEnable
#endif /* SMART_ANTENNA */
)
rtmp_pre_ant_switch(pAd, &pMacTable->Content[lastClient]);
#endif /* defined (RT2883) || defined (RT3883) */
#endif /* PRE_ANT_SWITCH */
#ifdef CFO_TRACK
#ifdef RT3883
rtmp_cfo_track(pAd, &pMacTable->Content[lastClient], lastClient);
#endif /* RT3883 */
#endif /* CFO_TRACK */
/* Update the state of port per MBSS */
for (bss_index = BSS0; bss_index < MAX_MBSSID_NUM(pAd); bss_index++)
{
struct wifi_dev *wdev = &pAd->ApCfg.MBSSID[bss_index].wdev;
if ((fAnyStationPortSecured[bss_index] > 0)
#ifdef CONFIG_FPGA_MODE
|| (pAd->fpga_ctl.fpga_on & 0x1)
#endif /* CONFIG_FPGA_MODE */
)
{
wdev->PortSecured = WPA_802_1X_PORT_SECURED;
pAd->MacTab.tr_entry[wdev->tr_tb_idx].PortSecured = WPA_802_1X_PORT_SECURED;
}
else {
wdev->PortSecured = WPA_802_1X_PORT_NOT_SECURED;
pAd->MacTab.tr_entry[wdev->tr_tb_idx].PortSecured = WPA_802_1X_PORT_NOT_SECURED;
}
}
#ifdef DOT11_N_SUPPORT
#ifdef DOT11N_DRAFT3
if (pAd->CommonCfg.Bss2040CoexistFlag & BSS_2040_COEXIST_INFO_NOTIFY)
pAd->CommonCfg.Bss2040CoexistFlag &= (~BSS_2040_COEXIST_INFO_NOTIFY);
#endif /* DOT11N_DRAFT3 */
/* If all associated STAs are Ralink-chipset, AP shall enable RDG. */
if (pAd->CommonCfg.bRdg && pMacTable->fAllStationAsRalink)
bRdgActive = TRUE;
else
bRdgActive = FALSE;
if (pAd->CommonCfg.bRalinkBurstMode && pMacTable->fAllStationGainGoodMCS)
bRalinkBurstMode = TRUE;
else
bRalinkBurstMode = FALSE;
#ifdef DOT11_N_SUPPORT
#ifdef GREENAP_SUPPORT
if (WMODE_CAP_N(pAd->CommonCfg.PhyMode))
{
if(pAd->MacTab.fAnyStationIsHT == FALSE
&& pAd->ApCfg.bGreenAPEnable == TRUE)
{
#ifdef RTMP_RBUS_SUPPORT
#ifdef COC_SUPPORT
if ((pAd->MacTab.Size==0) &&
(pAd->ApCfg.GreenAPLevel != GREENAP_WITHOUT_ANY_STAS_CONNECT))
{
RTMP_CHIP_ENABLE_AP_MIMOPS(pAd,TRUE);
pAd->ApCfg.GreenAPLevel = GREENAP_WITHOUT_ANY_STAS_CONNECT;
}
else
#endif /* COC_SUPPORT */
#endif /* RTMP_RBUS_SUPPORT */
if (pAd->ApCfg.GreenAPLevel!=GREENAP_ONLY_11BG_STAS)
{
RTMP_CHIP_ENABLE_AP_MIMOPS(pAd,FALSE);
pAd->ApCfg.GreenAPLevel=GREENAP_ONLY_11BG_STAS;
}
}
else
{
if (pAd->ApCfg.GreenAPLevel!=GREENAP_11BGN_STAS)
{
RTMP_CHIP_DISABLE_AP_MIMOPS(pAd);
pAd->ApCfg.GreenAPLevel=GREENAP_11BGN_STAS;
}
}
}
#endif /* GREENAP_SUPPORT */
if (pAd->MacTab.Size > 3)
{
AsicSetRTSTxCntLimit(pAd, TRUE, 0x7);
} else {
AsicSetRTSTxCntLimit(pAd, TRUE, MT_RTS_RETRY);
}
if (bRdgActive != RTMP_TEST_FLAG(pAd, fRTMP_ADAPTER_RDG_ACTIVE))
{
AsicSetRDG(pAd, bRdgActive);
#ifdef MT_MAC
if (pAd->chipCap.hif_type == HIF_MT)
{
#if 0
if (bRdgActive)
{
AsicUpdateTxOP(pAd, WMM_PARAM_AC_1, 0x80);
}
else if ((pAd->MacTab.Size == 1)
&& (pAd->CommonCfg.bEnableTxBurst))
{
AsicUpdateTxOP(pAd, WMM_PARAM_AC_1, 0x80);
}
else
{
AsicUpdateTxOP(pAd, WMM_PARAM_AC_1, 0x0);
}
#endif
AsicWtblSetRDG(pAd, bRdgActive);
}
#endif /* MT_MAC */
}
if (bRalinkBurstMode != RTMP_TEST_FLAG(pAd, fRTMP_ADAPTER_RALINK_BURST_MODE))
AsicSetRalinkBurstMode(pAd, bRalinkBurstMode);
#endif /* DOT11_N_SUPPORT */
if ((pMacTable->fAnyStationBadAtheros == FALSE) && (pAd->CommonCfg.IOTestParm.bRTSLongProtOn == TRUE))
{
AsicUpdateProtect(pAd, pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode, ALLN_SETPROTECT, FALSE, pMacTable->fAnyStationNonGF);
}
#endif /* DOT11_N_SUPPORT */
#ifdef RTMP_MAC_PCI
RTMP_IRQ_LOCK(&pAd->irq_lock, IrqFlags);
#endif /* RTMP_MAC_PCI */
/*
4.
garbage collect pAd->MacTab.McastPsQueue if backlogged MCAST/BCAST frames
stale in queue. Since MCAST/BCAST frames always been sent out whenever
DtimCount==0, the only case to let them stale is surprise removal of the NIC,
so that ASIC-based Tbcn interrupt stops and DtimCount dead.
*/
// TODO: shiang-usw. revise this becasue now we have per-BSS McastPsQueue!
if (pMacTable->McastPsQueue.Head)
{
UINT bss_index;
pMacTable->PsQIdleCount ++;
if (pMacTable->PsQIdleCount > 1)
{
#ifdef RELEASE_EXCLUDE
/*
Normally, should not be here, because bc/mc packets will be
moved to SwQueue when DTIM = 0 and DTIM period < 2 seconds;
If enter here, it is the kernel bug or driver bug
*/
#endif /* RELEASE_EXCLUDE */
APCleanupPsQueue(pAd, &pMacTable->McastPsQueue);
pMacTable->PsQIdleCount = 0;
if (pAd->ApCfg.BssidNum > MAX_MBSSID_NUM(pAd))
pAd->ApCfg.BssidNum = MAX_MBSSID_NUM(pAd);
/* clear MCAST/BCAST backlog bit for all BSS */
for(bss_index=BSS0; bss_index<pAd->ApCfg.BssidNum; bss_index++)
WLAN_MR_TIM_BCMC_CLEAR(bss_index);
}
}
else
pMacTable->PsQIdleCount = 0;
#ifdef RTMP_MAC_PCI
RTMP_IRQ_UNLOCK(&pAd->irq_lock, IrqFlags);
#endif /* RTMP_MAC_PCI */
}
UINT32 MacTableAssocStaNumGet(RTMP_ADAPTER *pAd)
{
UINT32 num = 0;
UINT32 i;
for (i = 1; i < MAX_LEN_OF_MAC_TABLE; i++)
{
MAC_TABLE_ENTRY *pEntry = &pAd->MacTab.Content[i];
if (!IS_ENTRY_CLIENT(pEntry))
continue;
if (pEntry->Sst == SST_ASSOC)
num ++;
}
return num;
}
/*
==========================================================================
Description:
Look up a STA MAC table. Return its Sst to decide if an incoming
frame from this STA or an outgoing frame to this STA is permitted.
Return:
==========================================================================
*/
MAC_TABLE_ENTRY *APSsPsInquiry(
IN RTMP_ADAPTER *pAd,
IN UCHAR *pAddr,
OUT SST *Sst,
OUT USHORT *Aid,
OUT UCHAR *PsMode,
OUT UCHAR *Rate)
{
MAC_TABLE_ENTRY *pEntry = NULL;
if (MAC_ADDR_IS_GROUP(pAddr)) /* mcast & broadcast address */
{
*Sst = SST_ASSOC;
*Aid = MCAST_WCID; /* Softap supports 1 BSSID and use WCID=0 as multicast Wcid index */
*PsMode = PWR_ACTIVE;
*Rate = pAd->CommonCfg.MlmeRate;
}
else /* unicast address */
{
pEntry = MacTableLookup(pAd, pAddr);
if (pEntry)
{
*Sst = pEntry->Sst;
*Aid = pEntry->Aid;
*PsMode = pEntry->PsMode;
if ((pEntry->AuthMode >= Ndis802_11AuthModeWPA) && (pEntry->GTKState != REKEY_ESTABLISHED))
*Rate = pAd->CommonCfg.MlmeRate;
else
*Rate = pEntry->CurrTxRate;
}
else
{
*Sst = SST_NOT_AUTH;
*Aid = MCAST_WCID;
*PsMode = PWR_ACTIVE;
*Rate = pAd->CommonCfg.MlmeRate;
}
}
return pEntry;
}
#ifdef SYSTEM_LOG_SUPPORT
/*
==========================================================================
Description:
This routine is called to log a specific event into the event table.
The table is a QUERY-n-CLEAR array that stop at full.
==========================================================================
*/
VOID ApLogEvent(RTMP_ADAPTER *pAd, UCHAR *pAddr, USHORT Event)
{
if (pAd->EventTab.Num < MAX_NUM_OF_EVENT)
{
RT_802_11_EVENT_LOG *pLog = &pAd->EventTab.Log[pAd->EventTab.Num];
RTMP_GetCurrentSystemTime(&pLog->SystemTime);
COPY_MAC_ADDR(pLog->Addr, pAddr);
pLog->Event = Event;
DBGPRINT_RAW(RT_DEBUG_TRACE,("LOG#%ld %02x:%02x:%02x:%02x:%02x:%02x %s\n",
pAd->EventTab.Num, pAddr[0], pAddr[1], pAddr[2],
pAddr[3], pAddr[4], pAddr[5], pEventText[Event]));
pAd->EventTab.Num += 1;
}
}
#endif /* SYSTEM_LOG_SUPPORT */
#ifdef DOT11_N_SUPPORT
/*
==========================================================================
Description:
Operationg mode is as defined at 802.11n for how proteciton in this BSS operates.
Ap broadcast the operation mode at Additional HT Infroamtion Element Operating Mode fields.
802.11n D1.0 might has bugs so this operating mode use EWC MAC 1.24 definition first.
Called when receiving my bssid beacon or beaconAtJoin to update protection mode.
40MHz or 20MHz protection mode in HT 40/20 capabale BSS.
As STA, this obeys the operation mode in ADDHT IE.
As AP, update protection when setting ADDHT IE and after new STA joined.
==========================================================================
*/
VOID APUpdateOperationMode(RTMP_ADAPTER *pAd)
{
BOOLEAN bDisableBGProtect = FALSE, bNonGFExist = FALSE;
pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode = 0;
if ((pAd->ApCfg.LastNoneHTOLBCDetectTime + (5 * OS_HZ)) > pAd->Mlme.Now32) /* non HT BSS exist within 5 sec */
{
pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode = 1;
bDisableBGProtect = FALSE;
bNonGFExist = TRUE;
}
/* If I am 40MHz BSS, and there exist HT-20MHz station. */
/* Update to 2 when it's zero. Because OperaionMode = 1 or 3 has more protection. */
if ((pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode == 0) &&
(pAd->MacTab.fAnyStation20Only) &&
(pAd->CommonCfg.DesiredHtPhy.ChannelWidth == 1))
{
pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode = 2;
bDisableBGProtect = TRUE;
}
#ifdef RELEASE_EXCLUDE
/*
To improve hidden node problem,
we let STA use RTS/CTS to protect HT traffic, too.
*/
#endif
if (pAd->MacTab.fAnyStationIsLegacy || pAd->MacTab.Size > 1)
{
pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode = 3;
bDisableBGProtect = TRUE;
}
if (bNonGFExist == FALSE)
bNonGFExist = pAd->MacTab.fAnyStationNonGF;
AsicUpdateProtect(pAd,
pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode,
(ALLN_SETPROTECT),
bDisableBGProtect,
bNonGFExist);
pAd->CommonCfg.AddHTInfo.AddHtInfo2.NonGfPresent = pAd->MacTab.fAnyStationNonGF;
}
#endif /* DOT11_N_SUPPORT */
/*
==========================================================================
Description:
Check to see the exist of long preamble STA in associated list
==========================================================================
*/
BOOLEAN ApCheckLongPreambleSTA(RTMP_ADAPTER *pAd)
{
UCHAR i;
for (i=0; i<MAX_LEN_OF_MAC_TABLE; i++)
{
PMAC_TABLE_ENTRY pEntry = &pAd->MacTab.Content[i];
if (!IS_ENTRY_CLIENT(pEntry) || (pEntry->Sst != SST_ASSOC))
continue;
if (!CAP_IS_SHORT_PREAMBLE_ON(pEntry->CapabilityInfo))
{
#ifdef RELEASE_EXCLUDE
DBGPRINT(RT_DEBUG_INFO, ("Long preamble capable STA exist\n"));
#endif /* RELEASE_EXCLUDE */
return TRUE;
}
}
return FALSE;
}
/*
==========================================================================
Description:
Update ERP IE and CapabilityInfo based on STA association status.
The result will be auto updated into the next outgoing BEACON in next
TBTT interrupt service routine
==========================================================================
*/
VOID APUpdateCapabilityAndErpIe(RTMP_ADAPTER *pAd)
{
UCHAR i, ErpIeContent = 0;
BOOLEAN ShortSlotCapable = pAd->CommonCfg.bUseShortSlotTime;
UCHAR apidx;
BOOLEAN bUseBGProtection;
BOOLEAN LegacyBssExist;
if (WMODE_EQUAL(pAd->CommonCfg.PhyMode, WMODE_B))
return;
for (i=1; i<MAX_LEN_OF_MAC_TABLE; i++)
{
MAC_TABLE_ENTRY *pEntry = &pAd->MacTab.Content[i];
if (!IS_ENTRY_CLIENT(pEntry) || (pEntry->Sst != SST_ASSOC))
continue;
/* at least one 11b client associated, turn on ERP.NonERPPresent bit */
/* almost all 11b client won't support "Short Slot" time, turn off for maximum compatibility */
if (pEntry->MaxSupportedRate < RATE_FIRST_OFDM_RATE)
{
ShortSlotCapable = FALSE;
ErpIeContent |= 0x01;
}
/* at least one client can't support short slot */
if ((pEntry->CapabilityInfo & 0x0400) == 0)
ShortSlotCapable = FALSE;
}
/* legacy BSS exist within 5 sec */
if ((pAd->ApCfg.LastOLBCDetectTime + (5 * OS_HZ)) > pAd->Mlme.Now32)
LegacyBssExist = TRUE;
else
LegacyBssExist = FALSE;
/* decide ErpIR.UseProtection bit, depending on pAd->CommonCfg.UseBGProtection
AUTO (0): UseProtection = 1 if any 11b STA associated
ON (1): always USE protection
OFF (2): always NOT USE protection
*/
if (pAd->CommonCfg.UseBGProtection == 0)
{
ErpIeContent = (ErpIeContent)? 0x03 : 0x00;
/*if ((pAd->ApCfg.LastOLBCDetectTime + (5 * OS_HZ)) > pAd->Mlme.Now32) // legacy BSS exist within 5 sec */
if (LegacyBssExist)
{
#ifdef RELEASE_EXCLUDE
DBGPRINT(RT_DEBUG_INFO, ("APUpdateCapabilityAndErpIe - Legacy 802.11b BSS overlaped\n"));
#endif /* RELEASE_EXCLUDE */
ErpIeContent |= 0x02; /* set Use_Protection bit */
}
}
else if (pAd->CommonCfg.UseBGProtection == 1)
ErpIeContent |= 0x02;
bUseBGProtection = (pAd->CommonCfg.UseBGProtection == 1) || /* always use */
((pAd->CommonCfg.UseBGProtection == 0) && ERP_IS_USE_PROTECTION(ErpIeContent));
#ifdef A_BAND_SUPPORT
/* always no BG protection in A-band. falsely happened when switching A/G band to a dual-band AP */
if (pAd->CommonCfg.Channel > 14)
bUseBGProtection = FALSE;
#endif /* A_BAND_SUPPORT */
if (bUseBGProtection != OPSTATUS_TEST_FLAG(pAd, fOP_STATUS_BG_PROTECTION_INUSED))
{
USHORT OperationMode = 0;
BOOLEAN bNonGFExist = 0;
#ifdef DOT11_N_SUPPORT
OperationMode = pAd->CommonCfg.AddHTInfo.AddHtInfo2.OperaionMode;
bNonGFExist = pAd->MacTab.fAnyStationNonGF;
#endif /* DOT11_N_SUPPORT */
if (bUseBGProtection)
{
OPSTATUS_SET_FLAG(pAd, fOP_STATUS_BG_PROTECTION_INUSED);
AsicUpdateProtect(pAd, OperationMode, (OFDMSETPROTECT), FALSE, bNonGFExist);
}
else
{
OPSTATUS_CLEAR_FLAG(pAd, fOP_STATUS_BG_PROTECTION_INUSED);
AsicUpdateProtect(pAd, OperationMode, (OFDMSETPROTECT), TRUE, bNonGFExist);
}
#ifdef RELEASE_EXCLUDE
DBGPRINT(RT_DEBUG_INFO, ("SYNC - AP changed B/G protection to %d\n", bUseBGProtection));
#endif /* RELEASE_EXCLUDE */
}
/* Decide Barker Preamble bit of ERP IE */
if ((pAd->CommonCfg.TxPreamble == Rt802_11PreambleLong) || (ApCheckLongPreambleSTA(pAd) == TRUE))
pAd->ApCfg.ErpIeContent = (ErpIeContent | 0x04);
else
pAd->ApCfg.ErpIeContent = ErpIeContent;
#ifdef A_BAND_SUPPORT
/* Force to use ShortSlotTime at A-band */
if (pAd->CommonCfg.Channel > 14)
ShortSlotCapable = TRUE;
#endif /* A_BAND_SUPPORT */
/* deicide CapabilityInfo.ShortSlotTime bit */
for (apidx=0; apidx<pAd->ApCfg.BssidNum; apidx++)
{
USHORT *pCapInfo = &(pAd->ApCfg.MBSSID[apidx].CapabilityInfo);
/* In A-band, the ShortSlotTime bit should be ignored. */
if (ShortSlotCapable
#ifdef A_BAND_SUPPORT
&& (pAd->CommonCfg.Channel <= 14)
#endif /* A_BAND_SUPPORT */
)
(*pCapInfo) |= 0x0400;
else
(*pCapInfo) &= 0xfbff;
if (pAd->CommonCfg.TxPreamble == Rt802_11PreambleLong)
(*pCapInfo) &= (~0x020);
else
(*pCapInfo) |= 0x020;
#ifdef RELEASE_EXCLUDE
DBGPRINT(RT_DEBUG_INFO, ("APUpdateCapabilityAndErpIe - Capability= 0x%04x, ERP is 0x%02x\n",
*pCapInfo, ErpIeContent));
#endif /* RELEASE_EXCLUDE */
}
AsicSetSlotTime(pAd, ShortSlotCapable, pAd->CommonCfg.Channel);
}
/*
==========================================================================
Description:
Check if the specified STA pass the Access Control List checking.
If fails to pass the checking, then no authentication nor association
is allowed
Return:
MLME_SUCCESS - this STA passes ACL checking
==========================================================================
*/
BOOLEAN ApCheckAccessControlList(RTMP_ADAPTER *pAd, UCHAR *pAddr, UCHAR Apidx)
{
BOOLEAN Result = TRUE;
if (pAd->ApCfg.MBSSID[Apidx].AccessControlList.Policy == 0) /* ACL is disabled */
Result = TRUE;
else
{
ULONG i;
if (pAd->ApCfg.MBSSID[Apidx].AccessControlList.Policy == 1) /* ACL is a positive list */
Result = FALSE;
else /* ACL is a negative list */
Result = TRUE;
for (i=0; i<pAd->ApCfg.MBSSID[Apidx].AccessControlList.Num; i++)
{
if (MAC_ADDR_EQUAL(pAddr, pAd->ApCfg.MBSSID[Apidx].AccessControlList.Entry[i].Addr))
{
Result = !Result;
break;
}
}
}
if (Result == FALSE)
{
DBGPRINT(RT_DEBUG_TRACE, ("%02x:%02x:%02x:%02x:%02x:%02x failed in ACL checking\n",
PRINT_MAC(pAddr)));
}
return Result;
}
/*
==========================================================================
Description:
This routine update the current MAC table based on the current ACL.
If ACL change causing an associated STA become un-authorized. This STA
will be kicked out immediately.
==========================================================================
*/
VOID ApUpdateAccessControlList(RTMP_ADAPTER *pAd, UCHAR Apidx)
{
USHORT AclIdx, MacIdx;
BOOLEAN Matched;
PUCHAR pOutBuffer = NULL;
NDIS_STATUS NStatus;
ULONG FrameLen = 0;
HEADER_802_11 DisassocHdr;
USHORT Reason;
MAC_TABLE_ENTRY *pEntry;
BSS_STRUCT *pMbss;
BOOLEAN drop;
ASSERT(Apidx < MAX_MBSSID_NUM(pAd));
if (Apidx >= MAX_MBSSID_NUM(pAd))
return;
DBGPRINT(RT_DEBUG_TRACE, ("ApUpdateAccessControlList : Apidx = %d\n", Apidx));
/* ACL is disabled. Do nothing about the MAC table. */
pMbss = &pAd->ApCfg.MBSSID[Apidx];
if (pMbss->AccessControlList.Policy == 0)
return;
for (MacIdx=0; MacIdx < MAX_LEN_OF_MAC_TABLE; MacIdx++)
{
pEntry = &pAd->MacTab.Content[MacIdx];
if (!IS_ENTRY_CLIENT(pEntry))
continue;
/* We only need to update associations related to ACL of MBSSID[Apidx]. */
if (pEntry->func_tb_idx != Apidx)
continue;
drop = FALSE;
Matched = FALSE;
for (AclIdx = 0; AclIdx < pMbss->AccessControlList.Num; AclIdx++)
{
if (MAC_ADDR_EQUAL(&pEntry->Addr[0], pMbss->AccessControlList.Entry[AclIdx].Addr))
{
Matched = TRUE;
break;
}
}
if ((Matched == FALSE) && (pMbss->AccessControlList.Policy == 1))
{
drop = TRUE;
DBGPRINT(RT_DEBUG_TRACE, ("STA not on positive ACL. remove it...\n"));
}
else if ((Matched == TRUE) && (pMbss->AccessControlList.Policy == 2))
{
drop = TRUE;
DBGPRINT(RT_DEBUG_TRACE, ("STA on negative ACL. remove it...\n"));
}
if (drop == TRUE) {
DBGPRINT(RT_DEBUG_TRACE, ("Apidx = %d\n", Apidx));
DBGPRINT(RT_DEBUG_TRACE, ("pAd->ApCfg.MBSSID[%d].AccessControlList.Policy = %ld\n", Apidx,
pMbss->AccessControlList.Policy));
/* Before delete the entry from MacTable, send disassociation packet to client. */
if (pEntry->Sst == SST_ASSOC)
{
/* send out a DISASSOC frame */
NStatus = MlmeAllocateMemory(pAd, &pOutBuffer);
if (NStatus != NDIS_STATUS_SUCCESS)
{
DBGPRINT(RT_DEBUG_TRACE, (" MlmeAllocateMemory fail ..\n"));
return;
}
Reason = REASON_DECLINED;
DBGPRINT(RT_DEBUG_ERROR, ("ASSOC - Send DISASSOC Reason = %d frame TO %x %x %x %x %x %x \n",
Reason, PRINT_MAC(pEntry->Addr)));
MgtMacHeaderInit(pAd, &DisassocHdr, SUBTYPE_DISASSOC, 0,
pEntry->Addr,
pMbss->wdev.if_addr,
pMbss->wdev.bssid);
MakeOutgoingFrame(pOutBuffer, &FrameLen, sizeof(HEADER_802_11), &DisassocHdr, 2, &Reason, END_OF_ARGS);
MiniportMMRequest(pAd, 0, pOutBuffer, FrameLen);
MlmeFreeMemory(pAd, pOutBuffer);
RtmpusecDelay(5000);
}
MacTableDeleteEntry(pAd, pEntry->wcid, pEntry->Addr);
}
}
}
#ifdef DOT11_N_SUPPORT
#ifdef DOT11N_DRAFT3
/*
Depends on the 802.11n Draft 4.0, Before the HT AP start a BSS, it should scan some specific channels to
collect information of existing BSSs, then depens on the collected channel information, adjust the primary channel
and secondary channel setting.
For 5GHz,
Rule 1: If the AP chooses to start a 20/40 MHz BSS in 5GHz and that occupies the same two channels
as any existing 20/40 MHz BSSs, then the AP shall ensure that the primary channel of the
new BSS is identical to the primary channel of the existing 20/40 MHz BSSs and that the
secondary channel of the new 20/40 MHz BSS is identical to the secondary channel of the
existing 20/40 MHz BSSs, unless the AP discoverr that on those two channels are existing
20/40 MHz BSSs with different primary and secondary channels.
Rule 2: If the AP chooses to start a 20/40MHz BSS in 5GHz, the selected secondary channel should
correspond to a channel on which no beacons are detected during the overlapping BSS
scan time performed by the AP, unless there are beacons detected on both the selected
primary and secondary channels.
Rule 3: An HT AP should not start a 20 MHz BSS in 5GHz on a channel that is the secondary channel
of a 20/40 MHz BSS.
For 2.4GHz,
Rule 1: The AP shall not start a 20/40 MHz BSS in 2.4GHz if the value of the local variable "20/40
Operation Permitted" is FALSE.
20/40OperationPermitted = (P == OPi for all values of i) AND
(P == OTi for all values of i) AND
(S == OSi for all values if i)
where
P is the operating or intended primary channel of the 20/40 MHz BSS
S is the operating or intended secondary channel of the 20/40 MHz BSS
OPi is member i of the set of channels that are members of the channel set C and that are the
primary operating channel of at least one 20/40 MHz BSS that is detected within the AP's
BSA during the previous X seconds
OSi is member i of the set of channels that are members of the channel set C and that are the
secondary operating channel of at least one 20/40 MHz BSS that is detected within AP's
BSA during the previous X seconds
OTi is member i of the set of channels that comparises all channels that are members of the
channel set C that were listed once in the Channel List fields of 20/40 BSS Intolerant Channel
Report elements receved during the previous X seconds and all channels that are members
of the channel set C and that are the primary operating channel of at least one 20/40 MHz
BSS that were detected within the AP's BSA during the previous X seconds.
C is the set of all channels that are allowed operating channels within the current operational
regulatory domain and whose center frequency falls within the 40 MHz affected channel
range given by following equation:
Fp + Fs Fp + Fs
40MHz affected channel range = [ ------ - 25MHz, ------- + 25MHz ]
2 2
Where
Fp = the center frequency of channel P
Fs = the center frequency of channel S
"==" means that the values on either side of the "==" are to be tested for equaliy with a resulting
Boolean value.
=>When the value of OPi is the empty set, then the expression (P == OPi for all values of i)
is defined to be TRUE
=>When the value of OTi is the empty set, then the expression (P == OTi for all values of i)
is defined to be TRUE
=>When the value of OSi is the empty set, then the expression (S == OSi for all values of i)
is defined to be TRUE
*/
INT GetBssCoexEffectedChRange(
IN RTMP_ADAPTER *pAd,
IN BSS_COEX_CH_RANGE *pCoexChRange)
{
INT index, cntrCh = 0;
memset(pCoexChRange, 0, sizeof(BSS_COEX_CH_RANGE));
/* Build the effected channel list, if something wrong, return directly. */
#ifdef A_BAND_SUPPORT
if (pAd->CommonCfg.Channel > 14)
{ /* For 5GHz band */
for (index = 0; index < pAd->ChannelListNum; index++)
{
if(pAd->ChannelList[index].Channel == pAd->CommonCfg.Channel)
break;
}
if (index < pAd->ChannelListNum)
{
/* First get the primary channel */
pCoexChRange->primaryCh = pAd->ChannelList[index].Channel;
/* Now check about the secondary and central channel */
if(pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE)
{
pCoexChRange->effectChStart = pCoexChRange->primaryCh;
pCoexChRange->effectChEnd = pCoexChRange->primaryCh + 4;
pCoexChRange->secondaryCh = pCoexChRange->effectChEnd;
}
else
{
pCoexChRange->effectChStart = pCoexChRange->primaryCh -4;
pCoexChRange->effectChEnd = pCoexChRange->primaryCh;
pCoexChRange->secondaryCh = pCoexChRange->effectChStart;
}
DBGPRINT(RT_DEBUG_TRACE,("5.0GHz: Found CtrlCh idx(%d) from the ChList, ExtCh=%s, PriCh=[Idx:%d, CH:%d], SecCh=[Idx:%d, CH:%d], effected Ch=[CH:%d~CH:%d]!\n",
index,
((pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE) ? "ABOVE" : "BELOW"),
pCoexChRange->primaryCh, pAd->ChannelList[pCoexChRange->primaryCh].Channel,
pCoexChRange->secondaryCh, pAd->ChannelList[pCoexChRange->secondaryCh].Channel,
pAd->ChannelList[pCoexChRange->effectChStart].Channel,
pAd->ChannelList[pCoexChRange->effectChEnd].Channel));
return TRUE;
}
else
{
/* It should not happened! */
DBGPRINT(RT_DEBUG_ERROR, ("5GHz: Cannot found the CtrlCh(%d) in ChList, something wrong?\n",
pAd->CommonCfg.Channel));
}
}
else
#endif /* A_BAND_SUPPORT */
{ /* For 2.4GHz band */
for (index = 0; index < pAd->ChannelListNum; index++)
{
if(pAd->ChannelList[index].Channel == pAd->CommonCfg.Channel)
break;
}
if (index < pAd->ChannelListNum)
{
/* First get the primary channel */
pCoexChRange->primaryCh = index;
/* Now check about the secondary and central channel */
if(pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE)
{
if ((index + 4) < pAd->ChannelListNum)
{
cntrCh = index + 2;
pCoexChRange->secondaryCh = index + 4;
}
}
else
{
if ((index - 4) >=0)
{
cntrCh = index - 2;
pCoexChRange->secondaryCh = index - 4;
}
}
if (cntrCh)
{
pCoexChRange->effectChStart = (cntrCh - 5) > 0 ? (cntrCh - 5) : 0;
pCoexChRange->effectChEnd= (cntrCh + 5) > 0 ? (cntrCh + 5) : 0;
DBGPRINT(RT_DEBUG_TRACE,("2.4GHz: Found CtrlCh idx(%d) from the ChList, ExtCh=%s, PriCh=[Idx:%d, CH:%d], SecCh=[Idx:%d, CH:%d], effected Ch=[CH:%d~CH:%d]!\n",
index,
((pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE) ? "ABOVE" : "BELOW"),
pCoexChRange->primaryCh, pAd->ChannelList[pCoexChRange->primaryCh].Channel,
pCoexChRange->secondaryCh, pAd->ChannelList[pCoexChRange->secondaryCh].Channel,
pAd->ChannelList[pCoexChRange->effectChStart].Channel,
pAd->ChannelList[pCoexChRange->effectChEnd].Channel));
}
return TRUE;
}
/* It should not happened! */
DBGPRINT(RT_DEBUG_ERROR, ("2.4GHz: Didn't found valid channel range, Ch index=%d, ChListNum=%d, CtrlCh=%d\n",
index, pAd->ChannelListNum, pAd->CommonCfg.Channel));
}
return FALSE;
}
VOID APOverlappingBSSScan(RTMP_ADAPTER *pAd)
{
BOOLEAN needFallBack = FALSE;
UCHAR Channel = pAd->CommonCfg.Channel;
INT chStartIdx, chEndIdx, index,curPriChIdx, curSecChIdx;
/* We just care BSS who operating in 40MHz N Mode. */
if ((!WMODE_CAP_N(pAd->CommonCfg.PhyMode)) ||
(pAd->CommonCfg.RegTransmitSetting.field.BW == BW_20)
|| (pAd->CommonCfg.Channel > 14)
)
{
DBGPRINT(RT_DEBUG_TRACE, ("The pAd->PhyMode=%d, BW=%d, didn't need channel adjustment!\n",
pAd->CommonCfg.PhyMode, pAd->CommonCfg.RegTransmitSetting.field.BW));
return;
}
/* Build the effected channel list, if something wrong, return directly. */
#ifdef A_BAND_SUPPORT
if (pAd->CommonCfg.Channel > 14)
{ /* For 5GHz band */
for (index = 0; index < pAd->ChannelListNum; index++)
{
if(pAd->ChannelList[index].Channel == pAd->CommonCfg.Channel)
break;
}
if (index < pAd->ChannelListNum)
{
curPriChIdx = index;
if(pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE)
{
chStartIdx = index;
chEndIdx = chStartIdx + 1;
curSecChIdx = chEndIdx;
}
else
{
chStartIdx = index - 1;
chEndIdx = index;
curSecChIdx = chStartIdx;
}
}
else
{
/* It should not happened! */
DBGPRINT(RT_DEBUG_ERROR, ("5GHz: Cannot found the ControlChannel(%d) in ChannelList, something wrong?\n",
pAd->CommonCfg.Channel));
return;
}
}
else
#endif /* A_BAND_SUPPORT */
{ /* For 2.4GHz band */
for (index = 0; index < pAd->ChannelListNum; index++)
{
if(pAd->ChannelList[index].Channel == pAd->CommonCfg.Channel)
break;
}
if (index < pAd->ChannelListNum)
{
if(pAd->CommonCfg.RegTransmitSetting.field.EXTCHA == EXTCHA_ABOVE)
{
curPriChIdx = index;
curSecChIdx = ((index + 4) < pAd->ChannelListNum) ? (index + 4) : (pAd->ChannelListNum - 1);
chStartIdx = (curPriChIdx >= 3) ? (curPriChIdx - 3) : 0;
chEndIdx = ((curSecChIdx + 3) < pAd->ChannelListNum) ? (curSecChIdx + 3) : (pAd->ChannelListNum - 1);
}
else
{
curPriChIdx = index;
curSecChIdx = ((index - 4) >=0 ) ? (index - 4) : 0;
chStartIdx =(curSecChIdx >= 3) ? (curSecChIdx - 3) : 0;
chEndIdx = ((curPriChIdx + 3) < pAd->ChannelListNum) ? (curPriChIdx + 3) : (pAd->ChannelListNum - 1);;
}
}
else
{
/* It should not happened! */
DBGPRINT(RT_DEBUG_ERROR, ("2.4GHz: Cannot found the Control Channel(%d) in ChannelList, something wrong?\n",
pAd->CommonCfg.Channel));
return;
}
}
{
BSS_COEX_CH_RANGE coexChRange;
GetBssCoexEffectedChRange(pAd, &coexChRange);
}
/* Before we do the scanning, clear the bEffectedChannel as zero for latter use. */
for (index = 0; index < pAd->ChannelListNum; index++)
pAd->ChannelList[index].bEffectedChannel = 0;
pAd->CommonCfg.BssCoexApCnt = 0;
/* If we are not ready for Tx/Rx Pakcet, enable it now for receiving Beacons. */
if (RTMP_TEST_FLAG(pAd, fRTMP_ADAPTER_START_UP) == 0)
{
DBGPRINT(RT_DEBUG_TRACE, ("Card still not enable Tx/Rx, enable it now!\n"));
#ifdef RTMP_MAC_PCI
/* Enable Interrupt */
rtmp_irq_init(pAd);
RTMP_IRQ_ENABLE(pAd);
#endif /* RTMP_MAC_PCI */
#ifdef RTMP_MAC_USB
RTMP_CLEAR_FLAG(pAd, fRTMP_ADAPTER_RESET_IN_PROGRESS);
RTMP_CLEAR_FLAG(pAd, fRTMP_ADAPTER_REMOVE_IN_PROGRESS);
/*
Support multiple BulkIn IRP,
the value on pAd->CommonCfg.NumOfBulkInIRP may be large than 1.
*/
for(index=0; index<pAd->CommonCfg.NumOfBulkInIRP; index++)
{
RTUSBBulkReceive(pAd);
DBGPRINT(RT_DEBUG_TRACE, ("RTUSBBulkReceive!\n" ));
}
#endif /* RTMP_MAC_USB */
/* rtmp_rx_done_handle() API will check this flag to decide accept incoming packet or not. */
/* Set the flag be ready to receive Beacon frame for autochannel select. */
RTMP_SET_FLAG(pAd, fRTMP_ADAPTER_START_UP);
}
RTMPEnableRxTx(pAd);
DBGPRINT(RT_DEBUG_TRACE, ("Ready to do passive scanning for Channel[%d] to Channel[%d]!\n",
pAd->ChannelList[chStartIdx].Channel, pAd->ChannelList[chEndIdx].Channel));
/* Now start to do the passive scanning. */
pAd->CommonCfg.bOverlapScanning = TRUE;
for (index = chStartIdx; index<=chEndIdx; index++)
{
Channel = pAd->ChannelList[index].Channel;
AsicSetChannel(pAd, Channel, BW_20, EXTCHA_NONE, TRUE);
DBGPRINT(RT_DEBUG_ERROR, ("SYNC - BBP R4 to 20MHz.l\n"));
/*DBGPRINT(RT_DEBUG_TRACE, ("Passive scanning for Channel %d.....\n", Channel)); */
OS_WAIT(300); /* wait for 200 ms at each channel. */
}
pAd->CommonCfg.bOverlapScanning = FALSE;
/* After scan all relate channels, now check the scan result to find out if we need fallback to 20MHz. */
for (index = chStartIdx; index <= chEndIdx; index++)
{
DBGPRINT(RT_DEBUG_TRACE, ("Channel[Idx=%d, Ch=%d].bEffectedChannel=0x%x!\n",
index, pAd->ChannelList[index].Channel, pAd->ChannelList[index].bEffectedChannel));
if ((pAd->ChannelList[index].bEffectedChannel & (EFFECTED_CH_PRIMARY | EFFECTED_CH_LEGACY)) && (index != curPriChIdx) )
{
needFallBack = TRUE;
DBGPRINT(RT_DEBUG_TRACE, ("needFallBack=TRUE due to OP/OT!\n"));
}
if ((pAd->ChannelList[index].bEffectedChannel & EFFECTED_CH_SECONDARY) && (index != curSecChIdx))
{
needFallBack = TRUE;
DBGPRINT(RT_DEBUG_TRACE, ("needFallBack=TRUE due to OS!\n"));
}
}
/* If need fallback, now do it. */
if ((needFallBack == TRUE)
&& (pAd->CommonCfg.BssCoexApCnt > pAd->CommonCfg.BssCoexApCntThr)
)
{
pAd->CommonCfg.AddHTInfo.AddHtInfo.RecomWidth = 0;
pAd->CommonCfg.AddHTInfo.AddHtInfo.ExtChanOffset = 0;
pAd->CommonCfg.LastBSSCoexist2040.field.BSS20WidthReq = 1;
pAd->CommonCfg.Bss2040CoexistFlag |= BSS_2040_COEXIST_INFO_SYNC;
#if 1 //for 20/40 coex patch, need fallback
pAd->CommonCfg.Bss2040NeedFallBack = 1;
pAd->CommonCfg.RegTransmitSetting.field.EXTCHA = 0;
#endif
}
return;
}
#endif /* DOT11N_DRAFT3 */
#endif /* DOT11_N_SUPPORT */
#ifdef DOT1X_SUPPORT
/*
========================================================================
Routine Description:
Send Leyer 2 Frame to notify 802.1x daemon. This is a internal command
Arguments:
Return Value:
TRUE - send successfully
FAIL - send fail
Note:
========================================================================
*/
BOOLEAN DOT1X_InternalCmdAction(
IN PRTMP_ADAPTER pAd,
IN MAC_TABLE_ENTRY *pEntry,
IN UINT8 cmd)
{
// TODO: shiang-usw, fix me for pEntry->apidx to func_tb_idx
INT apidx = MAIN_MBSSID;
UCHAR RalinkIe[9] = {221, 7, 0x00, 0x0c, 0x43, 0x00, 0x00, 0x00, 0x00};
UCHAR s_addr[MAC_ADDR_LEN];
UCHAR EAPOL_IE[] = {0x88, 0x8e};
UINT8 frame_len = LENGTH_802_3 + sizeof(RalinkIe);
UCHAR FrameBuf[frame_len];
UINT8 offset = 0;
/* Init the frame buffer */
NdisZeroMemory(FrameBuf, frame_len);
if (pEntry)
{
apidx = pEntry->func_tb_idx;
NdisMoveMemory(s_addr, pEntry->Addr, MAC_ADDR_LEN);
}
else
{
/* Fake a Source Address for transmission */
NdisMoveMemory(s_addr, pAd->ApCfg.MBSSID[apidx].wdev.bssid, MAC_ADDR_LEN);
s_addr[0] |= 0x80;
}
/* Assign internal command for Ralink dot1x daemon */
RalinkIe[5] = cmd;
/* Prepare the 802.3 header */
MAKE_802_3_HEADER(FrameBuf,
pAd->ApCfg.MBSSID[apidx].wdev.bssid,
s_addr,
EAPOL_IE);
offset += LENGTH_802_3;
/* Prepare the specific header of internal command */
NdisMoveMemory(&FrameBuf[offset], RalinkIe, sizeof(RalinkIe));
/* Report to upper layer */
if (RTMP_L2_FRAME_TX_ACTION(pAd, apidx, FrameBuf, frame_len) == FALSE)
return FALSE;
DBGPRINT(RT_DEBUG_TRACE, ("%s done. (cmd=%d)\n", __FUNCTION__, cmd));
return TRUE;
}
/*
========================================================================
Routine Description:
Send Leyer 2 Frame to trigger 802.1x EAP state machine.
Arguments:
Return Value:
TRUE - send successfully
FAIL - send fail
Note:
========================================================================
*/
BOOLEAN DOT1X_EapTriggerAction(RTMP_ADAPTER *pAd, MAC_TABLE_ENTRY *pEntry)
{
// TODO: shiang-usw, fix me for pEntry->apidx to func_tb_idx
INT apidx = MAIN_MBSSID;
UCHAR eapol_start_1x_hdr[4] = {0x01, 0x01, 0x00, 0x00};
UINT8 frame_len = LENGTH_802_3 + sizeof(eapol_start_1x_hdr);
UCHAR FrameBuf[frame_len+32];
UINT8 offset = 0;
if((pEntry->AuthMode == Ndis802_11AuthModeWPA) || (pEntry->AuthMode == Ndis802_11AuthModeWPA2) || (pAd->ApCfg.MBSSID[apidx].wdev.IEEE8021X == TRUE))
{
/* Init the frame buffer */
NdisZeroMemory(FrameBuf, frame_len);
/* Assign apidx */
apidx = pEntry->func_tb_idx;
/* Prepare the 802.3 header */
MAKE_802_3_HEADER(FrameBuf, pAd->ApCfg.MBSSID[apidx].wdev.bssid, pEntry->Addr, EAPOL);
offset += LENGTH_802_3;
/* Prepare a fake eapol-start body */
NdisMoveMemory(&FrameBuf[offset], eapol_start_1x_hdr, sizeof(eapol_start_1x_hdr));
#ifdef CONFIG_HOTSPOT_R2
if (pEntry)
{
BSS_STRUCT *pMbss = pEntry->pMbss;
if ((pMbss->HotSpotCtrl.HotSpotEnable == 1) && (pMbss->wdev.AuthMode == Ndis802_11AuthModeWPA2) && (pEntry->hs_info.ppsmo_exist == 1))
{
UCHAR HS2_Header[4] = {0x50,0x6f,0x9a,0x12};
memcpy(&FrameBuf[offset+sizeof(eapol_start_1x_hdr)], HS2_Header, 4);
memcpy(&FrameBuf[offset+sizeof(eapol_start_1x_hdr)+4], &pEntry->hs_info, sizeof(struct _sta_hs_info));
frame_len += 4+sizeof(struct _sta_hs_info);
DBGPRINT(RT_DEBUG_OFF, ("event eapol start, %x:%x:%x:%x\n",
FrameBuf[offset+sizeof(eapol_start_1x_hdr)+4],
FrameBuf[offset+sizeof(eapol_start_1x_hdr)+5],
FrameBuf[offset+sizeof(eapol_start_1x_hdr)+6],
FrameBuf[offset+sizeof(eapol_start_1x_hdr)+7]));
}
}
#endif
/* Report to upper layer */
if (RTMP_L2_FRAME_TX_ACTION(pAd, apidx, FrameBuf, frame_len) == FALSE)
return FALSE;
DBGPRINT(RT_DEBUG_TRACE, ("Notify 8021.x daemon to trigger EAP-SM for this sta(%02x:%02x:%02x:%02x:%02x:%02x)\n", PRINT_MAC(pEntry->Addr)));
}
return TRUE;
}
#endif /* DOT1X_SUPPORT */