blob: 7c933c23b2353fdd722b7580b56ce4eb669e1c7d [file] [log] [blame]
/******************************************************************************
*
* This file is provided under a dual license. When you use or
* distribute this software, you may choose to be licensed under
* version 2 of the GNU General Public License ("GPLv2 License")
* or BSD License.
*
* GPLv2 License
*
* Copyright(C) 2016 MediaTek Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* See http://www.gnu.org/licenses/gpl-2.0.html for more details.
*
* BSD LICENSE
*
* Copyright(C) 2016 MediaTek Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of the copyright holder nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*****************************************************************************/
/*
** Id: //Department/DaVinci/BRANCHES/MT6620_WIFI_DRIVER_V2_3/mgmt/rlm_domain.c#2
*/
/*! \file "rlm_domain.c"
* \brief
*
*/
/*******************************************************************************
* C O M P I L E R F L A G S
********************************************************************************
*/
/*******************************************************************************
* E X T E R N A L R E F E R E N C E S
********************************************************************************
*/
#include "precomp.h"
#include "rlm_txpwr_init.h"
/*******************************************************************************
* C O N S T A N T S
********************************************************************************
*/
/*******************************************************************************
* D A T A T Y P E S
********************************************************************************
*/
/*******************************************************************************
* P U B L I C D A T A
********************************************************************************
*/
/* The following country or domain shall be set from host driver.
* And host driver should pass specified DOMAIN_INFO_ENTRY to MT6620 as
* the channel list of being a STA to do scanning/searching AP or being an
* AP to choose an adequate channel if auto-channel is set.
*/
/* Define mapping tables between country code and its channel set
*/
static const UINT_16 g_u2CountryGroup0[] = {
COUNTRY_CODE_AO, COUNTRY_CODE_BZ, COUNTRY_CODE_BJ, COUNTRY_CODE_BT,
COUNTRY_CODE_BO, COUNTRY_CODE_BI, COUNTRY_CODE_CM, COUNTRY_CODE_CF,
COUNTRY_CODE_TD, COUNTRY_CODE_KM, COUNTRY_CODE_CD, COUNTRY_CODE_CG,
COUNTRY_CODE_CI, COUNTRY_CODE_DJ, COUNTRY_CODE_GQ, COUNTRY_CODE_ER,
COUNTRY_CODE_FJ, COUNTRY_CODE_GA, COUNTRY_CODE_GM, COUNTRY_CODE_GN,
COUNTRY_CODE_GW, COUNTRY_CODE_RKS, COUNTRY_CODE_KG, COUNTRY_CODE_LY,
COUNTRY_CODE_MG, COUNTRY_CODE_ML, COUNTRY_CODE_NR, COUNTRY_CODE_NC,
COUNTRY_CODE_ST, COUNTRY_CODE_SC, COUNTRY_CODE_SL, COUNTRY_CODE_SB,
COUNTRY_CODE_SO, COUNTRY_CODE_SR, COUNTRY_CODE_SZ, COUNTRY_CODE_TJ,
COUNTRY_CODE_TG, COUNTRY_CODE_TO, COUNTRY_CODE_TM, COUNTRY_CODE_TV,
COUNTRY_CODE_VU, COUNTRY_CODE_YE
};
static const UINT_16 g_u2CountryGroup1[] = {
COUNTRY_CODE_AS, COUNTRY_CODE_AI, COUNTRY_CODE_BM, COUNTRY_CODE_CA,
COUNTRY_CODE_KY, COUNTRY_CODE_GU, COUNTRY_CODE_FM, COUNTRY_CODE_PR,
COUNTRY_CODE_US, COUNTRY_CODE_VI,
};
static const UINT_16 g_u2CountryGroup2[] = {
COUNTRY_CODE_AR, COUNTRY_CODE_AU, COUNTRY_CODE_AZ, COUNTRY_CODE_BW,
COUNTRY_CODE_KH, COUNTRY_CODE_CX, COUNTRY_CODE_CO, COUNTRY_CODE_CR,
COUNTRY_CODE_EC, COUNTRY_CODE_GD, COUNTRY_CODE_GT, COUNTRY_CODE_HK,
COUNTRY_CODE_KI, COUNTRY_CODE_LB, COUNTRY_CODE_LR, COUNTRY_CODE_MN,
COUNTRY_CODE_AN, COUNTRY_CODE_NZ, COUNTRY_CODE_NI, COUNTRY_CODE_PW,
COUNTRY_CODE_PY, COUNTRY_CODE_PE, COUNTRY_CODE_PH, COUNTRY_CODE_WS,
COUNTRY_CODE_SG, COUNTRY_CODE_LK, COUNTRY_CODE_TH, COUNTRY_CODE_TT,
COUNTRY_CODE_UY, COUNTRY_CODE_VN
};
static const UINT_16 g_u2CountryGroup3[] = {
COUNTRY_CODE_AW, COUNTRY_CODE_LA, COUNTRY_CODE_SA, COUNTRY_CODE_AE,
COUNTRY_CODE_UG
};
static const UINT_16 g_u2CountryGroup4[] = { COUNTRY_CODE_MM };
static const UINT_16 g_u2CountryGroup5[] = {
COUNTRY_CODE_AL, COUNTRY_CODE_DZ, COUNTRY_CODE_AD, COUNTRY_CODE_AT,
COUNTRY_CODE_BY, COUNTRY_CODE_BE, COUNTRY_CODE_BA, COUNTRY_CODE_VG,
COUNTRY_CODE_BG, COUNTRY_CODE_CV, COUNTRY_CODE_HR, COUNTRY_CODE_CY,
COUNTRY_CODE_CZ, COUNTRY_CODE_DK, COUNTRY_CODE_EE, COUNTRY_CODE_ET,
COUNTRY_CODE_FI, COUNTRY_CODE_FR, COUNTRY_CODE_GF, COUNTRY_CODE_PF,
COUNTRY_CODE_TF, COUNTRY_CODE_GE, COUNTRY_CODE_DE, COUNTRY_CODE_GH,
COUNTRY_CODE_GR, COUNTRY_CODE_GP, COUNTRY_CODE_HU, COUNTRY_CODE_IS,
COUNTRY_CODE_IQ, COUNTRY_CODE_IE, COUNTRY_CODE_IT, COUNTRY_CODE_KE,
COUNTRY_CODE_LV, COUNTRY_CODE_LS, COUNTRY_CODE_LI, COUNTRY_CODE_LT,
COUNTRY_CODE_LU, COUNTRY_CODE_MK, COUNTRY_CODE_MT, COUNTRY_CODE_MQ,
COUNTRY_CODE_MR, COUNTRY_CODE_MU, COUNTRY_CODE_YT, COUNTRY_CODE_MD,
COUNTRY_CODE_MC, COUNTRY_CODE_ME, COUNTRY_CODE_MS, COUNTRY_CODE_NL,
COUNTRY_CODE_NO, COUNTRY_CODE_OM, COUNTRY_CODE_PL, COUNTRY_CODE_PT,
COUNTRY_CODE_RE, COUNTRY_CODE_RO, COUNTRY_CODE_MF, COUNTRY_CODE_SM,
COUNTRY_CODE_SN, COUNTRY_CODE_RS, COUNTRY_CODE_SK, COUNTRY_CODE_SI,
COUNTRY_CODE_ZA, COUNTRY_CODE_ES, COUNTRY_CODE_SE, COUNTRY_CODE_CH,
COUNTRY_CODE_TR, COUNTRY_CODE_TC, COUNTRY_CODE_GB, COUNTRY_CODE_VA,
COUNTRY_CODE_EU
};
static const UINT_16 g_u2CountryGroup6[] = { COUNTRY_CODE_JP };
static const UINT_16 g_u2CountryGroup7[] = {
COUNTRY_CODE_AM, COUNTRY_CODE_IL, COUNTRY_CODE_KW, COUNTRY_CODE_MA,
COUNTRY_CODE_NE, COUNTRY_CODE_TN,
};
static const UINT_16 g_u2CountryGroup8[] = { COUNTRY_CODE_NP };
static const UINT_16 g_u2CountryGroup9[] = { COUNTRY_CODE_AF };
static const UINT_16 g_u2CountryGroup10[] = {
COUNTRY_CODE_AG, COUNTRY_CODE_BS, COUNTRY_CODE_BH, COUNTRY_CODE_BB,
COUNTRY_CODE_BN, COUNTRY_CODE_CL, COUNTRY_CODE_CN, COUNTRY_CODE_EG,
COUNTRY_CODE_SV, COUNTRY_CODE_IN, COUNTRY_CODE_MY, COUNTRY_CODE_MV,
COUNTRY_CODE_PA, COUNTRY_CODE_VE, COUNTRY_CODE_ZM,
};
static const UINT_16 g_u2CountryGroup11[] = { COUNTRY_CODE_JO, COUNTRY_CODE_PG };
static const UINT_16 g_u2CountryGroup12[] = {
COUNTRY_CODE_BF, COUNTRY_CODE_GY, COUNTRY_CODE_HT, COUNTRY_CODE_HN,
COUNTRY_CODE_JM, COUNTRY_CODE_MO, COUNTRY_CODE_MW, COUNTRY_CODE_PK,
COUNTRY_CODE_QA, COUNTRY_CODE_RW, COUNTRY_CODE_KN, COUNTRY_CODE_TZ,
};
static const UINT_16 g_u2CountryGroup13[] = { COUNTRY_CODE_ID };
static const UINT_16 g_u2CountryGroup14[] = { COUNTRY_CODE_KR };
static const UINT_16 g_u2CountryGroup15[] = { COUNTRY_CODE_NG };
static const UINT_16 g_u2CountryGroup16[] = {
COUNTRY_CODE_BD, COUNTRY_CODE_BR, COUNTRY_CODE_DM, COUNTRY_CODE_DO,
COUNTRY_CODE_FK, COUNTRY_CODE_KZ, COUNTRY_CODE_MX, COUNTRY_CODE_MZ,
COUNTRY_CODE_NA, COUNTRY_CODE_RU, COUNTRY_CODE_LC, COUNTRY_CODE_VC,
COUNTRY_CODE_UA, COUNTRY_CODE_UZ, COUNTRY_CODE_ZW
};
static const UINT_16 g_u2CountryGroup17[] = { COUNTRY_CODE_MP };
static const UINT_16 g_u2CountryGroup18[] = { COUNTRY_CODE_TW };
static const UINT_16 g_u2CountryGroup19[] = {
COUNTRY_CODE_CK, COUNTRY_CODE_CU, COUNTRY_CODE_TL, COUNTRY_CODE_FO,
COUNTRY_CODE_GI, COUNTRY_CODE_GG, COUNTRY_CODE_IR, COUNTRY_CODE_IM,
COUNTRY_CODE_JE, COUNTRY_CODE_KP, COUNTRY_CODE_MH, COUNTRY_CODE_NU,
COUNTRY_CODE_NF, COUNTRY_CODE_PS, COUNTRY_CODE_PN, COUNTRY_CODE_PM,
COUNTRY_CODE_SS, COUNTRY_CODE_SD, COUNTRY_CODE_SY
};
static const UINT_16 g_u2CountryGroup20[] = {
COUNTRY_CODE_DF
/* When country code is not found, this domain info will be used.
* So mark all country codes to reduce search time. 20110908
*/
};
#if (CFG_SUPPORT_SINGLE_SKU == 1)
mtk_regd_control g_mtk_regd_control = {
.en = FALSE,
.state = REGD_STATE_UNDEFINED
};
#if (CFG_SUPPORT_SINGLE_SKU_LOCAL_DB == 1)
const struct ieee80211_regdomain default_regdom_ww = {
.n_reg_rules = 4,
.alpha2 = "99",
.reg_rules = {
/* channels 1..13 */
REG_RULE_LIGHT(2412-10, 2472+10, 40, 0),
/* channels 14 */
REG_RULE_LIGHT(2484-10, 2484+10, 20, 0),
/* channel 36..64 */
REG_RULE_LIGHT(5150-10, 5350+10, 80, 0),
/* channel 100..165 */
REG_RULE_LIGHT(5470-10, 5850+10, 80, 0),
}
};
#endif
const char *gTx_Pwr_Limit_Section[TX_PWR_LIMIT_SECTION_NUM] = {
"legacy", "ht20", "ht40", "vht20", "offset"
};
const u8 gTx_Pwr_Limit_Element_Num[TX_PWR_LIMIT_SECTION_NUM] = {
7, 6, 7, 7, 5
};
const char *gTx_Pwr_Limit_Element[TX_PWR_LIMIT_SECTION_NUM][TX_PWR_LIMIT_ELEMENT_NUM] = {
{"cck1_2", "cck_5_11", "ofdm6_9", "ofdm12_18", "ofdm24_36", "ofdm48", "ofdm54"},
{"mcs0_8", "mcs1_2_9_10", "mcs3_4_11_12", "mcs5_13", "mcs6_14", "mcs7_15"},
{"mcs0_8", "mcs1_2_9_10", "mcs3_4_11_12", "mcs5_13", "mcs6_14", "mcs7_15", "mcs32"},
{"mcs0", "mcs1_2", "mcs3_4", "mcs5_6", "mcs7", "mcs8", "mcs9"},
{"lg40", "lg80", "vht40", "vht80", "vht160nc"}
};
static const INT_8 gTx_Pwr_Limit_2g_Ch[] = {
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14};
static const INT_8 gTx_Pwr_Limit_5g_Ch[] = {
36, 38, 40, 42, 44, 46, 48, 50, 52, 54, 56, 58, 60, 62, 64, 100, 102,
104, 106, 108, 110, 112, 114, 116, 118, 120, 122, 124, 126, 128, 132,
134, 136, 138, 140, 142, 144, 149, 151, 153, 155, 157, 159, 161, 165};
#define TX_PWR_LIMIT_2G_CH_NUM (ARRAY_SIZE(gTx_Pwr_Limit_2g_Ch))
#define TX_PWR_LIMIT_5G_CH_NUM (ARRAY_SIZE(gTx_Pwr_Limit_5g_Ch))
#endif
DOMAIN_INFO_ENTRY arSupportedRegDomains[] = {
{
(PUINT_16) g_u2CountryGroup0, sizeof(g_u2CountryGroup0) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_LOW_NA */
{118, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_MID_NA */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup1, sizeof(g_u2CountryGroup1) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 11, FALSE}
, /* CH_SET_2G4_1_11 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup2, sizeof(g_u2CountryGroup2) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup3, sizeof(g_u2CountryGroup3) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 4, FALSE}
, /* CH_SET_UNII_UPPER_149_161 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup4, sizeof(g_u2CountryGroup4) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup5, sizeof(g_u2CountryGroup5) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, TRUE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup6, sizeof(g_u2CountryGroup6) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{82, BAND_2G4, CHNL_SPAN_5, 14, 1, FALSE}
, /* CH_SET_2G4_14_14 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, TRUE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
}
}
,
{
(PUINT_16) g_u2CountryGroup7, sizeof(g_u2CountryGroup7) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup8, sizeof(g_u2CountryGroup8) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_5G, CHNL_SPAN_20, 149, 4, FALSE}
, /* CH_SET_UNII_UPPER_149_161 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup9, sizeof(g_u2CountryGroup9) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_MID_NA */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_UPPER_NA */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup10, sizeof(g_u2CountryGroup10) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup11, sizeof(g_u2CountryGroup11) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_MID_NA */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup12, sizeof(g_u2CountryGroup12) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_LOW_NA */
{118, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_MID_NA */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup13, sizeof(g_u2CountryGroup13) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_LOW_NA */
{118, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_MID_NA */
{121, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_WW_NA */
{125, BAND_5G, CHNL_SPAN_20, 149, 4, FALSE}
, /* CH_SET_UNII_UPPER_149_161 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup14, sizeof(g_u2CountryGroup14) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, FALSE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 8, FALSE}
, /* CH_SET_UNII_WW_100_128 */
{125, BAND_5G, CHNL_SPAN_20, 149, 4, FALSE}
, /* CH_SET_UNII_UPPER_149_161 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup15, sizeof(g_u2CountryGroup15) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_NULL, 0, 0, 0, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, TRUE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup16, sizeof(g_u2CountryGroup16) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, FALSE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, TRUE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup17, sizeof(g_u2CountryGroup17) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 11, FALSE}
, /* CH_SET_2G4_1_11 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, TRUE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup18, sizeof(g_u2CountryGroup18) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 11, FALSE}
, /* CH_SET_2G4_1_11 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, FALSE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, FALSE}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
(PUINT_16) g_u2CountryGroup19, sizeof(g_u2CountryGroup19) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 5, FALSE}
, /* CH_SET_UNII_UPPER_149_165 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
,
{
/* Note: The final one is for Europe union now.(Default group if no matched country code) */
(PUINT_16) g_u2CountryGroup20, sizeof(g_u2CountryGroup20) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 13, FALSE}
, /* CH_SET_2G4_1_13 */
{115, BAND_5G, CHNL_SPAN_20, 36, 4, FALSE}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, TRUE}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, TRUE}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 7, FALSE}
, /* CH_SET_UNII_UPPER_149_173 */
{0, BAND_NULL, 0, 0, 0, FALSE}
}
}
};
#define REG_DOMAIN_PASSIVE_DEF_IDX 1
static const UINT_16 g_u2CountryGroup0_Passive[] = {
COUNTRY_CODE_TW
};
DOMAIN_INFO_ENTRY arSupportedRegDomains_Passive[] = {
{
(PUINT_16) g_u2CountryGroup0_Passive, sizeof(g_u2CountryGroup0_Passive) / 2,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 0, 0}
, /* CH_SET_2G4_1_14_NA */
{82, BAND_2G4, CHNL_SPAN_5, 14, 0, 0}
,
{115, BAND_5G, CHNL_SPAN_20, 36, 4, 0}
, /* CH_SET_UNII_LOW_36_48 */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, 0}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 11, 0}
, /* CH_SET_UNII_WW_100_140 */
{125, BAND_5G, CHNL_SPAN_20, 149, 0, 0}
, /* CH_SET_UNII_UPPER_NA */
}
}
,
{
/* default passive channel table is empty */
COUNTRY_CODE_NULL, 0,
{
{81, BAND_2G4, CHNL_SPAN_5, 1, 0, 0}
, /* CH_SET_2G4_1_14_NA */
{82, BAND_2G4, CHNL_SPAN_5, 14, 0, 0}
,
{115, BAND_5G, CHNL_SPAN_20, 36, 0, 0}
, /* CH_SET_UNII_LOW_NA */
{118, BAND_5G, CHNL_SPAN_20, 52, 4, 0}
, /* CH_SET_UNII_MID_52_64 */
{121, BAND_5G, CHNL_SPAN_20, 100, 12, 0}
, /* CH_SET_UNII_WW_100_144 */
{125, BAND_5G, CHNL_SPAN_20, 149, 0, 0}
, /* CH_SET_UNII_UPPER_NA */
}
}
};
#define REG_DOMAIN_PASSIVE_GROUP_NUM \
(sizeof(arSupportedRegDomains_Passive) / sizeof(DOMAIN_INFO_ENTRY))
SUBBAND_CHANNEL_T g_rRlmSubBand[] = {
{BAND_2G4_LOWER_BOUND, BAND_2G4_UPPER_BOUND, 1, 0}
, /* 2.4G */
{UNII1_LOWER_BOUND, UNII1_UPPER_BOUND, 2, 0}
, /* ch36,38,40,..,48 */
{UNII2A_LOWER_BOUND, UNII2A_UPPER_BOUND, 2, 0}
, /* ch52,54,56,..,64 */
{UNII2C_LOWER_BOUND, UNII2C_UPPER_BOUND, 2, 0}
, /* ch100,102,104,...,144 */
{UNII3_LOWER_BOUND, UNII3_UPPER_BOUND, 2, 0} /* ch149,151,153,....,173 */
};
/*******************************************************************************
* P R I V A T E D A T A
********************************************************************************
*/
/*******************************************************************************
* M A C R O S
********************************************************************************
*/
/*******************************************************************************
* F U N C T I O N D E C L A R A T I O N S
********************************************************************************
*/
/*******************************************************************************
* F U N C T I O N S
********************************************************************************
*/
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in/out]
*
* \return none
*/
/*----------------------------------------------------------------------------*/
P_DOMAIN_INFO_ENTRY rlmDomainGetDomainInfo(P_ADAPTER_T prAdapter)
{
#define REG_DOMAIN_DEF_IDX 20 /* EU (Europe Union) */
#define REG_DOMAIN_GROUP_NUM \
(sizeof(arSupportedRegDomains) / sizeof(DOMAIN_INFO_ENTRY))
P_DOMAIN_INFO_ENTRY prDomainInfo;
P_REG_INFO_T prRegInfo;
UINT_16 u2TargetCountryCode;
UINT_16 i, j;
ASSERT(prAdapter);
if (prAdapter->prDomainInfo)
return prAdapter->prDomainInfo;
prRegInfo = &prAdapter->prGlueInfo->rRegInfo;
DBGLOG(RLM, TRACE, "eRegChannelListMap=%d, u2CountryCode=0x%04x\n",
prRegInfo->eRegChannelListMap,
prAdapter->rWifiVar.rConnSettings.u2CountryCode);
/*
* Domain info can be specified by given idx of arSupportedRegDomains table,
* customized, or searched by country code,
* only one is set among these three methods in NVRAM.
*/
if (prRegInfo->eRegChannelListMap == REG_CH_MAP_TBL_IDX &&
prRegInfo->ucRegChannelListIndex < REG_DOMAIN_GROUP_NUM) {
/* by given table idx */
DBGLOG(RLM, TRACE, "ucRegChannelListIndex=%d\n", prRegInfo->ucRegChannelListIndex);
prDomainInfo = &arSupportedRegDomains[prRegInfo->ucRegChannelListIndex];
} else if (prRegInfo->eRegChannelListMap == REG_CH_MAP_CUSTOMIZED) {
/* by customized */
prDomainInfo = &prRegInfo->rDomainInfo;
} else {
/* by country code */
u2TargetCountryCode = prAdapter->rWifiVar.rConnSettings.u2CountryCode;
for (i = 0; i < REG_DOMAIN_GROUP_NUM; i++) {
prDomainInfo = &arSupportedRegDomains[i];
if ((prDomainInfo->u4CountryNum && prDomainInfo->pu2CountryGroup) ||
prDomainInfo->u4CountryNum == 0) {
for (j = 0; j < prDomainInfo->u4CountryNum; j++) {
if (prDomainInfo->pu2CountryGroup[j] == u2TargetCountryCode)
break;
}
if (j < prDomainInfo->u4CountryNum)
break; /* Found */
}
}
/* If no matched country code, use the default regulatory domain */
if (i >= REG_DOMAIN_GROUP_NUM) {
DBGLOG(RLM, INFO, "No matched country code, use the default regulatory domain\n");
prDomainInfo = &arSupportedRegDomains[REG_DOMAIN_DEF_IDX];
}
}
prAdapter->prDomainInfo = prDomainInfo;
return prDomainInfo;
}
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in/out] The input variable pointed by pucNumOfChannel is the max
* arrary size. The return value indciates meaning list size.
*
* \return none
*/
/*----------------------------------------------------------------------------*/
VOID
rlmDomainGetChnlList_V2(P_ADAPTER_T prAdapter,
ENUM_BAND_T eSpecificBand, BOOLEAN fgNoDfs,
UINT_8 ucMaxChannelNum, PUINT_8 pucNumOfChannel, P_RF_CHANNEL_INFO_T paucChannelList)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
ENUM_BAND_T band;
UINT_8 max_count, i, ucNum;
struct channel *prCh;
if (eSpecificBand == BAND_2G4) {
i = 0;
max_count = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
} else if (eSpecificBand == BAND_5G) {
i = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
max_count = rlmDomainGetActiveChannelCount(KAL_BAND_5GHZ) +
rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
} else {
i = 0;
max_count = rlmDomainGetActiveChannelCount(KAL_BAND_5GHZ) +
rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
}
ucNum = 0;
for (; i < max_count; i++) {
prCh = rlmDomainGetActiveChannels() + i;
if (fgNoDfs && (prCh->flags & IEEE80211_CHAN_RADAR))
continue; /*not match*/
if (i < rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ))
band = BAND_2G4;
else
band = BAND_5G;
paucChannelList[ucNum].eBand = band;
paucChannelList[ucNum].ucChannelNum = prCh->chNum;
ucNum++;
if (ucMaxChannelNum == ucNum)
break;
}
*pucNumOfChannel = ucNum;
#else
*pucNumOfChannel = 0;
#endif
}
VOID
rlmDomainGetChnlList(P_ADAPTER_T prAdapter,
ENUM_BAND_T eSpecificBand, BOOLEAN fgNoDfs,
UINT_8 ucMaxChannelNum, PUINT_8 pucNumOfChannel, P_RF_CHANNEL_INFO_T paucChannelList)
{
UINT_8 i, j, ucNum;
P_DOMAIN_SUBBAND_INFO prSubband;
P_DOMAIN_INFO_ENTRY prDomainInfo;
ASSERT(prAdapter);
ASSERT(paucChannelList);
ASSERT(pucNumOfChannel);
if (regd_is_single_sku_en())
return rlmDomainGetChnlList_V2(prAdapter, eSpecificBand,
fgNoDfs, ucMaxChannelNum,
pucNumOfChannel, paucChannelList);
/* If no matched country code, the final one will be used */
prDomainInfo = rlmDomainGetDomainInfo(prAdapter);
ASSERT(prDomainInfo);
ucNum = 0;
for (i = 0; i < MAX_SUBBAND_NUM; i++) {
prSubband = &prDomainInfo->rSubBand[i];
if (prSubband->ucBand == BAND_NULL || prSubband->ucBand >= BAND_NUM ||
(prSubband->ucBand == BAND_5G && !prAdapter->fgEnable5GBand))
continue;
/*repoert to upper layer only non-DFS channel for ap mode usage*/
if (fgNoDfs == TRUE && prSubband->fgDfs == TRUE)
continue;
if (eSpecificBand == BAND_NULL || prSubband->ucBand == eSpecificBand) {
for (j = 0; j < prSubband->ucNumChannels; j++) {
if (ucNum >= ucMaxChannelNum)
break;
paucChannelList[ucNum].eBand = prSubband->ucBand;
paucChannelList[ucNum].ucChannelNum =
prSubband->ucFirstChannelNum + j * prSubband->ucChannelSpan;
ucNum++;
}
}
}
*pucNumOfChannel = ucNum;
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainSendCmd(P_ADAPTER_T prAdapter, BOOLEAN fgIsOid)
{
if (!regd_is_single_sku_en())
rlmDomainSendPassiveScanInfoCmd(prAdapter, fgIsOid);
rlmDomainSendDomainInfoCmd(prAdapter, fgIsOid);
#if CFG_SUPPORT_PWR_LIMIT_COUNTRY
rlmDomainSendPwrLimitCmd(prAdapter);
#endif
}
VOID rlmDomainSendDomainInfoCmd_V2(P_ADAPTER_T prAdapter, BOOLEAN fgIsOid)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
u8 max_channel_count = 0;
u32 buff_max_size, buff_valid_size;
P_CMD_SET_DOMAIN_INFO_V2_T prCmd;
struct acctive_channel_list *prChs;
struct wiphy *pWiphy;
pWiphy = priv_to_wiphy(prAdapter->prGlueInfo);
if (pWiphy->bands[KAL_BAND_2GHZ] != NULL)
max_channel_count += pWiphy->bands[KAL_BAND_2GHZ]->n_channels;
if (pWiphy->bands[KAL_BAND_5GHZ] != NULL)
max_channel_count += pWiphy->bands[KAL_BAND_5GHZ]->n_channels;
if (max_channel_count == 0) {
DBGLOG(RLM, ERROR, "%s, invalid channel count.\n", __func__);
ASSERT(0);
}
buff_max_size = sizeof(CMD_SET_DOMAIN_INFO_V2_T) +
max_channel_count * sizeof(struct channel);
prCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, buff_max_size);
prChs = &(prCmd->active_chs);
/*
* Fill in the active channels
*/
rlmExtractChannelInfo(max_channel_count, prChs);
prCmd->u4CountryCode = rlmDomainGetCountryCode();
prCmd->uc2G4Bandwidth = prAdapter->rWifiVar.rConnSettings.uc2G4BandwidthMode;
prCmd->uc5GBandwidth = prAdapter->rWifiVar.rConnSettings.uc5GBandwidthMode;
prCmd->aucReserved[0] = 0;
prCmd->aucReserved[1] = 0;
buff_valid_size = sizeof(CMD_SET_DOMAIN_INFO_V2_T) +
(prChs->n_channels_2g + prChs->n_channels_5g) *
sizeof(struct channel);
DBGLOG(RLM, INFO, "rlmDomainSendDomainInfoCmd_V2(), buff_valid_size = 0x%x\n", buff_valid_size);
/* Set domain info to chip */
wlanSendSetQueryCmd(prAdapter, /* prAdapter */
CMD_ID_SET_DOMAIN_INFO, /* ucCID */
TRUE, /* fgSetQuery */
FALSE, /* fgNeedResp */
fgIsOid, /* fgIsOid */
NULL, /* pfCmdDoneHandler */
NULL, /* pfCmdTimeoutHandler */
buff_valid_size,
(PUINT_8) prCmd, /* pucInfoBuffer */
NULL, /* pvSetQueryBuffer */
0 /* u4SetQueryBufferLen */
);
cnmMemFree(prAdapter, prCmd);
#endif
}
VOID rlmDomainSendDomainInfoCmd(P_ADAPTER_T prAdapter, BOOLEAN fgIsOid)
{
P_DOMAIN_INFO_ENTRY prDomainInfo;
P_CMD_SET_DOMAIN_INFO_T prCmd;
P_DOMAIN_SUBBAND_INFO prSubBand;
UINT_8 i;
if (regd_is_single_sku_en())
return rlmDomainSendDomainInfoCmd_V2(prAdapter, fgIsOid);
prDomainInfo = rlmDomainGetDomainInfo(prAdapter);
ASSERT(prDomainInfo);
prCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, sizeof(CMD_SET_DOMAIN_INFO_T));
if (!prCmd) {
DBGLOG(RLM, ERROR, "Alloc cmd buffer failed\n");
return;
}
kalMemZero(prCmd, sizeof(CMD_SET_DOMAIN_INFO_T));
prCmd->u2CountryCode = prAdapter->rWifiVar.rConnSettings.u2CountryCode;
prCmd->u2IsSetPassiveScan = 0;
prCmd->uc2G4Bandwidth = prAdapter->rWifiVar.rConnSettings.uc2G4BandwidthMode;
prCmd->uc5GBandwidth = prAdapter->rWifiVar.rConnSettings.uc5GBandwidthMode;
prCmd->aucReserved[0] = 0;
prCmd->aucReserved[1] = 0;
for (i = 0; i < MAX_SUBBAND_NUM; i++) {
prSubBand = &prDomainInfo->rSubBand[i];
prCmd->rSubBand[i].ucRegClass = prSubBand->ucRegClass;
prCmd->rSubBand[i].ucBand = prSubBand->ucBand;
if (prSubBand->ucBand != BAND_NULL && prSubBand->ucBand < BAND_NUM) {
prCmd->rSubBand[i].ucChannelSpan = prSubBand->ucChannelSpan;
prCmd->rSubBand[i].ucFirstChannelNum = prSubBand->ucFirstChannelNum;
prCmd->rSubBand[i].ucNumChannels = prSubBand->ucNumChannels;
}
}
/* Set domain info to chip */
wlanSendSetQueryCmd(prAdapter, /* prAdapter */
CMD_ID_SET_DOMAIN_INFO, /* ucCID */
TRUE, /* fgSetQuery */
FALSE, /* fgNeedResp */
fgIsOid, /* fgIsOid */
NULL, /* pfCmdDoneHandler */
NULL, /* pfCmdTimeoutHandler */
sizeof(CMD_SET_DOMAIN_INFO_T), /* u4SetQueryInfoLen */
(PUINT_8) prCmd, /* pucInfoBuffer */
NULL, /* pvSetQueryBuffer */
0 /* u4SetQueryBufferLen */
);
cnmMemFree(prAdapter, prCmd);
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainSendPassiveScanInfoCmd(P_ADAPTER_T prAdapter, BOOLEAN fgIsOid)
{
P_DOMAIN_INFO_ENTRY prDomainInfo;
P_CMD_SET_DOMAIN_INFO_T prCmd;
WLAN_STATUS rStatus;
P_DOMAIN_SUBBAND_INFO prSubBand;
UINT_16 u2TargetCountryCode;
UINT_8 i, j;
prCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, sizeof(CMD_SET_DOMAIN_INFO_T));
if (!prCmd) {
DBGLOG(RLM, ERROR, "Alloc cmd buffer failed\n");
return;
}
kalMemZero(prCmd, sizeof(CMD_SET_DOMAIN_INFO_T));
prCmd->u2CountryCode = prAdapter->rWifiVar.rConnSettings.u2CountryCode;
prCmd->u2IsSetPassiveScan = 1;
prCmd->uc2G4Bandwidth = prAdapter->rWifiVar.rConnSettings.uc2G4BandwidthMode;
prCmd->uc5GBandwidth = prAdapter->rWifiVar.rConnSettings.uc5GBandwidthMode;
prCmd->aucReserved[0] = 0;
prCmd->aucReserved[1] = 0;
DBGLOG(RLM, TRACE, "u2CountryCode=0x%04x\n", prAdapter->rWifiVar.rConnSettings.u2CountryCode);
u2TargetCountryCode = prAdapter->rWifiVar.rConnSettings.u2CountryCode;
for (i = 0; i < REG_DOMAIN_PASSIVE_GROUP_NUM; i++) {
prDomainInfo = &arSupportedRegDomains_Passive[i];
for (j = 0; j < prDomainInfo->u4CountryNum; j++) {
if (prDomainInfo->pu2CountryGroup[j] == u2TargetCountryCode)
break;
}
if (j < prDomainInfo->u4CountryNum)
break; /* Found */
}
if (i >= REG_DOMAIN_PASSIVE_GROUP_NUM)
prDomainInfo = &arSupportedRegDomains_Passive[REG_DOMAIN_PASSIVE_DEF_IDX];
for (i = 0; i < MAX_SUBBAND_NUM; i++) {
prSubBand = &prDomainInfo->rSubBand[i];
prCmd->rSubBand[i].ucRegClass = prSubBand->ucRegClass;
prCmd->rSubBand[i].ucBand = prSubBand->ucBand;
if (prSubBand->ucBand != BAND_NULL && prSubBand->ucBand < BAND_NUM) {
prCmd->rSubBand[i].ucChannelSpan = prSubBand->ucChannelSpan;
prCmd->rSubBand[i].ucFirstChannelNum = prSubBand->ucFirstChannelNum;
prCmd->rSubBand[i].ucNumChannels = prSubBand->ucNumChannels;
}
}
/* Set passive scan channel info to chip */
rStatus = wlanSendSetQueryCmd(prAdapter, /* prAdapter */
CMD_ID_SET_DOMAIN_INFO, /* ucCID */
TRUE, /* fgSetQuery */
FALSE, /* fgNeedResp */
fgIsOid, /* fgIsOid */
NULL, /* pfCmdDoneHandler */
NULL, /* pfCmdTimeoutHandler */
sizeof(CMD_SET_DOMAIN_INFO_T), /* u4SetQueryInfoLen */
(PUINT_8) prCmd, /* pucInfoBuffer */
NULL, /* pvSetQueryBuffer */
0 /* u4SetQueryBufferLen */
);
ASSERT(rStatus == WLAN_STATUS_PENDING);
cnmMemFree(prAdapter, prCmd);
}
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in/out]
*
* \return TRUE Legal channel
* FALSE Illegal channel for current regulatory domain
*/
/*----------------------------------------------------------------------------*/
BOOLEAN rlmDomainIsLegalChannel_V2(P_ADAPTER_T prAdapter, ENUM_BAND_T eBand, UINT_8 ucChannel)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
UINT_8 idx, start_idx, end_idx;
struct channel *prCh;
if (eBand == BAND_2G4) {
start_idx = 0;
end_idx = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
} else {
start_idx = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
end_idx = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ) +
rlmDomainGetActiveChannelCount(KAL_BAND_5GHZ);
}
for (idx = start_idx; idx < end_idx; idx++) {
prCh = rlmDomainGetActiveChannels() + idx;
if (prCh->chNum == ucChannel)
return TRUE;
}
return FALSE;
#else
return FALSE;
#endif
}
BOOLEAN rlmDomainIsLegalChannel(P_ADAPTER_T prAdapter, ENUM_BAND_T eBand, UINT_8 ucChannel)
{
UINT_8 i, j;
P_DOMAIN_SUBBAND_INFO prSubband;
P_DOMAIN_INFO_ENTRY prDomainInfo;
if (regd_is_single_sku_en())
return rlmDomainIsLegalChannel_V2(prAdapter, eBand, ucChannel);
prDomainInfo = rlmDomainGetDomainInfo(prAdapter);
ASSERT(prDomainInfo);
for (i = 0; i < MAX_SUBBAND_NUM; i++) {
prSubband = &prDomainInfo->rSubBand[i];
if (prSubband->ucBand == BAND_5G && !prAdapter->fgEnable5GBand)
continue;
if (prSubband->ucBand == eBand) {
for (j = 0; j < prSubband->ucNumChannels; j++) {
if ((prSubband->ucFirstChannelNum + j * prSubband->ucChannelSpan)
== ucChannel) {
return TRUE;
}
}
}
}
return FALSE;
}
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in/out]
*
* \return none
*/
/*----------------------------------------------------------------------------*/
UINT_32 rlmDomainSupOperatingClassIeFill(PUINT_8 pBuf)
{
/*
* The Country element should only be included for Status Code 0 (Successful).
*/
UINT_32 u4IeLen;
UINT_8 aucClass[12] = { 0x01, 0x02, 0x03, 0x05, 0x16, 0x17, 0x19, 0x1b,
0x1c, 0x1e, 0x20, 0x21
};
/*
* The Supported Operating Classes element is used by a STA to advertise the
* operating classes that it is capable of operating with in this country.
*
* The Country element (see 8.4.2.10) allows a STA to configure its PHY and MAC
* for operation when the operating triplet of Operating Extension Identifier,
* Operating Class, and Coverage Class fields is present.
*/
SUP_OPERATING_CLASS_IE(pBuf)->ucId = ELEM_ID_SUP_OPERATING_CLASS;
SUP_OPERATING_CLASS_IE(pBuf)->ucLength = 1 + sizeof(aucClass);
SUP_OPERATING_CLASS_IE(pBuf)->ucCur = 0x0c; /* 0x51 */
kalMemCopy(SUP_OPERATING_CLASS_IE(pBuf)->ucSup, aucClass, sizeof(aucClass));
u4IeLen = (SUP_OPERATING_CLASS_IE(pBuf)->ucLength + 2);
pBuf += u4IeLen;
COUNTRY_IE(pBuf)->ucId = ELEM_ID_COUNTRY_INFO;
COUNTRY_IE(pBuf)->ucLength = 6;
COUNTRY_IE(pBuf)->aucCountryStr[0] = 0x55;
COUNTRY_IE(pBuf)->aucCountryStr[1] = 0x53;
COUNTRY_IE(pBuf)->aucCountryStr[2] = 0x20;
COUNTRY_IE(pBuf)->arCountryStr[0].ucFirstChnlNum = 1;
COUNTRY_IE(pBuf)->arCountryStr[0].ucNumOfChnl = 11;
COUNTRY_IE(pBuf)->arCountryStr[0].cMaxTxPwrLv = 0x1e;
u4IeLen += (COUNTRY_IE(pBuf)->ucLength + 2);
return u4IeLen;
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (fgValid) : 0 -> inValid, 1 -> Valid
*/
/*----------------------------------------------------------------------------*/
BOOLEAN rlmDomainCheckChannelEntryValid(P_ADAPTER_T prAdapter, UINT_8 ucCentralCh)
{
BOOLEAN fgValid = FALSE;
UINT_8 ucTemp = 0xff;
UINT_8 i;
/*Check Power limit table channel efficient or not */
for (i = POWER_LIMIT_2G4; i < POWER_LIMIT_SUBAND_NUM; i++) {
if ((ucCentralCh >= g_rRlmSubBand[i].ucStartCh) && (ucCentralCh <= g_rRlmSubBand[i].ucEndCh))
ucTemp = (ucCentralCh - g_rRlmSubBand[i].ucStartCh) % g_rRlmSubBand[i].ucInterval;
}
#if 0
/*2.4G, ex 1, 2, 3 */
if (ucCentralCh >= BAND_2G4_LOWER_BOUND && ucCentralCh <= BAND_2G4_UPPER_BOUND)
ucTemp = 0;
/*FCC- Spec : Band UNII-1, ex 36, 38, 40.... */
else if (ucCentralCh >= UNII1_LOWER_BOUND && ucCentralCh <= UNII1_UPPER_BOUND)
ucTemp = (ucCentralCh - UNII1_LOWER_BOUND) % 2;
/*FCC- Spec : Band UNII-2A, ex 52, 54, 56.... */
else if (ucCentralCh >= UNII2A_LOWER_BOUND && ucCentralCh <= UNII2A_UPPER_BOUND)
ucTemp = (ucCentralCh - UNII2A_LOWER_BOUND) % 2;
/*FCC- Spec : Band UNII-2C, ex 100, 102, 104.... */
else if (ucCentralCh >= UNII2C_LOWER_BOUND && ucCentralCh <= UNII2C_UPPER_BOUND)
ucTemp = (ucCentralCh - UNII2C_LOWER_BOUND) % 2;
/*FCC- Spec : Band UNII-3, ex 149, 151, 153... */
else if (ucCentralCh >= UNII3_LOWER_BOUND && ucCentralCh <= UNII3_UPPER_BOUND)
ucTemp = (ucCentralCh - UNII3_LOWER_BOUND) % 2;
#endif
if (ucTemp == 0)
fgValid = TRUE;
return fgValid;
}
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in]
*
* \return
*/
/*----------------------------------------------------------------------------*/
UINT_8 rlmDomainGetCenterChannel(ENUM_BAND_T eBand, UINT_8 ucPriChannel, ENUM_CHNL_EXT_T eExtend)
{
UINT_8 ucCenterChannel;
if (eExtend == CHNL_EXT_SCA)
ucCenterChannel = ucPriChannel + 2;
else if (eExtend == CHNL_EXT_SCB)
ucCenterChannel = ucPriChannel - 2;
else
ucCenterChannel = ucPriChannel;
return ucCenterChannel;
}
/*----------------------------------------------------------------------------*/
/*!
* \brief
*
* \param[in]
*
* \return
*/
/*----------------------------------------------------------------------------*/
BOOLEAN
rlmDomainIsValidRfSetting(P_ADAPTER_T prAdapter,
ENUM_BAND_T eBand,
UINT_8 ucPriChannel,
ENUM_CHNL_EXT_T eExtend,
ENUM_CHANNEL_WIDTH_T eChannelWidth, UINT_8 ucChannelS1, UINT_8 ucChannelS2)
{
UINT_8 ucCenterChannel = 0;
UINT_8 ucUpperChannel;
UINT_8 ucLowerChannel;
BOOLEAN fgValidChannel = TRUE;
BOOLEAN fgUpperChannel = TRUE;
BOOLEAN fgLowerChannel = TRUE;
BOOLEAN fgValidBW = TRUE;
BOOLEAN fgValidRfSetting = TRUE;
UINT_32 u4PrimaryOffset;
/*DBG msg for Channel InValid */
if (eChannelWidth == CW_20_40MHZ) {
ucCenterChannel = rlmDomainGetCenterChannel(eBand, ucPriChannel, eExtend);
/* Check Central Channel Valid or Not */
fgValidChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucCenterChannel);
if (fgValidChannel == FALSE)
DBGLOG(RLM, WARN, "Rf20: CentralCh=%d\n", ucCenterChannel);
/* Check Upper Channel and Lower Channel */
switch (eExtend) {
case CHNL_EXT_SCA:
ucUpperChannel = ucPriChannel + 4;
ucLowerChannel = ucPriChannel;
break;
case CHNL_EXT_SCB:
ucUpperChannel = ucPriChannel;
ucLowerChannel = ucPriChannel - 4;
break;
default:
ucUpperChannel = ucPriChannel;
ucLowerChannel = ucPriChannel;
break;
}
fgUpperChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucUpperChannel);
if (fgUpperChannel == FALSE)
DBGLOG(RLM, WARN, "Rf20: UpperCh=%d\n", ucUpperChannel);
fgLowerChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucLowerChannel);
if (fgLowerChannel == FALSE)
DBGLOG(RLM, WARN, "Rf20: LowerCh=%d\n", ucLowerChannel);
} else if ((eChannelWidth == CW_80MHZ) || (eChannelWidth == CW_160MHZ)) {
ucCenterChannel = ucChannelS1;
/* Check Central Channel Valid or Not */
if (eChannelWidth != CW_160MHZ) {
/*BW not check , ex: primary 36 and central channel 50 will fail the check*/
fgValidChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucCenterChannel);
}
if (fgValidChannel == FALSE)
DBGLOG(RLM, WARN, "Rf80/160C: CentralCh=%d\n", ucCenterChannel);
} else if (eChannelWidth == CW_80P80MHZ) {
ucCenterChannel = ucChannelS1;
fgValidChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucCenterChannel);
if (fgValidChannel == FALSE)
DBGLOG(RLM, WARN, "Rf160NC: CentralCh1=%d\n", ucCenterChannel);
ucCenterChannel = ucChannelS2;
fgValidChannel = rlmDomainCheckChannelEntryValid(prAdapter, ucCenterChannel);
if (fgValidChannel == FALSE)
DBGLOG(RLM, WARN, "Rf160NC: CentralCh2=%d\n", ucCenterChannel);
/* Check Central Channel Valid or Not */
} else {
DBGLOG(RLM, ERROR, "Wrong BW =%d\n", eChannelWidth);
fgValidChannel = FALSE;
}
/* Check BW Setting Correct or Not */
if (eBand == BAND_2G4) {
if (eChannelWidth != CW_20_40MHZ) {
fgValidBW = FALSE;
DBGLOG(RLM, WARN, "Rf: B=%d, W=%d\n", eBand, eChannelWidth);
}
} else {
if ((eChannelWidth == CW_80MHZ) || (eChannelWidth == CW_80P80MHZ)) {
u4PrimaryOffset = CAL_CH_OFFSET_80M(ucPriChannel, ucChannelS1);
if (u4PrimaryOffset > 4) {
fgValidBW = FALSE;
DBGLOG(RLM, WARN, "Rf: PriOffSet=%d, W=%d\n", u4PrimaryOffset, eChannelWidth);
}
} else if (eChannelWidth == CW_160MHZ) {
u4PrimaryOffset = CAL_CH_OFFSET_160M(ucPriChannel, ucCenterChannel);
if (u4PrimaryOffset > 8) {
fgValidBW = FALSE;
DBGLOG(RLM, WARN, "Rf: PriOffSet=%d, W=%d\n", u4PrimaryOffset, eChannelWidth);
}
}
}
if ((fgValidBW == FALSE) || (fgValidChannel == FALSE) || (fgUpperChannel == FALSE) || (fgLowerChannel == FALSE))
fgValidRfSetting = FALSE;
return fgValidRfSetting;
}
#if (CFG_SUPPORT_SINGLE_SKU == 1)
/*
* This function coverts country code from alphabet chars to u32,
* the caller need to pass country code chars and do size check
*/
UINT_32 rlmDomainAlpha2ToU32(PCHAR pcAlpha2, UINT_8 ucAlpha2Size)
{
UINT_8 ucIdx;
UINT_32 u4CountryCode = 0;
if (ucAlpha2Size > TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN) {
DBGLOG(RLM, ERROR, "alpha2 size %d is invalid!(max: %d)\n",
ucAlpha2Size, TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN);
ucAlpha2Size = TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN;
}
for (ucIdx = 0; ucIdx < ucAlpha2Size; ucIdx++)
u4CountryCode |= (pcAlpha2[ucIdx] << (ucIdx * 8));
return u4CountryCode;
}
/*----------------------------------------------------------------------------*/
/*!
* \brief Search the tx power limit setting range of the specified in the text file
*
* \param[IN] u4CountryCode The u32 type of the specified country.
* \param[IN] pucBuf The content of the text file.
* \param[IN] u4cBufLen End boundary of the text file.
* \param[OUT] pu4CountryStart Store the start position of the desired country settings.
* \param[OUT] pu4CountryEnd Store the end position of the desired country settings.
*
* \retval TRUE Success.
* \retval FALSE Failure.
*/
/*----------------------------------------------------------------------------*/
BOOL rlmDomainTxPwrLimitGetCountryRange(
UINT_32 u4CountryCode, PUINT_8 pucBuf, UINT_32 u4BufLen,
PUINT_32 pu4CountryStart, PUINT_32 pu4CountryEnd)
{
UINT_32 u4TmpPos = 0;
char pcrCountryStr[TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN + 1] = {0};
UINT_8 cIdx = 0;
while (1) {
while (u4TmpPos < u4BufLen && pucBuf[u4TmpPos] != '[')
u4TmpPos++;
u4TmpPos++; /* skip the '[' char */
cIdx = 0;
while (u4TmpPos < u4BufLen && cIdx < TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN &&
pucBuf[u4TmpPos] != ']') {
pcrCountryStr[cIdx++] = pucBuf[u4TmpPos];
u4TmpPos++;
}
u4TmpPos++; /* skip the ']' char */
if (u4TmpPos >= u4BufLen || cIdx > TX_PWR_LIMIT_COUNTRY_STR_MAX_LEN)
return FALSE;
if (u4CountryCode == rlmDomainAlpha2ToU32(pcrCountryStr, cIdx)) {
DBGLOG(RLM, INFO, "Found TxPwrLimit table for CountryCode \"%s\"\n",
pcrCountryStr);
*pu4CountryStart = u4TmpPos; /* the location after char ']' */
break;
}
}
while (u4TmpPos < u4BufLen && pucBuf[u4TmpPos] != '[')
u4TmpPos++;
*pu4CountryEnd = u4TmpPos;
return TRUE;
}
BOOL rlmDomainTxPwrLimitSearchSection(const char *pSectionName,
PINT_8 pucBuf, PUINT_32 pu4Pos, UINT_32 u4BufEnd)
{
UINT_32 u4TmpPos = *pu4Pos;
UINT_8 uSectionNameLen = kalStrLen(pSectionName);
while (1) {
while (u4TmpPos < u4BufEnd && pucBuf[u4TmpPos] != '<')
u4TmpPos++;
u4TmpPos++; /* skip char '<' */
if (u4TmpPos + uSectionNameLen >= u4BufEnd)
return FALSE;
if (kalStrnCmp(&pucBuf[u4TmpPos], pSectionName, uSectionNameLen) == 0) {
/* Go to the end of section header line */
while (u4TmpPos < u4BufEnd && pucBuf[u4TmpPos] != '\n')
u4TmpPos++;
*pu4Pos = u4TmpPos;
break;
}
}
return TRUE;
}
BOOL rlmDomainTxPwrLimitSectionEnd(PUINT_8 pucBuf,
const char *pSectionName, PUINT_32 pu4Pos, UINT_32 u4BufEnd)
{
UINT_32 u4TmpPos = *pu4Pos;
char cTmpChar = 0;
UINT_8 uSectionNameLen = kalStrLen(pSectionName);
while (u4TmpPos < u4BufEnd) {
cTmpChar = pucBuf[u4TmpPos];
/* skip blank lines */
if (cTmpChar == ' ' || cTmpChar == '\t' ||
cTmpChar == '\n' || cTmpChar == '\r') {
u4TmpPos++;
continue;
}
break;
}
if (u4TmpPos + uSectionNameLen + 2 >= u4BufEnd) {/* 2 means '/' and '>' */
*pu4Pos = u4BufEnd;
return FALSE;
}
if (pucBuf[u4TmpPos] != '<')
return FALSE;
if (pucBuf[u4TmpPos + 1] != '/' ||
pucBuf[u4TmpPos + 2 + uSectionNameLen] != '>' ||
kalStrnCmp(&pucBuf[u4TmpPos + 2], pSectionName, uSectionNameLen)) {
*pu4Pos = u4TmpPos + uSectionNameLen + 2;
return FALSE;
}
*pu4Pos = u4TmpPos + uSectionNameLen + 3; /* 3 means go to the location after '>' */
return TRUE;
}
INT_8 rlmDomainTxPwrLimitGetChIdx(
struct TX_PWR_LIMIT_DATA *pTxPwrLimit, UINT_8 ucChannel)
{
INT_8 cIdx = 0;
for (cIdx = 0; cIdx < pTxPwrLimit->ucChNum; cIdx++)
if (ucChannel == pTxPwrLimit->rChannelTxPwrLimit[cIdx].ucChannel)
return cIdx;
DBGLOG(RLM, ERROR, "Can't find idx of channel %d in TxPwrLimit data\n",
ucChannel);
return -1;
}
BOOL rlmDomainTxPwrLimitLoadChannelSetting(
PUINT_8 pucBuf, PUINT_32 pu4Pos, UINT_32 u4BufEnd,
struct TX_PWR_LIMIT_DATA *pTxPwrLimit, UINT_8 ucSectionIdx)
{
UINT_32 u4TmpPos = *pu4Pos;
char cTmpChar = 0;
struct CHANNEL_TX_PWR_LIMIT *prChTxPwrLimit = NULL;
BOOL bNeg = FALSE;
INT_8 cLimitValue = 0, cChIdx = 0;
UINT_8 ucIdx = 0, ucChannel = 0;
/* skip blank lines */
while (u4TmpPos < u4BufEnd) {
cTmpChar = pucBuf[u4TmpPos];
if (cTmpChar == ' ' || cTmpChar == '\t' ||
cTmpChar == '\n' || cTmpChar == '\r') {
u4TmpPos++;
continue;
}
break;
}
/* current is at the location of 'c', check remaining buf length for 'chxxx' */
if (u4TmpPos + 5 >= u4BufEnd) {
DBGLOG(RLM, ERROR,
"Invalid location of ch setting: %u/%u\n",
u4TmpPos, u4BufEnd);
return FALSE;
}
if (pucBuf[u4TmpPos] == 'c' && pucBuf[u4TmpPos + 1] == 'h') {
ucChannel = (pucBuf[u4TmpPos + 2] - '0') * 100 +
(pucBuf[u4TmpPos + 3] - '0') * 10 +
(pucBuf[u4TmpPos + 4] - '0');
} else { /* invalid format */
*pu4Pos = u4TmpPos;
DBGLOG(RLM, ERROR, "Invalid ch setting starting chars: %c%c\n",
pucBuf[u4TmpPos], pucBuf[u4TmpPos + 1]);
/* goto next line */
while (*pu4Pos < u4BufEnd && pucBuf[*pu4Pos] != '\n')
(*pu4Pos)++;
return TRUE;
}
cChIdx = rlmDomainTxPwrLimitGetChIdx(pTxPwrLimit, ucChannel);
if (cChIdx == -1) {
*pu4Pos = u4TmpPos;
DBGLOG(RLM, ERROR, "Invalid ch %u %c%c%c\n", ucChannel,
pucBuf[u4TmpPos + 2], pucBuf[u4TmpPos + 3], pucBuf[u4TmpPos + 4]);
/* goto next line */
while (*pu4Pos < u4BufEnd && pucBuf[*pu4Pos] != '\n')
(*pu4Pos)++;
return TRUE;
}
u4TmpPos += 5;
prChTxPwrLimit = &pTxPwrLimit->rChannelTxPwrLimit[cChIdx];
/* read the channel TxPwrLimit settings */
for (ucIdx = 0; ucIdx < gTx_Pwr_Limit_Element_Num[ucSectionIdx]; ucIdx++) {
/* skip blank and comma */
while (u4TmpPos < u4BufEnd) {
cTmpChar = pucBuf[u4TmpPos];
if (cTmpChar == ' ' || cTmpChar == '\t' || cTmpChar == ',') {
u4TmpPos++;
continue;
}
break;
}
if (u4TmpPos >= u4BufEnd) {
*pu4Pos = u4BufEnd;
DBGLOG(RLM, ERROR,
"Invalid location of ch tx pwr limit val: %u/%u\n",
u4TmpPos, u4BufEnd);
return FALSE;
}
bNeg = FALSE;
cTmpChar = pucBuf[u4TmpPos];
if (cTmpChar == '-') {
bNeg = TRUE;
u4TmpPos++;
} else if (cTmpChar == 'x') {
prChTxPwrLimit->rTxPwrLimitValue[ucSectionIdx][ucIdx] =
TX_PWR_LIMIT_MAX_VAL;
u4TmpPos++;
continue;
}
cLimitValue = 0;
while (u4TmpPos < u4BufEnd) {
cTmpChar = pucBuf[u4TmpPos];
if (cTmpChar < '0' || cTmpChar > '9')
break;
cLimitValue = (cLimitValue * 10) + (cTmpChar - '0');
u4TmpPos++;
}
if (bNeg)
cLimitValue = -cLimitValue;
prChTxPwrLimit->rTxPwrLimitValue[ucSectionIdx][ucIdx] = cLimitValue;
}
*pu4Pos = u4TmpPos;
return TRUE;
}
VOID rlmDomainTxPwrLimitRemoveComments(
PUINT_8 pucBuf, UINT_32 u4BufLen)
{
UINT_32 u4TmpPos = 0;
char cTmpChar = 0;
while (u4TmpPos < u4BufLen) {
cTmpChar = pucBuf[u4TmpPos];
if (cTmpChar == '#') {
while (cTmpChar != '\n') {
pucBuf[u4TmpPos] = ' ';
u4TmpPos++;
if (u4TmpPos >= u4BufLen)
break;
cTmpChar = pucBuf[u4TmpPos];
}
}
u4TmpPos++;
}
}
BOOL rlmDomainTxPwrLimitLoad(
P_ADAPTER_T prAdapter, PUINT_8 pucBuf, UINT_32 u4BufLen,
UINT_32 u4CountryCode, struct TX_PWR_LIMIT_DATA *pTxPwrLimit)
{
UINT_8 uSecIdx = 0;
UINT_32 u4CountryStart = 0, u4CountryEnd = 0, u4Pos = 0;
rlmDomainTxPwrLimitRemoveComments(pucBuf, u4BufLen);
if (!rlmDomainTxPwrLimitGetCountryRange(u4CountryCode, pucBuf,
u4BufLen, &u4CountryStart, &u4CountryEnd)) {
DBGLOG(RLM, ERROR, "Can't find specified table in %s\n",
WLAN_TX_PWR_LIMIT_FILE_NAME);
return FALSE;
}
u4Pos = u4CountryStart;
for (uSecIdx = 0; uSecIdx < TX_PWR_LIMIT_SECTION_NUM; uSecIdx++) {
if (!rlmDomainTxPwrLimitSearchSection(gTx_Pwr_Limit_Section[uSecIdx],
pucBuf, &u4Pos, u4CountryEnd)) {
DBGLOG(RLM, ERROR, "Can't find specified section %s in %s\n",
gTx_Pwr_Limit_Section[uSecIdx], WLAN_TX_PWR_LIMIT_FILE_NAME);
return FALSE;
}
DBGLOG(RLM, INFO, "Find specified section %s in %s\n",
gTx_Pwr_Limit_Section[uSecIdx], WLAN_TX_PWR_LIMIT_FILE_NAME);
while (!rlmDomainTxPwrLimitSectionEnd(pucBuf,
gTx_Pwr_Limit_Section[uSecIdx], &u4Pos, u4CountryEnd) &&
u4Pos < u4CountryEnd) {
if (!rlmDomainTxPwrLimitLoadChannelSetting(pucBuf, &u4Pos,
u4CountryEnd, pTxPwrLimit, uSecIdx))
return FALSE;
}
}
DBGLOG(RLM, INFO, "Load %s finished\n", WLAN_TX_PWR_LIMIT_FILE_NAME);
return TRUE;
}
VOID rlmDomainTxPwrLimitSetChValues(
P_CMD_CHANNEL_POWER_LIMIT_V2 pCmd, struct CHANNEL_TX_PWR_LIMIT *pChTxPwrLimit)
{
UINT_8 section = 0, e = 0;
pCmd->tx_pwr_dsss_cck = pChTxPwrLimit->rTxPwrLimitValue[0][0];
pCmd->tx_pwr_dsss_bpsk = pChTxPwrLimit->rTxPwrLimitValue[0][1];
pCmd->tx_pwr_ofdm_bpsk = pChTxPwrLimit->rTxPwrLimitValue[0][2]; /* 6M, 9M */
pCmd->tx_pwr_ofdm_qpsk = pChTxPwrLimit->rTxPwrLimitValue[0][3]; /* 12M, 18M */
pCmd->tx_pwr_ofdm_16qam = pChTxPwrLimit->rTxPwrLimitValue[0][4]; /* 24M, 36M */
pCmd->tx_pwr_ofdm_48m = pChTxPwrLimit->rTxPwrLimitValue[0][5];
pCmd->tx_pwr_ofdm_54m = pChTxPwrLimit->rTxPwrLimitValue[0][6];
pCmd->tx_pwr_ht20_bpsk = pChTxPwrLimit->rTxPwrLimitValue[1][0]; /* MCS0*/
pCmd->tx_pwr_ht20_qpsk = pChTxPwrLimit->rTxPwrLimitValue[1][1]; /* MCS1, MCS2*/
pCmd->tx_pwr_ht20_16qam = pChTxPwrLimit->rTxPwrLimitValue[1][2]; /* MCS3, MCS4*/
pCmd->tx_pwr_ht20_mcs5 = pChTxPwrLimit->rTxPwrLimitValue[1][3]; /* MCS5*/
pCmd->tx_pwr_ht20_mcs6 = pChTxPwrLimit->rTxPwrLimitValue[1][4]; /* MCS6*/
pCmd->tx_pwr_ht20_mcs7 = pChTxPwrLimit->rTxPwrLimitValue[1][5]; /* MCS7*/
pCmd->tx_pwr_ht40_bpsk = pChTxPwrLimit->rTxPwrLimitValue[2][0]; /* MCS0*/
pCmd->tx_pwr_ht40_qpsk = pChTxPwrLimit->rTxPwrLimitValue[2][1]; /* MCS1, MCS2*/
pCmd->tx_pwr_ht40_16qam = pChTxPwrLimit->rTxPwrLimitValue[2][2]; /* MCS3, MCS4*/
pCmd->tx_pwr_ht40_mcs5 = pChTxPwrLimit->rTxPwrLimitValue[2][3]; /* MCS5*/
pCmd->tx_pwr_ht40_mcs6 = pChTxPwrLimit->rTxPwrLimitValue[2][4]; /* MCS6*/
pCmd->tx_pwr_ht40_mcs7 = pChTxPwrLimit->rTxPwrLimitValue[2][5]; /* MCS7*/
pCmd->tx_pwr_ht40_mcs32 = pChTxPwrLimit->rTxPwrLimitValue[2][6]; /* MCS32*/
pCmd->tx_pwr_vht20_bpsk = pChTxPwrLimit->rTxPwrLimitValue[3][0]; /* MCS0*/
pCmd->tx_pwr_vht20_qpsk = pChTxPwrLimit->rTxPwrLimitValue[3][1]; /* MCS1, MCS2*/
pCmd->tx_pwr_vht20_16qam = pChTxPwrLimit->rTxPwrLimitValue[3][2]; /* MCS3, MCS4*/
pCmd->tx_pwr_vht20_64qam = pChTxPwrLimit->rTxPwrLimitValue[3][3]; /* MCS5, MCS6*/
pCmd->tx_pwr_vht20_mcs7 = pChTxPwrLimit->rTxPwrLimitValue[3][4];
pCmd->tx_pwr_vht20_mcs8 = pChTxPwrLimit->rTxPwrLimitValue[3][5];
pCmd->tx_pwr_vht20_mcs9 = pChTxPwrLimit->rTxPwrLimitValue[3][6];
pCmd->tx_pwr_vht_40 = pChTxPwrLimit->rTxPwrLimitValue[4][2];
pCmd->tx_pwr_vht_80 = pChTxPwrLimit->rTxPwrLimitValue[4][3];
pCmd->tx_pwr_vht_160c = pChTxPwrLimit->rTxPwrLimitValue[4][5];
pCmd->tx_pwr_vht_160nc = pChTxPwrLimit->rTxPwrLimitValue[4][4];
pCmd->tx_pwr_lg_40 = pChTxPwrLimit->rTxPwrLimitValue[4][0];
pCmd->tx_pwr_lg_80 = pChTxPwrLimit->rTxPwrLimitValue[4][1];
DBGLOG(RLM, TRACE, "ch %d\n", pCmd->ucCentralCh);
for (section = 0; section < TX_PWR_LIMIT_SECTION_NUM; section++)
for (e = 0; e < gTx_Pwr_Limit_Element_Num[section]; e++)
DBGLOG(RLM, TRACE, "TxPwrLimit[%s][%s]= %d\n",
gTx_Pwr_Limit_Section[section],
gTx_Pwr_Limit_Element[section][e],
pChTxPwrLimit->rTxPwrLimitValue[section][e]);
}
VOID rlmDomainTxPwrLimitSetValues(
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T pSetCmd,
struct TX_PWR_LIMIT_DATA *pTxPwrLimit)
{
UINT_8 ucIdx = 0;
INT_8 cChIdx = 0;
P_CMD_CHANNEL_POWER_LIMIT_V2 pCmd = NULL;
struct CHANNEL_TX_PWR_LIMIT *pChTxPwrLimit = NULL;
if (!pSetCmd || !pTxPwrLimit) {
DBGLOG(RLM, ERROR, "%s: Invalid request!!!\n", __func__);
return;
}
for (ucIdx = 0; ucIdx < pSetCmd->ucNum; ucIdx++) {
pCmd = &(pSetCmd->rChannelPowerLimit[ucIdx]);
cChIdx = rlmDomainTxPwrLimitGetChIdx(pTxPwrLimit, pCmd->ucCentralCh);
if (cChIdx == -1) {
DBGLOG(RLM, ERROR, "Invalid ch idx found while assigning values\n");
continue;
}
pChTxPwrLimit = &pTxPwrLimit->rChannelTxPwrLimit[cChIdx];
rlmDomainTxPwrLimitSetChValues(pCmd, pChTxPwrLimit);
}
}
BOOL rlmDomainTxPwrLimitLoadFromFile(P_ADAPTER_T prAdapter,
UINT_32 u4CountryCode, struct TX_PWR_LIMIT_DATA *pTxPwrLimit)
{
PUINT_8 pucConfigBuf;
UINT_32 u4ConfigReadLen;
BOOL bRet = TRUE;
pucConfigBuf = (PUINT_8) kalMemAlloc(WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE, VIR_MEM_TYPE);
if (!pucConfigBuf)
return FALSE;
kalMemZero(pucConfigBuf, WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE);
u4ConfigReadLen = 0;
if (wlanGetFileContent(prAdapter, WLAN_TX_PWR_LIMIT_FILE_NAME, pucConfigBuf,
WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE, &u4ConfigReadLen, TRUE) == 0) {
/* ToDo:: Nothing */
} else if (wlanGetFileContent(prAdapter, "/storage/sdcard0/" WLAN_TX_PWR_LIMIT_FILE_NAME, pucConfigBuf,
WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE, &u4ConfigReadLen, FALSE) == 0) {
/* ToDo:: Nothing */
} else if (wlanGetFileContent(prAdapter, "/data/misc/" WLAN_TX_PWR_LIMIT_FILE_NAME, pucConfigBuf,
WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE, &u4ConfigReadLen, FALSE) == 0) {
/* ToDo:: Nothing */
} else if (wlanGetFileContent(prAdapter, "/data/misc/wifi/" WLAN_TX_PWR_LIMIT_FILE_NAME, pucConfigBuf,
WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE, &u4ConfigReadLen, FALSE) == 0) {
/* ToDo:: Nothing */
} else {
bRet = FALSE;
goto error;
}
if (pucConfigBuf[0] == '\0' || u4ConfigReadLen == 0) {
bRet = FALSE;
goto error;
}
if (!rlmDomainTxPwrLimitLoad(prAdapter, pucConfigBuf, u4ConfigReadLen,
u4CountryCode, pTxPwrLimit)) {
bRet = FALSE;
goto error;
}
error:
kalMemFree(pucConfigBuf, VIR_MEM_TYPE, WLAN_TX_PWR_LIMIT_FILE_BUF_SIZE);
return bRet;
}
BOOLEAN rlmDomainGetTxPwrLimit(u32 country_code,
P_GLUE_INFO_T prGlueInfo,
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T pSetCmd_2g,
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T pSetCmd_5g
)
{
int bRet = TRUE;
UINT_8 ucIdx = 0, ucCnt = 0;
struct TX_PWR_LIMIT_DATA *pTxPwrLimit =
(struct TX_PWR_LIMIT_DATA *) kalMemAlloc(sizeof(struct TX_PWR_LIMIT_DATA), VIR_MEM_TYPE);
if (!pTxPwrLimit) {
bRet = FALSE;
DBGLOG(RLM, ERROR, "Alloc buffer for TxPwrLimit main struct failed\n");
return bRet;
}
pTxPwrLimit->ucChNum = (pSetCmd_2g ? pSetCmd_2g->ucNum : 0) + (pSetCmd_5g ? pSetCmd_5g->ucNum : 0);
pTxPwrLimit->rChannelTxPwrLimit =
(struct CHANNEL_TX_PWR_LIMIT *) kalMemAlloc(sizeof(struct CHANNEL_TX_PWR_LIMIT) *
(pTxPwrLimit->ucChNum), VIR_MEM_TYPE);
if (!pTxPwrLimit->rChannelTxPwrLimit) {
bRet = FALSE;
DBGLOG(RLM, ERROR, "Alloc buffer for TxPwrLimit ch values failed\n");
goto error;
}
kalMemSet(pTxPwrLimit->rChannelTxPwrLimit, MAX_TX_POWER,
sizeof(struct CHANNEL_TX_PWR_LIMIT) * (pTxPwrLimit->ucChNum));
if (pSetCmd_2g)
for (ucIdx = 0; ucIdx < pSetCmd_2g->ucNum; ucIdx++) {
pTxPwrLimit->rChannelTxPwrLimit[ucCnt].ucChannel =
pSetCmd_2g->rChannelPowerLimit[ucIdx].ucCentralCh;
ucCnt++;
}
if (pSetCmd_5g)
for (ucIdx = 0; ucIdx < pSetCmd_5g->ucNum; ucIdx++) {
pTxPwrLimit->rChannelTxPwrLimit[ucCnt].ucChannel =
pSetCmd_5g->rChannelPowerLimit[ucIdx].ucCentralCh;
ucCnt++;
}
bRet = rlmDomainTxPwrLimitLoadFromFile(prGlueInfo->prAdapter,
country_code, pTxPwrLimit);
if (bRet) {
rlmDomainTxPwrLimitSetValues(pSetCmd_2g, pTxPwrLimit);
rlmDomainTxPwrLimitSetValues(pSetCmd_5g, pTxPwrLimit);
}
kalMemFree(pTxPwrLimit->rChannelTxPwrLimit, VIR_MEM_TYPE,
sizeof(CHANNEL_TX_PWR_LIMIT) * pTxPwrLimit->ucChNum);
error:
kalMemFree(pTxPwrLimit, VIR_MEM_TYPE, sizeof(TX_PWR_LIMIT_DATA));
return bRet;
}
#endif
#if CFG_SUPPORT_PWR_LIMIT_COUNTRY
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (fgValid) : 0 -> inValid, 1 -> Valid
*/
/*----------------------------------------------------------------------------*/
BOOLEAN
rlmDomainCheckPowerLimitValid(P_ADAPTER_T prAdapter,
COUNTRY_POWER_LIMIT_TABLE_CONFIGURATION rPowerLimitTableConfiguration,
UINT_8 ucPwrLimitNum)
{
UINT_8 i;
BOOLEAN fgValid = TRUE;
PINT_8 prPwrLimit;
prPwrLimit = &rPowerLimitTableConfiguration.aucPwrLimit[0];
for (i = 0; i < ucPwrLimitNum; i++, prPwrLimit++) {
if (*prPwrLimit > MAX_TX_POWER || *prPwrLimit < MIN_TX_POWER) {
fgValid = FALSE;
break; /*Find out Wrong Power limit */
}
}
return fgValid;
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainCheckCountryPowerLimitTable(P_ADAPTER_T prAdapter)
{
UINT_8 i, j;
UINT_16 u2CountryCodeTable, u2CountryCodeCheck;
BOOLEAN fgChannelValid = FALSE;
BOOLEAN fgPowerLimitValid = FALSE;
BOOLEAN fgEntryRepetetion = FALSE;
BOOLEAN fgTableValid = TRUE;
/*Configuration Table Check */
for (i = 0; i < sizeof(g_rRlmPowerLimitConfiguration) / sizeof(COUNTRY_POWER_LIMIT_TABLE_CONFIGURATION); i++) {
/*Table Country Code */
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitConfiguration[i].aucCountryCode[0], &u2CountryCodeTable);
/*Repetition Entry Check */
for (j = i + 1;
j < ARRAY_SIZE(g_rRlmPowerLimitConfiguration);
j++) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitConfiguration[j].aucCountryCode[0], &u2CountryCodeCheck);
if (((g_rRlmPowerLimitConfiguration[i].ucCentralCh) ==
g_rRlmPowerLimitConfiguration[j].ucCentralCh)
&& (u2CountryCodeTable == u2CountryCodeCheck)) {
fgEntryRepetetion = TRUE;
DBGLOG(RLM, INFO, "Domain: Configuration Repetition CC=%c%c, Ch=%d\n",
g_rRlmPowerLimitConfiguration[i].aucCountryCode[0],
g_rRlmPowerLimitConfiguration[i].aucCountryCode[1],
g_rRlmPowerLimitConfiguration[i].ucCentralCh);
}
}
/*Channel Number Check */
fgChannelValid =
rlmDomainCheckChannelEntryValid(prAdapter, g_rRlmPowerLimitConfiguration[i].ucCentralCh);
/*Power Limit Check */
fgPowerLimitValid =
rlmDomainCheckPowerLimitValid(prAdapter, g_rRlmPowerLimitConfiguration[i], PWR_LIMIT_NUM);
if (fgChannelValid == FALSE || fgPowerLimitValid == FALSE) {
fgTableValid = FALSE;
DBGLOG(RLM, INFO, "Domain: CC=%c%c, Ch=%d, Limit: %d,%d,%d,%d,%d\n",
g_rRlmPowerLimitConfiguration[i].aucCountryCode[0],
g_rRlmPowerLimitConfiguration[i].aucCountryCode[1],
g_rRlmPowerLimitConfiguration[i].ucCentralCh,
g_rRlmPowerLimitConfiguration[i].aucPwrLimit[PWR_LIMIT_CCK],
g_rRlmPowerLimitConfiguration[i].aucPwrLimit[PWR_LIMIT_20M],
g_rRlmPowerLimitConfiguration[i].aucPwrLimit[PWR_LIMIT_40M],
g_rRlmPowerLimitConfiguration[i].aucPwrLimit[PWR_LIMIT_80M],
g_rRlmPowerLimitConfiguration[i].aucPwrLimit[PWR_LIMIT_160M]);
}
if (u2CountryCodeTable == COUNTRY_CODE_NULL) {
DBGLOG(RLM, INFO, "Domain: Full search down\n");
break; /*End of country table entry */
}
}
if (fgEntryRepetetion == FALSE)
DBGLOG(RLM, INFO, "Domain: Configuration Table no Repetiton.\n");
/*Configuration Table no error */
if (fgTableValid == TRUE)
prAdapter->fgIsPowerLimitTableValid = TRUE;
else
prAdapter->fgIsPowerLimitTableValid = FALSE;
/*Default Table Check */
fgEntryRepetetion = FALSE;
for (i = 0; i < sizeof(g_rRlmPowerLimitDefault) / sizeof(COUNTRY_POWER_LIMIT_TABLE_DEFAULT); i++) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitDefault[i].aucCountryCode[0], &u2CountryCodeTable);
for (j = i + 1; j < sizeof(g_rRlmPowerLimitDefault) / sizeof(COUNTRY_POWER_LIMIT_TABLE_DEFAULT); j++) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitDefault[j].aucCountryCode[0], &u2CountryCodeCheck);
if (u2CountryCodeTable == u2CountryCodeCheck) {
fgEntryRepetetion = TRUE;
DBGLOG(RLM, INFO,
"Domain: Default Repetition CC=%c%c\n",
g_rRlmPowerLimitDefault[j].aucCountryCode[0],
g_rRlmPowerLimitDefault[j].aucCountryCode[1]);
}
}
}
if (fgEntryRepetetion == FALSE)
DBGLOG(RLM, INFO, "Domain: Default Table no Repetiton.\n");
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (u2TableIndex) : if 0xFFFF -> No Table Match
*/
/*----------------------------------------------------------------------------*/
UINT_16 rlmDomainPwrLimitDefaultTableDecision(P_ADAPTER_T prAdapter, UINT_16 u2CountryCode)
{
UINT_16 i;
UINT_16 u2CountryCodeTable = COUNTRY_CODE_NULL;
UINT_16 u2TableIndex = POWER_LIMIT_TABLE_NULL; /* No Table Match */
/*Default Table Index */
for (i = 0; i < sizeof(g_rRlmPowerLimitDefault) / sizeof(COUNTRY_POWER_LIMIT_TABLE_DEFAULT); i++) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitDefault[i].aucCountryCode[0], &u2CountryCodeTable);
if (u2CountryCodeTable == u2CountryCode) {
u2TableIndex = i;
break; /*match country code */
} else if (u2CountryCodeTable == COUNTRY_CODE_NULL) {
u2TableIndex = i;
break; /*find last one country- Default */
}
}
DBGLOG(RLM, INFO, "Domain: Default Table Index = %d\n", u2TableIndex);
return u2TableIndex;
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainBuildCmdByDefaultTable(P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T prCmd, UINT_16 u2DefaultTableIndex)
{
UINT_8 i, k;
P_COUNTRY_POWER_LIMIT_TABLE_DEFAULT prPwrLimitSubBand;
P_CMD_CHANNEL_POWER_LIMIT prCmdPwrLimit;
prCmdPwrLimit = &prCmd->rChannelPowerLimit[0];
prPwrLimitSubBand = &g_rRlmPowerLimitDefault[u2DefaultTableIndex];
/*Build power limit cmd by default table information */
for (i = POWER_LIMIT_2G4; i < POWER_LIMIT_SUBAND_NUM; i++) {
if (prPwrLimitSubBand->aucPwrLimitSubBand[i] < MAX_TX_POWER) {
for (k = g_rRlmSubBand[i].ucStartCh; k <= g_rRlmSubBand[i].ucEndCh;
k += g_rRlmSubBand[i].ucInterval) {
if ((prPwrLimitSubBand->ucPwrUnit & BIT(i)) == 0) {
prCmdPwrLimit->ucCentralCh = k;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK,
prPwrLimitSubBand->aucPwrLimitSubBand[i], PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
} else {
/* ex: 40MHz power limit(mW\MHz) = 20MHz power limit(mW\MHz) * 2
* ---> 40MHz power limit(dBm) = 20MHz power limit(dBm) + 6;
*/
prCmdPwrLimit->ucCentralCh = k;
prCmdPwrLimit->cPwrLimitCCK = prPwrLimitSubBand->aucPwrLimitSubBand[i];
prCmdPwrLimit->cPwrLimit20 = prPwrLimitSubBand->aucPwrLimitSubBand[i];
prCmdPwrLimit->cPwrLimit40 = prPwrLimitSubBand->aucPwrLimitSubBand[i] + 6;
if (prCmdPwrLimit->cPwrLimit40 > MAX_TX_POWER)
prCmdPwrLimit->cPwrLimit40 = MAX_TX_POWER;
prCmdPwrLimit->cPwrLimit80 = prPwrLimitSubBand->aucPwrLimitSubBand[i] + 12;
if (prCmdPwrLimit->cPwrLimit80 > MAX_TX_POWER)
prCmdPwrLimit->cPwrLimit80 = MAX_TX_POWER;
prCmdPwrLimit->cPwrLimit160 = prPwrLimitSubBand->aucPwrLimitSubBand[i] + 18;
if (prCmdPwrLimit->cPwrLimit160 > MAX_TX_POWER)
prCmdPwrLimit->cPwrLimit160 = MAX_TX_POWER;
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
}
}
#if 0
if (prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_2G4] < MAX_TX_POWER) {
for (i = BAND_2G4_LOWER_BOUND; i <= BAND_2G4_UPPER_BOUND; i++) {
prCmdPwrLimit->ucCentralCh = i;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK, prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_2G4],
PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
if (prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII1] < MAX_TX_POWER) {
if (prCmd->u2CountryCode != COUNTRY_CODE_KR) {
for (i = UNII1_LOWER_BOUND; i <= UNII1_UPPER_BOUND; i += 2) {
prCmdPwrLimit->ucCentralCh = i;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK,
prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII1], PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
}
} else {
for (i = UNII1_LOWER_BOUND; i <= UNII1_UPPER_BOUND; i += 2) {
/* ex: 40MHz power limit(mW\MHz) = 20MHz power limit(mW\MHz) * 2
* ---> 40MHz power limit(dBm) = 20MHz power limit(dBm) + 6;
*/
prCmdPwrLimit->ucCentralCh = i;
prCmdPwrLimit->cPwrLimitCCK =
g_rRlmPowerLimitDefault[u2DefaultTableIndex].cPwrLimitUnii1;
prCmdPwrLimit->cPwrLimit20 =
g_rRlmPowerLimitDefault[u2DefaultTableIndex].cPwrLimitUnii1;
prCmdPwrLimit->cPwrLimit40 =
g_rRlmPowerLimitDefault[u2DefaultTableIndex].cPwrLimitUnii1 + 6;
prCmdPwrLimit->cPwrLimit80 =
g_rRlmPowerLimitDefault[u2DefaultTableIndex].cPwrLimitUnii1 + 12;
prCmdPwrLimit->cPwrLimit160 =
g_rRlmPowerLimitDefault[u2DefaultTableIndex].cPwrLimitUnii1 + 18;
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
}
if (prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII2A] < MAX_TX_POWER) {
for (i = UNII2A_LOWER_BOUND; i <= UNII2A_UPPER_BOUND; i += 2) {
prCmdPwrLimit->ucCentralCh = i;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK,
prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII2A], PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
if (prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII2C] < MAX_TX_POWER) {
for (i = UNII2C_LOWER_BOUND; i <= UNII2C_UPPER_BOUND; i += 2) {
prCmdPwrLimit->ucCentralCh = i;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK,
prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII2C], PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
if (prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII3] < MAX_TX_POWER) {
for (i = UNII3_LOWER_BOUND; i <= UNII3_UPPER_BOUND; i += 2) {
prCmdPwrLimit->ucCentralCh = i;
kalMemSet(&prCmdPwrLimit->cPwrLimitCCK,
prPwrLimitSubBand->aucPwrLimitSubBand[POWER_LIMIT_UNII3], PWR_LIMIT_NUM);
prCmdPwrLimit++;
prCmd->ucNum++;
}
}
#endif
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainBuildCmdByConfigTable(P_ADAPTER_T prAdapter, P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T prCmd)
{
UINT_8 i, k;
UINT_16 u2CountryCodeTable = COUNTRY_CODE_NULL;
P_CMD_CHANNEL_POWER_LIMIT prCmdPwrLimit;
BOOLEAN fgChannelValid;
/*Build power limit cmd by configuration table information */
for (i = 0; i < sizeof(g_rRlmPowerLimitConfiguration) / sizeof(COUNTRY_POWER_LIMIT_TABLE_CONFIGURATION); i++) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitConfiguration[i].aucCountryCode[0], &u2CountryCodeTable);
fgChannelValid =
rlmDomainCheckChannelEntryValid(prAdapter, g_rRlmPowerLimitConfiguration[i].ucCentralCh);
if (u2CountryCodeTable == COUNTRY_CODE_NULL) {
DBGLOG(RLM, INFO, "Domain: full search configuration table done.\n");
break; /*end of configuration table */
} else if ((u2CountryCodeTable == prCmd->u2CountryCode) && (fgChannelValid == TRUE)) {
prCmdPwrLimit = &prCmd->rChannelPowerLimit[0];
if (prCmd->ucNum != 0) {
for (k = 0; k < prCmd->ucNum; k++) {
if (prCmdPwrLimit->ucCentralCh ==
g_rRlmPowerLimitConfiguration[i].ucCentralCh) {
/*Cmd setting (Default table information) and
* Configuration table has repetition channel entry,
* ex : Default table (ex: 2.4G, limit = 20dBm) -->
* ch1~14 limit =20dBm,
* Configuration table (ex: ch1, limit = 22dBm) -->
* ch 1 = 22 dBm
* Cmd final setting --> ch1 = 22dBm, ch12~14 = 20dBm
*/
kalMemCopy(&prCmdPwrLimit->cPwrLimitCCK,
&g_rRlmPowerLimitConfiguration[i].aucPwrLimit,
PWR_LIMIT_NUM);
DBGLOG(RLM, INFO,
"Domain: CC=%c%c,ConfigCh=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n",
((prCmd->u2CountryCode & 0xff00) >> 8),
(prCmd->u2CountryCode & 0x00ff), prCmdPwrLimit->ucCentralCh,
prCmdPwrLimit->cPwrLimitCCK, prCmdPwrLimit->cPwrLimit20,
prCmdPwrLimit->cPwrLimit40, prCmdPwrLimit->cPwrLimit80,
prCmdPwrLimit->cPwrLimit160, prCmdPwrLimit->ucFlag);
break;
}
prCmdPwrLimit++;
}
if (k == prCmd->ucNum) {
/*Full search cmd (Default table setting) no match channey,
* ex : Default table (ex: 2.4G, limit = 20dBm) -->
* ch1~14 limit =20dBm,
* Configuration table (ex: ch36, limit = 22dBm) -->
* ch 36 = 22 dBm
* Cmd final setting --> ch1~14 = 20dBm, ch36= 22dBm
*/
kalMemCopy(&prCmdPwrLimit->cPwrLimitCCK,
&g_rRlmPowerLimitConfiguration[i].aucPwrLimit, PWR_LIMIT_NUM);
prCmd->ucNum++;
DBGLOG(RLM, INFO,
"Domain: Full CC=%c%c,ConfigCh=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n",
((prCmd->u2CountryCode & 0xff00) >> 8), (prCmd->u2CountryCode & 0x00ff),
prCmdPwrLimit->ucCentralCh, prCmdPwrLimit->cPwrLimitCCK,
prCmdPwrLimit->cPwrLimit20, prCmdPwrLimit->cPwrLimit40,
prCmdPwrLimit->cPwrLimit80, prCmdPwrLimit->cPwrLimit160,
prCmdPwrLimit->ucFlag);
}
} else {
/*Default table power limit value are 63--> cmd table no channel entry
* ex : Default table (ex: 2.4G, limit = 63Bm) --> no channel entry in cmd,
* Configuration table (ex: ch36, limit = 22dBm) --> ch 36 = 22 dBm
* Cmd final setting --> ch36= 22dBm
*/
prCmdPwrLimit->ucCentralCh = g_rRlmPowerLimitConfiguration[i].ucCentralCh;
kalMemCopy(&prCmdPwrLimit->cPwrLimitCCK, &g_rRlmPowerLimitConfiguration[i].aucPwrLimit,
PWR_LIMIT_NUM);
prCmd->ucNum++;
DBGLOG(RLM, INFO, "Domain: Default table power limit value are 63.\n");
DBGLOG(RLM, INFO, "Domain: CC=%c%c,ConfigCh=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n",
((prCmd->u2CountryCode & 0xff00) >> 8),
(prCmd->u2CountryCode & 0x00ff), prCmdPwrLimit->ucCentralCh,
prCmdPwrLimit->cPwrLimitCCK, prCmdPwrLimit->cPwrLimit20,
prCmdPwrLimit->cPwrLimit40, prCmdPwrLimit->cPwrLimit80,
prCmdPwrLimit->cPwrLimit160, prCmdPwrLimit->ucFlag);
}
}
}
}
/*----------------------------------------------------------------------------*/
/*!
* @brief
*
* @param[in]
*
* @return (none)
*/
/*----------------------------------------------------------------------------*/
VOID rlmDomainSendPwrLimitCmd_V2(P_ADAPTER_T prAdapter)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
WLAN_STATUS rStatus;
UINT_32 u4SetQueryInfoLen;
UINT_32 ch_cnt;
struct wiphy *wiphy;
u8 band_idx, ch_idx;
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T prCmd[KAL_NUM_BANDS] = {NULL};
UINT_32 u4SetCmdTableMaxSize[KAL_NUM_BANDS] = {0};
const INT_8 *prChannelList = NULL;
DBGLOG(RLM, INFO, "rlmDomainSendPwrLimitCmd()\n");
wiphy = priv_to_wiphy(prAdapter->prGlueInfo);
for (band_idx = 0; band_idx < KAL_NUM_BANDS; band_idx++) {
if (band_idx != KAL_BAND_2GHZ && band_idx != KAL_BAND_5GHZ)
continue;
prChannelList = (band_idx == KAL_BAND_2GHZ) ?
gTx_Pwr_Limit_2g_Ch : gTx_Pwr_Limit_5g_Ch;
ch_cnt = (band_idx == KAL_BAND_2GHZ) ? TX_PWR_LIMIT_2G_CH_NUM :
TX_PWR_LIMIT_5G_CH_NUM;
if (!ch_cnt)
continue;
u4SetCmdTableMaxSize[band_idx] =
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T) +
ch_cnt * sizeof(CMD_CHANNEL_POWER_LIMIT_V2);
/* compile time assertion on firmware command structs */
DATA_STRUCT_INSPECTING_ASSERT(sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T) == 8);
DATA_STRUCT_INSPECTING_ASSERT(sizeof(CMD_CHANNEL_POWER_LIMIT_V2) == 40);
DATA_STRUCT_INSPECTING_ASSERT(
OFFSET_OF(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T, rChannelPowerLimit) ==
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T));
DATA_STRUCT_INSPECTING_ASSERT(
OFFSET_OF(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T, rChannelPowerLimit[1]) ==
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T) + 1 * sizeof(CMD_CHANNEL_POWER_LIMIT_V2));
DATA_STRUCT_INSPECTING_ASSERT(
OFFSET_OF(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T, rChannelPowerLimit[2]) ==
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T) + 2 * sizeof(CMD_CHANNEL_POWER_LIMIT_V2));
prCmd[band_idx] = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, u4SetCmdTableMaxSize[band_idx]);
if (!prCmd[band_idx]) {
DBGLOG(RLM, ERROR, "Domain: no buf to send cmd\n");
goto error;
}
/*initialize tw pwr table*/
kalMemSet(prCmd[band_idx], MAX_TX_POWER, u4SetCmdTableMaxSize[band_idx]);
prCmd[band_idx]->ucNum = ch_cnt;
prCmd[band_idx]->eband = (band_idx == KAL_BAND_2GHZ) ? BAND_2G4 : BAND_5G;
prCmd[band_idx]->countryCode = rlmDomainGetCountryCode();
DBGLOG(RLM, INFO, "%s, active n_channels=%d, band=%d\n", __func__, ch_cnt, prCmd[band_idx]->eband);
for (ch_idx = 0; ch_idx < ch_cnt; ch_idx++) {
prCmd[band_idx]->rChannelPowerLimit[ch_idx].ucCentralCh =
prChannelList[ch_idx];
}
}
/*
* Get Max Tx Power from MT_TxPwrLimit.dat
*/
rlmDomainGetTxPwrLimit(rlmDomainGetCountryCode(),
prAdapter->prGlueInfo,
prCmd[KAL_BAND_2GHZ],
prCmd[KAL_BAND_5GHZ]);
for (band_idx = 0; band_idx < KAL_NUM_BANDS; band_idx++) {
UINT_8 ucRemainChNum, i, ucTempChNum, prCmdBatchNum;
UINT_32 u4BufSize = 0;
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T prTempCmd = NULL;
ENUM_BAND_T eBand = (band_idx == KAL_BAND_2GHZ) ?
BAND_2G4 : BAND_5G;
if (!prCmd[band_idx])
continue;
ucRemainChNum = prCmd[band_idx]->ucNum;
prCmdBatchNum =
(ucRemainChNum + TX_PWR_LIMIT_CMD_CH_NUM_THRESHOLD - 1) /
TX_PWR_LIMIT_CMD_CH_NUM_THRESHOLD;
for (i = 0; i < prCmdBatchNum; i++) {
if (i == prCmdBatchNum - 1)
ucTempChNum = ucRemainChNum;
else
ucTempChNum = TX_PWR_LIMIT_CMD_CH_NUM_THRESHOLD;
u4BufSize = sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_V2_T) +
ucTempChNum * sizeof(CMD_CHANNEL_POWER_LIMIT_V2);
prTempCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, u4BufSize);
if (!prTempCmd) {
DBGLOG(RLM, ERROR, "Domain: no buf to send cmd\n");
goto error;
}
/*copy partial tx pwr limit*/
prTempCmd->ucNum = ucTempChNum;
prTempCmd->eband = eBand;
prTempCmd->countryCode = rlmDomainGetCountryCode();
kalMemCopy(&prTempCmd->rChannelPowerLimit[0],
&prCmd[band_idx]->rChannelPowerLimit[i * TX_PWR_LIMIT_CMD_CH_NUM_THRESHOLD],
ucTempChNum * sizeof(CMD_CHANNEL_POWER_LIMIT_V2));
u4SetQueryInfoLen = u4BufSize;
/* Update tx max. power info to chip */
rStatus = wlanSendSetQueryCmd(prAdapter, /* prAdapter */
CMD_ID_SET_COUNTRY_POWER_LIMIT, /* ucCID */
TRUE, /* fgSetQuery */
FALSE, /* fgNeedResp */
FALSE, /* fgIsOid */
NULL, /* pfCmdDoneHandler */
NULL, /* pfCmdTimeoutHandler */
u4SetQueryInfoLen, /* u4SetQueryInfoLen */
(PUINT_8) prTempCmd, /* pucInfoBuffer */
NULL, /* pvSetQueryBuffer */
0 /* u4SetQueryBufferLen */
);
cnmMemFree(prAdapter, prTempCmd);
ucRemainChNum -= ucTempChNum;
}
}
error:
for (band_idx = 0; band_idx < KAL_NUM_BANDS; band_idx++) {
if (prCmd[band_idx])
cnmMemFree(prAdapter, prCmd[band_idx]);
}
#endif
}
VOID rlmDomainSendPwrLimitCmd(P_ADAPTER_T prAdapter)
{
P_CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T prCmd;
WLAN_STATUS rStatus;
UINT_8 i;
UINT_16 u2DefaultTableIndex;
UINT_32 u4SetCmdTableMaxSize;
UINT_32 u4SetQueryInfoLen;
P_CMD_CHANNEL_POWER_LIMIT prCmdPwrLimit; /* for print usage */
if (regd_is_single_sku_en())
return rlmDomainSendPwrLimitCmd_V2(prAdapter);
u4SetCmdTableMaxSize =
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T) +
MAX_CMD_SUPPORT_CHANNEL_NUM * sizeof(CMD_CHANNEL_POWER_LIMIT);
prCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, u4SetCmdTableMaxSize);
if (!prCmd) {
DBGLOG(RLM, ERROR, "Domain: no buf to send cmd\n");
return;
}
kalMemZero(prCmd, u4SetCmdTableMaxSize);
u2DefaultTableIndex =
rlmDomainPwrLimitDefaultTableDecision(prAdapter, prAdapter->rWifiVar.rConnSettings.u2CountryCode);
if (u2DefaultTableIndex != POWER_LIMIT_TABLE_NULL) {
WLAN_GET_FIELD_BE16(&g_rRlmPowerLimitDefault[u2DefaultTableIndex].aucCountryCode[0],
&prCmd->u2CountryCode);
prCmd->ucNum = 0;
if (prCmd->u2CountryCode != COUNTRY_CODE_NULL) {
/*Command - default table information */
rlmDomainBuildCmdByDefaultTable(prCmd, u2DefaultTableIndex);
/*Command - configuration table information */
rlmDomainBuildCmdByConfigTable(prAdapter, prCmd);
}
}
#if 0
u4SetCmdTableMaxSize =
sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T) +
MAX_CMD_SUPPORT_CHANNEL_NUM * sizeof(CMD_CHANNEL_POWER_LIMIT);
prCmd = cnmMemAlloc(prAdapter, RAM_TYPE_BUF, u4SetCmdTableMaxSize);
ASSERT(prCmd);
/* To do: exception handle */
if (!prCmd) {
DBGLOG(RLM, ERROR, "Domain: no buf to send cmd\n");
return;
}
kalMemZero(prCmd, u4SetCmdTableMaxSize); /* TODO memzero */
if (u2TableIndex != POWER_LIMIT_TABLE_NULL && u2TableIndex < MAX_DEFAULT_TABLE_COUNTRY_NUM) {
prCmd->u2CountryCode = (((UINT_16) g_rRlmCountryPowerLimitTable[u2TableIndex].aucCountryCode[0]) << 8) |
(((UINT_16) g_rRlmCountryPowerLimitTable[u2TableIndex].aucCountryCode[1]) & BITS(0, 7));
prChPwrLimit = &g_rRlmCountryPowerLimitTable[u2TableIndex].rChannelPowerLimit[0];
prCmdPwrLimit = &prCmd->rChannelPowerLimit[0];
prCmd->ucNum = 0;
for (i = 0; i < MAX_CMD_SUPPORT_CHANNEL_NUM; i++) {
if (prChPwrLimit->ucCentralCh != ENDCH) {
/*Check Power limit table channel efficient or not */
fgChannelValid = rlmDomainCheckChannelEntryValid(prAdapter, prChPwrLimit->ucCentralCh);
/*Cmd set up */
if (fgChannelValid) {
kalMemCopy(prCmdPwrLimit, prChPwrLimit, sizeof(CMD_CHANNEL_POWER_LIMIT));
DBGLOG(RLM, INFO,
"Domain: ValidCh=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n",
prCmdPwrLimit->ucCentralCh, prCmdPwrLimit->cPwrLimitCCK,
prCmdPwrLimit->cPwrLimit20, prCmdPwrLimit->cPwrLimit40,
prCmdPwrLimit->cPwrLimit80, prCmdPwrLimit->cPwrLimit160,
prCmdPwrLimit->ucFlag);
prCmd->ucNum++;
prCmdPwrLimit++;
} else {
DBGLOG(RLM, INFO,
"Domain: Non-Ch=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n",
prChPwrLimit->ucCentralCh, prChPwrLimit->cPwrLimitCCK,
prChPwrLimit->cPwrLimit20, prChPwrLimit->cPwrLimit40,
prChPwrLimit->cPwrLimit80, prChPwrLimit->cPwrLimit160,
prChPwrLimit->ucFlag);
}
prChPwrLimit++;
} else {
/*End of the chanel entry */
break;
}
};
}
#endif
if (prCmd->u2CountryCode != 0) {
DBGLOG(RLM, INFO,
"Domain: ValidCC =%c%c, ChNum=%d\n", ((prCmd->u2CountryCode & 0xff00) >> 8),
(prCmd->u2CountryCode & 0x00ff), prCmd->ucNum);
} else {
DBGLOG(RLM, INFO, "Domain: ValidCC =0x%04x, ChNum=%d\n", prCmd->u2CountryCode, prCmd->ucNum);
}
prCmdPwrLimit = &prCmd->rChannelPowerLimit[0];
for (i = 0; i < prCmd->ucNum; i++) {
DBGLOG(RLM, INFO, "Domain: Ch=%d,Limit=%d,%d,%d,%d,%d,Fg=%d\n", prCmdPwrLimit->ucCentralCh,
prCmdPwrLimit->cPwrLimitCCK, prCmdPwrLimit->cPwrLimit20, prCmdPwrLimit->cPwrLimit40,
prCmdPwrLimit->cPwrLimit80, prCmdPwrLimit->cPwrLimit160, prCmdPwrLimit->ucFlag);
prCmdPwrLimit++;
}
u4SetQueryInfoLen =
(sizeof(CMD_SET_COUNTRY_CHANNEL_POWER_LIMIT_T) + (prCmd->ucNum) * sizeof(CMD_CHANNEL_POWER_LIMIT));
/* Update domain info to chip */
if (prCmd->ucNum <= MAX_CMD_SUPPORT_CHANNEL_NUM) {
rStatus = wlanSendSetQueryCmd(prAdapter, /* prAdapter */
CMD_ID_SET_COUNTRY_POWER_LIMIT, /* ucCID */
TRUE, /* fgSetQuery */
FALSE, /* fgNeedResp */
FALSE, /* fgIsOid */
NULL, /* pfCmdDoneHandler */
NULL, /* pfCmdTimeoutHandler */
u4SetQueryInfoLen, /* u4SetQueryInfoLen */
(PUINT_8) prCmd, /* pucInfoBuffer */
NULL, /* pvSetQueryBuffer */
0 /* u4SetQueryBufferLen */
);
} else {
DBGLOG(RLM, INFO, "Domain: illegal power limit table");
}
/* ASSERT(rStatus == WLAN_STATUS_PENDING); */
cnmMemFree(prAdapter, prCmd);
}
#endif
BOOLEAN regd_is_single_sku_en(void)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
return g_mtk_regd_control.en;
#else
return FALSE;
#endif
}
#if (CFG_SUPPORT_SINGLE_SKU == 1)
BOOLEAN rlmDomainIsCtrlStateEqualTo(enum regd_state state)
{
return (g_mtk_regd_control.state == state) ? TRUE : FALSE;
}
enum regd_state rlmDomainGetCtrlState(void)
{
return g_mtk_regd_control.state;
}
void rlmDomainResetActiveChannel(void)
{
g_mtk_regd_control.n_channel_active_2g = 0;
g_mtk_regd_control.n_channel_active_5g = 0;
}
void rlmDomainAddActiveChannel(u8 band)
{
if (band == KAL_BAND_2GHZ)
g_mtk_regd_control.n_channel_active_2g += 1;
else if (band == KAL_BAND_5GHZ)
g_mtk_regd_control.n_channel_active_5g += 1;
}
u8 rlmDomainGetActiveChannelCount(u8 band)
{
if (band == KAL_BAND_2GHZ)
return g_mtk_regd_control.n_channel_active_2g;
else if (band == KAL_BAND_5GHZ)
return g_mtk_regd_control.n_channel_active_5g;
else
return 0;
}
struct channel *rlmDomainGetActiveChannels(void)
{
return g_mtk_regd_control.channels;
}
void rlmDomainSetDefaultCountryCode(void)
{
g_mtk_regd_control.alpha2 = COUNTRY_CODE_WW;
}
void rlmDomainResetCtrlInfo(void)
{
if (g_mtk_regd_control.state == REGD_STATE_UNDEFINED) {
memset(&g_mtk_regd_control, 0, sizeof(mtk_regd_control));
g_mtk_regd_control.state = REGD_STATE_INIT;
rlmDomainSetDefaultCountryCode();
#if (CFG_SUPPORT_SINGLE_SKU_LOCAL_DB == 1)
g_mtk_regd_control.flag |= REGD_CTRL_FLAG_SUPPORT_LOCAL_REGD_DB;
#endif
}
}
BOOLEAN rlmDomainIsUsingLocalRegDomainDataBase(void)
{
#if (CFG_SUPPORT_SINGLE_SKU_LOCAL_DB == 1)
return (g_mtk_regd_control.flag & REGD_CTRL_FLAG_SUPPORT_LOCAL_REGD_DB) ? TRUE : FALSE;
#else
return FALSE;
#endif
}
bool rlmDomainIsSameCountryCode(char *alpha2, u8 size_of_alpha2)
{
u8 idx;
u32 alpha2_hex = 0;
for (idx = 0; idx < size_of_alpha2; idx++)
alpha2_hex |= (alpha2[idx] << (idx * 8));
return (rlmDomainGetCountryCode() == alpha2_hex) ? TRUE : FALSE;
}
void rlmDomainSetCountryCode(char *alpha2, u8 size_of_alpha2)
{
u8 max;
u8 buf_size;
buf_size = sizeof(g_mtk_regd_control.alpha2);
max = (buf_size < size_of_alpha2) ? buf_size : size_of_alpha2;
g_mtk_regd_control.alpha2 = rlmDomainAlpha2ToU32(alpha2, max);
}
void rlmDomainSetDfsRegion(enum nl80211_dfs_regions dfs_region)
{
g_mtk_regd_control.dfs_region = dfs_region;
}
enum nl80211_dfs_regions rlmDomainGetDfsRegion(void)
{
return g_mtk_regd_control.dfs_region;
}
void rlmDomainSetTempCountryCode(char *alpha2, u8 size_of_alpha2)
{
u8 idx, max;
u8 buf_size;
buf_size = sizeof(g_mtk_regd_control.tmp_alpha2);
max = (buf_size < size_of_alpha2) ? buf_size : size_of_alpha2;
g_mtk_regd_control.tmp_alpha2 = 0;
for (idx = 0; idx < max; idx++)
g_mtk_regd_control.tmp_alpha2 |= (alpha2[idx] << (idx * 8));
}
enum regd_state rlmDomainStateTransition(enum regd_state request_state, struct regulatory_request *pRequest)
{
enum regd_state next_state, old_state;
bool the_same = 0;
old_state = g_mtk_regd_control.state;
next_state = REGD_STATE_INVALID;
if (old_state == REGD_STATE_INVALID)
DBGLOG(RLM, ERROR, "%s(): invalid state. trasntion is not allowed.\n", __func__);
switch (request_state) {
case REGD_STATE_SET_WW_CORE:
if ((old_state == REGD_STATE_SET_WW_CORE) || (old_state == REGD_STATE_INIT)
|| old_state == REGD_STATE_SET_COUNTRY_USER
|| old_state == REGD_STATE_SET_COUNTRY_IE)
next_state = request_state;
break;
case REGD_STATE_SET_COUNTRY_USER:
/* Allow user to set multiple times */
if ((old_state == REGD_STATE_SET_WW_CORE) || (old_state == REGD_STATE_INIT)
|| old_state == REGD_STATE_SET_COUNTRY_USER
|| old_state == REGD_STATE_SET_COUNTRY_IE)
next_state = request_state;
else
DBGLOG(RLM, ERROR, "Invalid old state = %d\n", old_state);
break;
case REGD_STATE_SET_COUNTRY_DRIVER:
if (old_state == REGD_STATE_SET_COUNTRY_USER) {
/*
* Error.
* Mixing using set_country_by_user and set_country_by_driver
* is not allowed.
*/
break;
}
next_state = request_state;
break;
case REGD_STATE_SET_COUNTRY_IE:
next_state = request_state;
break;
default:
break;
}
if (next_state == REGD_STATE_INVALID) {
DBGLOG(RLM, ERROR, "%s(): ERROR. trasntion to invalid state. o=%x, r=%x, s=%x\n",
__func__, old_state, request_state, the_same);
} else
DBGLOG(RLM, INFO, "%s(): trasntion to state = %x (old = %x)\n",
__func__, next_state, g_mtk_regd_control.state);
g_mtk_regd_control.state = next_state;
return g_mtk_regd_control.state;
}
void rlmDomainSetRefWiphy(struct wiphy *pWiphy)
{
g_mtk_regd_control.pRefWiphy = pWiphy;
}
struct wiphy *rlmDomainGetRefWiphy(void)
{
return g_mtk_regd_control.pRefWiphy;
}
/**
* rlmDomainChannelFlagString - Transform channel flags to readable string
*
* @ flags: the ieee80211_channel->flags for a channel
* @ buf: string buffer to put the transformed string
* @ buf_size: size of the buf
**/
void rlmDomainChannelFlagString(u32 flags, char *buf, size_t buf_size)
{
INT_32 buf_written = 0;
if (!flags || !buf || !buf_size)
return;
if (flags & IEEE80211_CHAN_DISABLED) {
LOGBUF(buf, ((INT_32)buf_size), buf_written, "DISABLED ");
/* If DISABLED, don't need to check other flags */
return;
}
if (flags & IEEE80211_CHAN_PASSIVE_FLAG)
LOGBUF(buf, ((INT_32)buf_size), buf_written, IEEE80211_CHAN_PASSIVE_STR " ");
if (flags & IEEE80211_CHAN_RADAR)
LOGBUF(buf, ((INT_32)buf_size), buf_written, "RADAR ");
if (flags & IEEE80211_CHAN_NO_HT40PLUS)
LOGBUF(buf, ((INT_32)buf_size), buf_written, "NO_HT40PLUS ");
if (flags & IEEE80211_CHAN_NO_HT40MINUS)
LOGBUF(buf, ((INT_32)buf_size), buf_written, "NO_HT40MINUS ");
if (flags & IEEE80211_CHAN_NO_80MHZ)
LOGBUF(buf, ((INT_32)buf_size), buf_written, "NO_80MHZ ");
if (flags & IEEE80211_CHAN_NO_160MHZ)
LOGBUF(buf, ((INT_32)buf_size), buf_written, "NO_160MHZ ");
}
void rlmDomainParsingChannel(IN struct wiphy *pWiphy)
{
u32 band_idx, ch_idx;
u32 ch_count;
struct ieee80211_supported_band *sband;
struct ieee80211_channel *chan;
struct channel *pCh;
char chan_flag_string[64] = {0};
P_GLUE_INFO_T prGlueInfo;
UINT_8 ucChannelNum = 0;
BOOLEAN fgDisconnection = FALSE;
WLAN_STATUS rStatus;
UINT_32 u4BufLen;
if (!pWiphy) {
DBGLOG(RLM, ERROR, "%s(): ERROR. pWiphy = NULL.\n", __func__);
ASSERT(0);
return;
}
/* Retrieve connected channel */
prGlueInfo = rlmDomainGetGlueInfo();
if (prGlueInfo && kalGetMediaStateIndicated(prGlueInfo) == PARAM_MEDIA_STATE_CONNECTED) {
ucChannelNum = wlanGetChannelNumberByNetwork(prGlueInfo->prAdapter,
prGlueInfo->prAdapter->prAisBssInfo->ucBssIndex);
}
/*
* Ready to parse the channel for bands
*/
rlmDomainResetActiveChannel();
rlmDomainSetRefWiphy(pWiphy);
ch_count = 0;
for (band_idx = 0; band_idx < KAL_NUM_BANDS; band_idx++) {
sband = pWiphy->bands[band_idx];
if (!sband)
continue;
for (ch_idx = 0; ch_idx < sband->n_channels; ch_idx++) {
chan = &sband->channels[ch_idx];
pCh = (rlmDomainGetActiveChannels() + ch_count);
/* Parse flags and get readable string */
rlmDomainChannelFlagString(chan->flags, chan_flag_string, sizeof(chan_flag_string));
if (chan->flags & IEEE80211_CHAN_DISABLED) {
DBGLOG(RLM, INFO, "channels[%d][%d]: ch%d (freq = %d) flags=0x%x [ %s]\n",
band_idx, ch_idx, chan->hw_value, chan->center_freq, chan->flags,
chan_flag_string);
/* Disconnect AP in the end of this function*/
if (chan->hw_value == ucChannelNum)
fgDisconnection = TRUE;
continue;
}
/* Allowable channel */
if (ch_count == MAX_SUPPORTED_CH_COUNT) {
DBGLOG(RLM, ERROR, "%s(): no buffer to store channel information.\n", __func__);
break;
}
rlmDomainAddActiveChannel(band_idx);
DBGLOG(RLM, INFO, "channels[%d][%d]: ch%d (freq = %d) flgs=0x%x [ %s]\n",
band_idx, ch_idx, chan->hw_value, chan->center_freq, chan->flags,
chan_flag_string);
pCh->chNum = chan->hw_value;
pCh->flags = chan->flags;
ch_count += 1;
}
}
/* Disconnect with AP if connected channel is disabled in new country */
if (fgDisconnection) {
DBGLOG(RLM, STATE, "Disconnect! CH%d is DISABLED in this country\n", ucChannelNum);
rStatus = kalIoctl(prGlueInfo, wlanoidSetDisassociate, NULL, 0, FALSE, FALSE, TRUE, &u4BufLen);
if (rStatus != WLAN_STATUS_SUCCESS)
DBGLOG(RLM, WARN, "disassociate error:%lx\n", rStatus);
}
}
void rlmExtractChannelInfo(u32 max_ch_count, struct acctive_channel_list *prBuff)
{
u32 ch_count, idx;
struct channel *pCh;
prBuff->n_channels_2g = rlmDomainGetActiveChannelCount(KAL_BAND_2GHZ);
prBuff->n_channels_5g = rlmDomainGetActiveChannelCount(KAL_BAND_5GHZ);
ch_count = prBuff->n_channels_2g + prBuff->n_channels_5g;
if (ch_count > max_ch_count) {
ch_count = max_ch_count;
DBGLOG(RLM, WARN, "%s(); active channel list is not a complete one.\n", __func__);
}
for (idx = 0; idx < ch_count; idx++) {
pCh = &(prBuff->channels[idx]);
pCh->chNum = (rlmDomainGetActiveChannels() + idx)->chNum;
pCh->flags = (rlmDomainGetActiveChannels() + idx)->flags;
}
}
const struct ieee80211_regdomain *rlmDomainSearchRegdomainFromLocalDataBase(char *alpha2)
{
#if (CFG_SUPPORT_SINGLE_SKU_LOCAL_DB == 1)
u8 idx;
const struct mtk_regdomain *prRegd;
idx = 0;
while (g_prRegRuleTable[idx]) {
prRegd = g_prRegRuleTable[idx];
if ((prRegd->country_code[0] == alpha2[0]) &&
(prRegd->country_code[1] == alpha2[1]) &&
(prRegd->country_code[2] == alpha2[2]) &&
(prRegd->country_code[3] == alpha2[3]))
return prRegd->prRegdRules;
idx++;
}
DBGLOG(RLM, ERROR, "%s(): Error, Cannot find the correct RegDomain. country = %s.\n",
__func__, alpha2);
DBGLOG(RLM, INFO, " Set as default WW.\n");
return &default_regdom_ww; /*default world wide*/
#else
return NULL;
#endif
}
const struct ieee80211_regdomain *rlmDomainGetLocalDefaultRegd(void)
{
#if (CFG_SUPPORT_SINGLE_SKU_LOCAL_DB == 1)
return &default_regdom_ww;
#else
return NULL;
#endif
}
P_GLUE_INFO_T rlmDomainGetGlueInfo(void)
{
return g_mtk_regd_control.pGlueInfo;
}
bool rlmDomainIsEfuseUsed(void)
{
return g_mtk_regd_control.isEfuseCountryCodeUsed;
}
#endif
WLAN_STATUS rlmDomainExtractSingleSkuInfoFromFirmware(IN P_ADAPTER_T prAdapter, IN PUINT_8 pucEventBuf)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
P_SINGLE_SKU_INFO prSkuInfo = (P_SINGLE_SKU_INFO)pucEventBuf;
g_mtk_regd_control.en = TRUE;
if (prSkuInfo->isEfuseValid) {
if (!rlmDomainIsUsingLocalRegDomainDataBase()) {
DBGLOG(RLM, ERROR, "%s(): Error. In efuse mode, must use local data base.\n", __func__);
ASSERT(0);
return WLAN_STATUS_NOT_SUPPORTED; /*force using local db if getting country code from efuse*/
}
rlmDomainSetCountryCode((char *)&prSkuInfo->u4EfuseCountryCode, sizeof(prSkuInfo->u4EfuseCountryCode));
g_mtk_regd_control.isEfuseCountryCodeUsed = TRUE;
}
#endif
return WLAN_STATUS_SUCCESS;
}
void rlmDomainSendInfoToFirmware(IN P_ADAPTER_T prAdapter)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
struct regulatory_request request;
struct regulatory_request *prReq = NULL;
if (!regd_is_single_sku_en())
return; /*not support single sku*/
if (g_mtk_regd_control.isEfuseCountryCodeUsed) {
request.initiator = NL80211_REGDOM_SET_BY_DRIVER;
prReq = &request;
}
g_mtk_regd_control.pGlueInfo = prAdapter->prGlueInfo;
mtk_reg_notify(priv_to_wiphy(prAdapter->prGlueInfo), prReq);
#endif
}
ENUM_CHNL_EXT_T rlmSelectSecondaryChannelType(P_ADAPTER_T prAdapter, ENUM_BAND_T band, u8 primary_ch)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
u8 below_ch, above_ch;
below_ch = primary_ch - CHNL_SPAN_20;
above_ch = primary_ch + CHNL_SPAN_20;
if (rlmDomainIsLegalChannel(prAdapter, band, above_ch))
return CHNL_EXT_SCA;
if (rlmDomainIsLegalChannel(prAdapter, band, below_ch))
return CHNL_EXT_SCB;
#endif
return CHNL_EXT_SCN;
}
void rlmDomainOidSetCountry(IN P_ADAPTER_T prAdapter, char *country, u8 size_of_country)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
struct regulatory_request request;
if (rlmDomainIsUsingLocalRegDomainDataBase()) {
rlmDomainSetTempCountryCode(country, size_of_country);
request.initiator = NL80211_REGDOM_SET_BY_DRIVER;
mtk_reg_notify(priv_to_wiphy(prAdapter->prGlueInfo), &request);
} else {
DBGLOG(RLM, INFO, "%s(): Using driver hint to query CRDA getting regd.\n", __func__);
regulatory_hint(priv_to_wiphy(prAdapter->prGlueInfo), country);
}
#endif
}
u32 rlmDomainGetCountryCode(void)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
return g_mtk_regd_control.alpha2;
#else
return 0;
#endif
}
u32 rlmDomainGetTempCountryCode(void)
{
#if (CFG_SUPPORT_SINGLE_SKU == 1)
return g_mtk_regd_control.tmp_alpha2;
#else
return 0;
#endif
}
void rlmDomainAssert(BOOLEAN cond)
{
/* bypass this check because single sku is not enable */
if (!regd_is_single_sku_en())
return;
if (!cond) {
WARN_ON(1);
DBGLOG(RLM, ERROR, "[WARNING!!] RLM unexpected case.\n");
}
}