1 // SPDX-License-Identifier: GPL-2.0
2 /******************************************************************************
3  *
4  * Copyright(c) 2009-2010  Realtek Corporation.
5  *
6  *****************************************************************************/
7 
8 #include <drv_types.h>
9 #include <rtw_debug.h>
10 
11 #include <rtw_wifi_regd.h>
12 
13 /*
14  * REG_RULE(freq start, freq end, bandwidth, max gain, eirp, reg_flags)
15  */
16 
17 /*
18  * Only these channels all allow active
19  * scan on all world regulatory domains
20  */
21 
22 /* 2G chan 01 - chan 11 */
23 #define RTW_2GHZ_CH01_11	\
24 	REG_RULE(2412 - 10, 2462 + 10, 40, 0, 20, 0)
25 
26 /*
27  * We enable active scan on these a case
28  * by case basis by regulatory domain
29  */
30 
31 /* 2G chan 12 - chan 13, PASSIV SCAN */
32 #define RTW_2GHZ_CH12_13	\
33 	REG_RULE(2467 - 10, 2472 + 10, 40, 0, 20,	\
34 	NL80211_RRF_PASSIVE_SCAN)
35 
36 static const struct ieee80211_regdomain rtw_regdom_rd = {
37 	.n_reg_rules = 3,
38 	.alpha2 = "99",
39 	.reg_rules = {
40 		RTW_2GHZ_CH01_11,
41 		RTW_2GHZ_CH12_13,
42 	}
43 };
44 
rtw_ieee80211_channel_to_frequency(int chan,int band)45 static int rtw_ieee80211_channel_to_frequency(int chan, int band)
46 {
47 	/* see 802.11 17.3.8.3.2 and Annex J
48 	 * there are overlapping channel numbers in 5GHz and 2GHz bands
49 	 */
50 
51 	/* NL80211_BAND_2GHZ */
52 	if (chan == 14)
53 		return 2484;
54 	else if (chan < 14)
55 		return 2407 + chan * 5;
56 	else
57 		return 0;	/* not supported */
58 }
59 
_rtw_reg_apply_flags(struct wiphy * wiphy)60 static void _rtw_reg_apply_flags(struct wiphy *wiphy)
61 {
62 	struct adapter *padapter = wiphy_to_adapter(wiphy);
63 	struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
64 	RT_CHANNEL_INFO *channel_set = pmlmeext->channel_set;
65 	u8 max_chan_nums = pmlmeext->max_chan_nums;
66 
67 	struct ieee80211_supported_band *sband;
68 	struct ieee80211_channel *ch;
69 	unsigned int i, j;
70 	u16 channel;
71 	u32 freq;
72 
73 	/* all channels disable */
74 	for (i = 0; i < NUM_NL80211_BANDS; i++) {
75 		sband = wiphy->bands[i];
76 
77 		if (sband) {
78 			for (j = 0; j < sband->n_channels; j++) {
79 				ch = &sband->channels[j];
80 
81 				if (ch)
82 					ch->flags = IEEE80211_CHAN_DISABLED;
83 			}
84 		}
85 	}
86 
87 	/* channels apply by channel plans. */
88 	for (i = 0; i < max_chan_nums; i++) {
89 		channel = channel_set[i].ChannelNum;
90 		freq =
91 		    rtw_ieee80211_channel_to_frequency(channel,
92 						       NL80211_BAND_2GHZ);
93 
94 		ch = ieee80211_get_channel(wiphy, freq);
95 		if (ch) {
96 			if (channel_set[i].ScanType == SCAN_PASSIVE)
97 				ch->flags = IEEE80211_CHAN_NO_IR;
98 			else
99 				ch->flags = 0;
100 		}
101 	}
102 }
103 
_rtw_reg_notifier_apply(struct wiphy * wiphy,struct regulatory_request * request,struct rtw_regulatory * reg)104 static int _rtw_reg_notifier_apply(struct wiphy *wiphy,
105 				   struct regulatory_request *request,
106 				   struct rtw_regulatory *reg)
107 {
108 	/* Hard code flags */
109 	_rtw_reg_apply_flags(wiphy);
110 	return 0;
111 }
112 
_rtw_regdomain_select(struct rtw_regulatory * reg)113 static const struct ieee80211_regdomain *_rtw_regdomain_select(struct
114 							       rtw_regulatory
115 							       *reg)
116 {
117 	return &rtw_regdom_rd;
118 }
119 
_rtw_regd_init_wiphy(struct rtw_regulatory * reg,struct wiphy * wiphy,void (* reg_notifier)(struct wiphy * wiphy,struct regulatory_request * request))120 static void _rtw_regd_init_wiphy(struct rtw_regulatory *reg,
121 				 struct wiphy *wiphy,
122 				 void (*reg_notifier)(struct wiphy *wiphy,
123 						      struct
124 						      regulatory_request *
125 						      request))
126 {
127 	const struct ieee80211_regdomain *regd;
128 
129 	wiphy->reg_notifier = reg_notifier;
130 
131 	wiphy->regulatory_flags |= REGULATORY_CUSTOM_REG;
132 	wiphy->regulatory_flags &= ~REGULATORY_STRICT_REG;
133 	wiphy->regulatory_flags &= ~REGULATORY_DISABLE_BEACON_HINTS;
134 
135 	regd = _rtw_regdomain_select(reg);
136 	wiphy_apply_custom_regulatory(wiphy, regd);
137 
138 	/* Hard code flags */
139 	_rtw_reg_apply_flags(wiphy);
140 }
141 
rtw_regd_init(struct adapter * padapter,void (* reg_notifier)(struct wiphy * wiphy,struct regulatory_request * request))142 int rtw_regd_init(struct adapter *padapter,
143 		  void (*reg_notifier)(struct wiphy *wiphy,
144 				       struct regulatory_request *request))
145 {
146 	struct wiphy *wiphy = padapter->rtw_wdev->wiphy;
147 
148 	_rtw_regd_init_wiphy(NULL, wiphy, reg_notifier);
149 
150 	return 0;
151 }
152 
rtw_reg_notifier(struct wiphy * wiphy,struct regulatory_request * request)153 void rtw_reg_notifier(struct wiphy *wiphy, struct regulatory_request *request)
154 {
155 	struct rtw_regulatory *reg = NULL;
156 
157 	DBG_8192C("%s\n", __func__);
158 
159 	_rtw_reg_notifier_apply(wiphy, request, reg);
160 }
161