| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424 |
- /******************************************************************************
- *
- * 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 \
- REG_RULE(2467-10, 2472+10, 40, 0, 20, \
- NL80211_RRF_PASSIVE_SCAN)
- /* 2G chan 14, PASSIVS SCAN, NO OFDM (B only) */
- #define RTW_2GHZ_CH14 \
- REG_RULE(2484-10, 2484+10, 40, 0, 20, \
- NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_OFDM)
- /* 5G chan 36 - chan 64 */
- #define RTW_5GHZ_5150_5350 \
- REG_RULE(5150-10, 5350+10, 40, 0, 30, \
- NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
- /* 5G chan 100 - chan 165 */
- #define RTW_5GHZ_5470_5850 \
- REG_RULE(5470-10, 5850+10, 40, 0, 30, \
- NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
- /* 5G chan 149 - chan 165 */
- #define RTW_5GHZ_5725_5850 \
- REG_RULE(5725-10, 5850+10, 40, 0, 30, \
- NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
- /* 5G chan 36 - chan 165 */
- #define RTW_5GHZ_5150_5850 \
- REG_RULE(5150-10, 5850+10, 40, 0, 30, \
- NL80211_RRF_PASSIVE_SCAN | NL80211_RRF_NO_IBSS)
- static const struct ieee80211_regdomain rtw_regdom_rd = {
- .n_reg_rules = 3,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_2GHZ_CH12_13,
- RTW_5GHZ_5150_5850,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_11 = {
- .n_reg_rules = 1,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_12_13 = {
- .n_reg_rules = 2,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_2GHZ_CH12_13,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_no_midband = {
- .n_reg_rules = 3,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_5GHZ_5150_5350,
- RTW_5GHZ_5725_5850,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_60_64 = {
- .n_reg_rules = 3,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_2GHZ_CH12_13,
- RTW_5GHZ_5725_5850,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_14_60_64 = {
- .n_reg_rules = 4,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_2GHZ_CH12_13,
- RTW_2GHZ_CH14,
- RTW_5GHZ_5725_5850,
- }
- };
- static const struct ieee80211_regdomain rtw_regdom_14 = {
- .n_reg_rules = 3,
- .alpha2 = "99",
- .reg_rules = {
- RTW_2GHZ_CH01_11,
- RTW_2GHZ_CH12_13,
- RTW_2GHZ_CH14,
- }
- };
- #if 0
- static struct rtw_regulatory *rtw_regd;
- #endif
- #if 0 /* not_yet */
- static void _rtw_reg_apply_beaconing_flags(struct wiphy *wiphy,
- enum nl80211_reg_initiator initiator)
- {
- enum nl80211_band band;
- struct ieee80211_supported_band *sband;
- const struct ieee80211_reg_rule *reg_rule;
- struct ieee80211_channel *ch;
- unsigned int i;
- u32 bandwidth = 0;
- int r;
- for (band = 0; band < NUM_NL80211_BANDS; band++) {
- if (!wiphy->bands[band])
- continue;
- sband = wiphy->bands[band];
- for (i = 0; i < sband->n_channels; i++) {
- ch = &sband->channels[i];
- if (rtw_is_dfs_ch(ch->hw_value) ||
- (ch->flags & IEEE80211_CHAN_RADAR))
- continue;
- if (initiator == NL80211_REGDOM_SET_BY_COUNTRY_IE) {
- r = freq_reg_info(wiphy, ch->center_freq,
- bandwidth, ®_rule);
- if (r)
- continue;
- /*
- *If 11d had a rule for this channel ensure
- *we enable adhoc/beaconing if it allows us to
- *use it. Note that we would have disabled it
- *by applying our static world regdomain by
- *default during init, prior to calling our
- *regulatory_hint().
- */
- if (!(reg_rule->flags & NL80211_RRF_NO_IBSS))
- ch->flags &= ~IEEE80211_CHAN_NO_IBSS;
- if (!
- (reg_rule->flags &
- NL80211_RRF_PASSIVE_SCAN))
- ch->flags &=
- ~IEEE80211_CHAN_PASSIVE_SCAN;
- } else {
- if (ch->beacon_found)
- ch->flags &= ~(IEEE80211_CHAN_NO_IBSS |
- IEEE80211_CHAN_PASSIVE_SCAN);
- }
- }
- }
- }
- /* Allows active scan scan on Ch 12 and 13 */
- static void _rtw_reg_apply_active_scan_flags(struct wiphy *wiphy,
- enum nl80211_reg_initiator
- initiator)
- {
- struct ieee80211_supported_band *sband;
- struct ieee80211_channel *ch;
- const struct ieee80211_reg_rule *reg_rule;
- u32 bandwidth = 0;
- int r;
- if (!wiphy->bands[NL80211_BAND_2GHZ])
- return;
- sband = wiphy->bands[NL80211_BAND_2GHZ];
- /*
- * If no country IE has been received always enable active scan
- * on these channels. This is only done for specific regulatory SKUs
- */
- if (initiator != NL80211_REGDOM_SET_BY_COUNTRY_IE) {
- ch = &sband->channels[11]; /* CH 12 */
- if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN)
- ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN;
- ch = &sband->channels[12]; /* CH 13 */
- if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN)
- ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN;
- return;
- }
- /*
- * If a country IE has been received check its rule for this
- * channel first before enabling active scan. The passive scan
- * would have been enforced by the initial processing of our
- * custom regulatory domain.
- */
- ch = &sband->channels[11]; /* CH 12 */
- r = freq_reg_info(wiphy, ch->center_freq, bandwidth, ®_rule);
- if (!r) {
- if (!(reg_rule->flags & NL80211_RRF_PASSIVE_SCAN))
- if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN)
- ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN;
- }
- ch = &sband->channels[12]; /* CH 13 */
- r = freq_reg_info(wiphy, ch->center_freq, bandwidth, ®_rule);
- if (!r) {
- if (!(reg_rule->flags & NL80211_RRF_PASSIVE_SCAN))
- if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN)
- ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN;
- }
- }
- #endif
- void rtw_regd_apply_flags(struct wiphy *wiphy)
- {
- struct dvobj_priv *dvobj = wiphy_to_dvobj(wiphy);
- struct rf_ctl_t *rfctl = dvobj_to_rfctl(dvobj);
- RT_CHANNEL_INFO *channel_set = rfctl->channel_set;
- u8 max_chan_nums = rfctl->max_chan_nums;
- struct ieee80211_supported_band *sband;
- struct ieee80211_channel *ch;
- unsigned int i, j;
- u16 channel;
- u32 freq;
- /* 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)
- ch->flags = IEEE80211_CHAN_DISABLED;
- }
- }
- }
- /* 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);
- if (!ch)
- continue;
- 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;
- #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
- }
- #endif /* CONFIG_DFS */
- }
- }
- static const struct ieee80211_regdomain *_rtw_regdomain_select(struct
- rtw_regulatory
- *reg)
- {
- #if 0
- switch (reg->country_code) {
- case COUNTRY_CODE_USER:
- default:
- return &rtw_regdom_rd;
- }
- #else
- return &rtw_regdom_rd;
- #endif
- }
- static void rtw_reg_notifier(struct wiphy *wiphy, struct regulatory_request *request)
- {
- 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;
- }
- rtw_regd_apply_flags(wiphy);
- }
- #if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 9, 0))
- static int rtw_reg_notifier_return(struct wiphy *wiphy, struct regulatory_request *request)
- {
- rtw_reg_notifier(wiphy, request);
- return 0;
- }
- #endif
- static void _rtw_regd_init_wiphy(struct rtw_regulatory *reg, struct wiphy *wiphy)
- {
- const struct ieee80211_regdomain *regd;
- #if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 9, 0))
- wiphy->reg_notifier = rtw_reg_notifier_return;
- #else
- wiphy->reg_notifier = rtw_reg_notifier;
- #endif
- #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);
- rtw_regd_apply_flags(wiphy);
- }
- 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;
- }
- int rtw_regd_init(struct wiphy *wiphy)
- {
- #if 0
- if (rtw_regd == NULL) {
- rtw_regd = (struct rtw_regulatory *)
- rtw_malloc(sizeof(struct rtw_regulatory));
- rtw_regd->alpha2[0] = '9';
- rtw_regd->alpha2[1] = '9';
- rtw_regd->country_code = COUNTRY_CODE_USER;
- }
- RTW_INFO("%s: Country alpha2 being used: %c%c\n",
- __func__, rtw_regd->alpha2[0], rtw_regd->alpha2[1]);
- #endif
- _rtw_regd_init_wiphy(NULL, wiphy);
- return 0;
- }
- #endif /* CONFIG_IOCTL_CFG80211 */
|