2018-06-22 16:48:32 +00:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Copyright(c) 2009-2010 - 2017 Realtek Corporation.
|
|
|
|
*
|
|
|
|
* 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 the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
#include <drv_types.h>
|
|
|
|
|
|
|
|
#ifdef CONFIG_IOCTL_CFG80211
|
|
|
|
|
|
|
|
#include <rtw_wifi_regd.h>
|
|
|
|
|
|
|
|
static struct country_code_to_enum_rd allCountries[] = {
|
|
|
|
{COUNTRY_CODE_USER, "RD"},
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* REG_RULE(freq start, freq end, bandwidth, max gain, eirp, reg_flags)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
*Only these channels all allow active
|
|
|
|
*scan on all world regulatory domains
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* 2G chan 01 - chan 11 */
|
|
|
|
#define RTW_2GHZ_CH01_11 \
|
|
|
|
REG_RULE(2412-10, 2462+10, 40, 0, 20, 0)
|
|
|
|
|
|
|
|
/*
|
|
|
|
*We enable active scan on these a case
|
|
|
|
*by case basis by regulatory domain
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* 2G chan 12 - chan 13, PASSIV SCAN */
|
|
|
|
#define RTW_2GHZ_CH12_13 \
|
2019-05-24 19:43:57 +00:00
|
|
|
REG_RULE(2467-10, 2472+10, 40, 0, 20, \
|
2019-06-08 06:35:18 +00:00
|
|
|
0)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* 2G chan 14, PASSIVS SCAN, NO OFDM (B only) */
|
|
|
|
#define RTW_2GHZ_CH14 \
|
2019-05-24 19:43:57 +00:00
|
|
|
REG_RULE(2484-10, 2484+10, 40, 0, 20, \
|
|
|
|
NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_OFDM)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* 5G chan 36 - chan 64 */
|
|
|
|
#define RTW_5GHZ_5150_5350 \
|
2018-08-24 20:52:34 +00:00
|
|
|
REG_RULE(5150-10, 5350+10, 40, 0, 30, \
|
|
|
|
NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* 5G chan 100 - chan 165 */
|
|
|
|
#define RTW_5GHZ_5470_5850 \
|
2018-08-24 20:52:34 +00:00
|
|
|
REG_RULE(5470-10, 5850+10, 40, 0, 30, \
|
|
|
|
NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* 5G chan 149 - chan 165 */
|
|
|
|
#define RTW_5GHZ_5725_5850 \
|
2018-08-24 20:52:34 +00:00
|
|
|
REG_RULE(5725-10, 5850+10, 40, 0, 30, \
|
|
|
|
NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* 5G chan 36 - chan 165 */
|
|
|
|
#define RTW_5GHZ_5150_5850 \
|
2018-08-24 20:52:34 +00:00
|
|
|
REG_RULE(5150-10, 5850+10, 40, 0, 30, \
|
2019-06-08 06:35:18 +00:00
|
|
|
0)
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
static const struct ieee80211_regdomain rtw_regdom_rd = {
|
2018-08-24 20:52:34 +00:00
|
|
|
.n_reg_rules = 3,
|
2018-06-22 16:48:32 +00:00
|
|
|
.alpha2 = "99",
|
|
|
|
.reg_rules = {
|
|
|
|
RTW_2GHZ_CH01_11,
|
|
|
|
RTW_2GHZ_CH12_13,
|
|
|
|
RTW_5GHZ_5150_5850,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
void rtw_regd_apply_flags(struct wiphy *wiphy)
|
2018-06-22 16:48:32 +00:00
|
|
|
{
|
2019-05-24 19:43:57 +00:00
|
|
|
struct dvobj_priv *dvobj = wiphy_to_dvobj(wiphy);
|
|
|
|
struct rf_ctl_t *rfctl = dvobj_to_rfctl(dvobj);
|
2018-08-24 20:52:34 +00:00
|
|
|
RT_CHANNEL_INFO *channel_set = rfctl->channel_set;
|
|
|
|
u8 max_chan_nums = rfctl->max_chan_nums;
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
struct ieee80211_supported_band *sband;
|
|
|
|
struct ieee80211_channel *ch;
|
|
|
|
unsigned int i, j;
|
|
|
|
u16 channel;
|
2018-08-24 20:52:34 +00:00
|
|
|
u32 freq;
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
/* all channels disable */
|
|
|
|
for (i = 0; i < NUM_NL80211_BANDS; i++) {
|
|
|
|
sband = wiphy->bands[i];
|
|
|
|
|
|
|
|
if (sband) {
|
|
|
|
for (j = 0; j < sband->n_channels; j++) {
|
|
|
|
ch = &sband->channels[j];
|
|
|
|
|
|
|
|
if (ch)
|
2019-06-08 06:35:18 +00:00
|
|
|
#ifndef CONFIG_DISABLE_REGD_C
|
2018-08-24 20:52:34 +00:00
|
|
|
ch->flags = IEEE80211_CHAN_DISABLED;
|
2019-06-08 06:35:18 +00:00
|
|
|
#else
|
|
|
|
ch->flags = 0;
|
|
|
|
#endif
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-06-08 06:35:18 +00:00
|
|
|
#ifndef CONFIG_DISABLE_REGD_C
|
2018-06-22 16:48:32 +00:00
|
|
|
/* channels apply by channel plans. */
|
|
|
|
for (i = 0; i < max_chan_nums; i++) {
|
|
|
|
channel = channel_set[i].ChannelNum;
|
|
|
|
freq = rtw_ch2freq(channel);
|
|
|
|
|
|
|
|
ch = ieee80211_get_channel(wiphy, freq);
|
2019-05-24 19:43:57 +00:00
|
|
|
if (!ch)
|
|
|
|
continue;
|
2018-06-22 16:48:32 +00:00
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
if (channel_set[i].ScanType == SCAN_PASSIVE
|
|
|
|
#if defined(CONFIG_DFS_MASTER)
|
|
|
|
&& rtw_odm_dfs_domain_unknown(dvobj)
|
|
|
|
#endif
|
|
|
|
) {
|
|
|
|
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 14, 0))
|
|
|
|
ch->flags = (IEEE80211_CHAN_NO_IBSS | IEEE80211_CHAN_PASSIVE_SCAN);
|
|
|
|
#else
|
|
|
|
ch->flags = IEEE80211_CHAN_NO_IR;
|
|
|
|
#endif
|
|
|
|
} else
|
|
|
|
ch->flags = 0;
|
2018-06-22 16:48:32 +00:00
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
#ifdef CONFIG_DFS
|
|
|
|
if (rtw_is_dfs_ch(ch->hw_value)
|
|
|
|
#if defined(CONFIG_DFS_MASTER)
|
|
|
|
&& rtw_odm_dfs_domain_unknown(dvobj)
|
|
|
|
#endif
|
|
|
|
) {
|
|
|
|
ch->flags |= IEEE80211_CHAN_RADAR;
|
|
|
|
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 14, 0))
|
|
|
|
ch->flags |= (IEEE80211_CHAN_NO_IBSS | IEEE80211_CHAN_PASSIVE_SCAN);
|
|
|
|
#else
|
|
|
|
ch->flags |= IEEE80211_CHAN_NO_IR;
|
|
|
|
#endif
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
2019-05-24 19:43:57 +00:00
|
|
|
#endif /* CONFIG_DFS */
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
2019-06-08 06:35:18 +00:00
|
|
|
#endif
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct ieee80211_regdomain *_rtw_regdomain_select(struct
|
|
|
|
rtw_regulatory
|
|
|
|
*reg)
|
|
|
|
{
|
|
|
|
return &rtw_regdom_rd;
|
|
|
|
}
|
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
static void rtw_reg_notifier(struct wiphy *wiphy, struct regulatory_request *request)
|
2018-06-22 16:48:32 +00:00
|
|
|
{
|
2019-05-24 19:43:57 +00:00
|
|
|
switch (request->initiator) {
|
|
|
|
case NL80211_REGDOM_SET_BY_DRIVER:
|
|
|
|
RTW_INFO("%s: %s\n", __func__, "NL80211_REGDOM_SET_BY_DRIVER");
|
|
|
|
break;
|
|
|
|
case NL80211_REGDOM_SET_BY_CORE:
|
|
|
|
RTW_INFO("%s: %s\n", __func__, "NL80211_REGDOM_SET_BY_CORE");
|
|
|
|
break;
|
|
|
|
case NL80211_REGDOM_SET_BY_USER:
|
|
|
|
RTW_INFO("%s: %s alpha2:%c%c\n", __func__, "NL80211_REGDOM_SET_BY_USER"
|
|
|
|
, request->alpha2[0], request->alpha2[1]);
|
|
|
|
rtw_set_country(wiphy_to_adapter(wiphy), request->alpha2);
|
|
|
|
break;
|
|
|
|
case NL80211_REGDOM_SET_BY_COUNTRY_IE:
|
|
|
|
RTW_INFO("%s: %s\n", __func__, "NL80211_REGDOM_SET_BY_COUNTRY_IE");
|
|
|
|
break;
|
|
|
|
}
|
2018-06-22 16:48:32 +00:00
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
rtw_regd_apply_flags(wiphy);
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 9, 0))
|
2019-05-24 19:43:57 +00:00
|
|
|
static int rtw_reg_notifier_return(struct wiphy *wiphy, struct regulatory_request *request)
|
2018-06-22 16:48:32 +00:00
|
|
|
{
|
2019-05-24 19:43:57 +00:00
|
|
|
rtw_reg_notifier(wiphy, request);
|
2018-06-22 16:48:32 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2019-05-24 19:43:57 +00:00
|
|
|
#endif
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
static void _rtw_regd_init_wiphy(struct rtw_regulatory *reg, struct wiphy *wiphy)
|
|
|
|
{
|
|
|
|
const struct ieee80211_regdomain *regd;
|
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 9, 0))
|
|
|
|
wiphy->reg_notifier = rtw_reg_notifier_return;
|
|
|
|
#else
|
2018-06-22 16:48:32 +00:00
|
|
|
wiphy->reg_notifier = rtw_reg_notifier;
|
2019-05-24 19:43:57 +00:00
|
|
|
#endif
|
2018-06-22 16:48:32 +00:00
|
|
|
|
|
|
|
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 14, 0))
|
|
|
|
wiphy->flags |= WIPHY_FLAG_CUSTOM_REGULATORY;
|
|
|
|
wiphy->flags &= ~WIPHY_FLAG_STRICT_REGULATORY;
|
|
|
|
wiphy->flags &= ~WIPHY_FLAG_DISABLE_BEACON_HINTS;
|
|
|
|
#else
|
|
|
|
wiphy->regulatory_flags |= REGULATORY_CUSTOM_REG;
|
|
|
|
wiphy->regulatory_flags &= ~REGULATORY_STRICT_REG;
|
|
|
|
wiphy->regulatory_flags &= ~REGULATORY_DISABLE_BEACON_HINTS;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
regd = _rtw_regdomain_select(reg);
|
|
|
|
wiphy_apply_custom_regulatory(wiphy, regd);
|
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
rtw_regd_apply_flags(wiphy);
|
2018-06-22 16:48:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct country_code_to_enum_rd *_rtw_regd_find_country(u16 countrycode)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(allCountries); i++) {
|
|
|
|
if (allCountries[i].countrycode == countrycode)
|
|
|
|
return &allCountries[i];
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-05-24 19:43:57 +00:00
|
|
|
int rtw_regd_init(struct wiphy *wiphy)
|
2018-06-22 16:48:32 +00:00
|
|
|
{
|
|
|
|
_rtw_regd_init_wiphy(NULL, wiphy);
|
2019-05-24 19:43:57 +00:00
|
|
|
|
2018-06-22 16:48:32 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_IOCTL_CFG80211 */
|