diff options
author | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
---|---|---|
committer | Glenn Elliott <gelliott@cs.unc.edu> | 2012-03-04 19:47:13 -0500 |
commit | c71c03bda1e86c9d5198c5d83f712e695c4f2a1e (patch) | |
tree | ecb166cb3e2b7e2adb3b5e292245fefd23381ac8 /drivers/net/wireless/rtlwifi | |
parent | ea53c912f8a86a8567697115b6a0d8152beee5c8 (diff) | |
parent | 6a00f206debf8a5c8899055726ad127dbeeed098 (diff) |
Merge branch 'mpi-master' into wip-k-fmlpwip-k-fmlp
Conflicts:
litmus/sched_cedf.c
Diffstat (limited to 'drivers/net/wireless/rtlwifi')
92 files changed, 47700 insertions, 0 deletions
diff --git a/drivers/net/wireless/rtlwifi/Kconfig b/drivers/net/wireless/rtlwifi/Kconfig new file mode 100644 index 000000000000..5aee8b22d74e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/Kconfig | |||
@@ -0,0 +1,44 @@ | |||
1 | config RTL8192CE | ||
2 | tristate "Realtek RTL8192CE/RTL8188CE Wireless Network Adapter" | ||
3 | depends on MAC80211 && PCI && EXPERIMENTAL | ||
4 | select FW_LOADER | ||
5 | select RTLWIFI | ||
6 | select RTL8192C_COMMON | ||
7 | ---help--- | ||
8 | This is the driver for Realtek RTL8192CE/RTL8188CE 802.11n PCIe | ||
9 | wireless network adapters. | ||
10 | |||
11 | If you choose to build it as a module, it will be called rtl8192ce | ||
12 | |||
13 | config RTL8192SE | ||
14 | tristate "Realtek RTL8192SE/RTL8191SE PCIe Wireless Network Adapter" | ||
15 | depends on MAC80211 && EXPERIMENTAL | ||
16 | select FW_LOADER | ||
17 | select RTLWIFI | ||
18 | ---help--- | ||
19 | This is the driver for Realtek RTL8192SE/RTL8191SE 802.11n PCIe | ||
20 | wireless network adapters. | ||
21 | |||
22 | If you choose to build it as a module, it will be called rtl8192se | ||
23 | |||
24 | config RTL8192CU | ||
25 | tristate "Realtek RTL8192CU/RTL8188CU USB Wireless Network Adapter" | ||
26 | depends on MAC80211 && USB && EXPERIMENTAL | ||
27 | select FW_LOADER | ||
28 | select RTLWIFI | ||
29 | select RTL8192C_COMMON | ||
30 | ---help--- | ||
31 | This is the driver for Realtek RTL8192CU/RTL8188CU 802.11n USB | ||
32 | wireless network adapters. | ||
33 | |||
34 | If you choose to build it as a module, it will be called rtl8192cu | ||
35 | |||
36 | config RTLWIFI | ||
37 | tristate | ||
38 | depends on RTL8192CE || RTL8192CU || RTL8192SE | ||
39 | default m | ||
40 | |||
41 | config RTL8192C_COMMON | ||
42 | tristate | ||
43 | depends on RTL8192CE || RTL8192CU || RTL8192SE | ||
44 | default m | ||
diff --git a/drivers/net/wireless/rtlwifi/Makefile b/drivers/net/wireless/rtlwifi/Makefile new file mode 100644 index 000000000000..7acce83c3785 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/Makefile | |||
@@ -0,0 +1,27 @@ | |||
1 | obj-$(CONFIG_RTLWIFI) += rtlwifi.o | ||
2 | rtlwifi-objs := \ | ||
3 | base.o \ | ||
4 | cam.o \ | ||
5 | core.o \ | ||
6 | debug.o \ | ||
7 | efuse.o \ | ||
8 | ps.o \ | ||
9 | rc.o \ | ||
10 | regd.o | ||
11 | |||
12 | rtl8192c_common-objs += \ | ||
13 | |||
14 | ifneq ($(CONFIG_PCI),) | ||
15 | rtlwifi-objs += pci.o | ||
16 | endif | ||
17 | |||
18 | ifneq ($(CONFIG_USB),) | ||
19 | rtlwifi-objs += usb.o | ||
20 | endif | ||
21 | |||
22 | obj-$(CONFIG_RTL8192C_COMMON) += rtl8192c/ | ||
23 | obj-$(CONFIG_RTL8192CE) += rtl8192ce/ | ||
24 | obj-$(CONFIG_RTL8192CU) += rtl8192cu/ | ||
25 | obj-$(CONFIG_RTL8192SE) += rtl8192se/ | ||
26 | |||
27 | ccflags-y += -D__CHECK_ENDIAN__ | ||
diff --git a/drivers/net/wireless/rtlwifi/base.c b/drivers/net/wireless/rtlwifi/base.c new file mode 100644 index 000000000000..ccb6da38fe22 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/base.c | |||
@@ -0,0 +1,1422 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include <linux/ip.h> | ||
31 | #include "wifi.h" | ||
32 | #include "rc.h" | ||
33 | #include "base.h" | ||
34 | #include "efuse.h" | ||
35 | #include "cam.h" | ||
36 | #include "ps.h" | ||
37 | #include "regd.h" | ||
38 | |||
39 | /* | ||
40 | *NOTICE!!!: This file will be very big, we hsould | ||
41 | *keep it clear under follwing roles: | ||
42 | * | ||
43 | *This file include follwing part, so, if you add new | ||
44 | *functions into this file, please check which part it | ||
45 | *should includes. or check if you should add new part | ||
46 | *for this file: | ||
47 | * | ||
48 | *1) mac80211 init functions | ||
49 | *2) tx information functions | ||
50 | *3) functions called by core.c | ||
51 | *4) wq & timer callback functions | ||
52 | *5) frame process functions | ||
53 | *6) IOT functions | ||
54 | *7) sysfs functions | ||
55 | *8) ... | ||
56 | */ | ||
57 | |||
58 | /********************************************************* | ||
59 | * | ||
60 | * mac80211 init functions | ||
61 | * | ||
62 | *********************************************************/ | ||
63 | static struct ieee80211_channel rtl_channeltable_2g[] = { | ||
64 | {.center_freq = 2412, .hw_value = 1,}, | ||
65 | {.center_freq = 2417, .hw_value = 2,}, | ||
66 | {.center_freq = 2422, .hw_value = 3,}, | ||
67 | {.center_freq = 2427, .hw_value = 4,}, | ||
68 | {.center_freq = 2432, .hw_value = 5,}, | ||
69 | {.center_freq = 2437, .hw_value = 6,}, | ||
70 | {.center_freq = 2442, .hw_value = 7,}, | ||
71 | {.center_freq = 2447, .hw_value = 8,}, | ||
72 | {.center_freq = 2452, .hw_value = 9,}, | ||
73 | {.center_freq = 2457, .hw_value = 10,}, | ||
74 | {.center_freq = 2462, .hw_value = 11,}, | ||
75 | {.center_freq = 2467, .hw_value = 12,}, | ||
76 | {.center_freq = 2472, .hw_value = 13,}, | ||
77 | {.center_freq = 2484, .hw_value = 14,}, | ||
78 | }; | ||
79 | |||
80 | static struct ieee80211_channel rtl_channeltable_5g[] = { | ||
81 | {.center_freq = 5180, .hw_value = 36,}, | ||
82 | {.center_freq = 5200, .hw_value = 40,}, | ||
83 | {.center_freq = 5220, .hw_value = 44,}, | ||
84 | {.center_freq = 5240, .hw_value = 48,}, | ||
85 | {.center_freq = 5260, .hw_value = 52,}, | ||
86 | {.center_freq = 5280, .hw_value = 56,}, | ||
87 | {.center_freq = 5300, .hw_value = 60,}, | ||
88 | {.center_freq = 5320, .hw_value = 64,}, | ||
89 | {.center_freq = 5500, .hw_value = 100,}, | ||
90 | {.center_freq = 5520, .hw_value = 104,}, | ||
91 | {.center_freq = 5540, .hw_value = 108,}, | ||
92 | {.center_freq = 5560, .hw_value = 112,}, | ||
93 | {.center_freq = 5580, .hw_value = 116,}, | ||
94 | {.center_freq = 5600, .hw_value = 120,}, | ||
95 | {.center_freq = 5620, .hw_value = 124,}, | ||
96 | {.center_freq = 5640, .hw_value = 128,}, | ||
97 | {.center_freq = 5660, .hw_value = 132,}, | ||
98 | {.center_freq = 5680, .hw_value = 136,}, | ||
99 | {.center_freq = 5700, .hw_value = 140,}, | ||
100 | {.center_freq = 5745, .hw_value = 149,}, | ||
101 | {.center_freq = 5765, .hw_value = 153,}, | ||
102 | {.center_freq = 5785, .hw_value = 157,}, | ||
103 | {.center_freq = 5805, .hw_value = 161,}, | ||
104 | {.center_freq = 5825, .hw_value = 165,}, | ||
105 | }; | ||
106 | |||
107 | static struct ieee80211_rate rtl_ratetable_2g[] = { | ||
108 | {.bitrate = 10, .hw_value = 0x00,}, | ||
109 | {.bitrate = 20, .hw_value = 0x01,}, | ||
110 | {.bitrate = 55, .hw_value = 0x02,}, | ||
111 | {.bitrate = 110, .hw_value = 0x03,}, | ||
112 | {.bitrate = 60, .hw_value = 0x04,}, | ||
113 | {.bitrate = 90, .hw_value = 0x05,}, | ||
114 | {.bitrate = 120, .hw_value = 0x06,}, | ||
115 | {.bitrate = 180, .hw_value = 0x07,}, | ||
116 | {.bitrate = 240, .hw_value = 0x08,}, | ||
117 | {.bitrate = 360, .hw_value = 0x09,}, | ||
118 | {.bitrate = 480, .hw_value = 0x0a,}, | ||
119 | {.bitrate = 540, .hw_value = 0x0b,}, | ||
120 | }; | ||
121 | |||
122 | static struct ieee80211_rate rtl_ratetable_5g[] = { | ||
123 | {.bitrate = 60, .hw_value = 0x04,}, | ||
124 | {.bitrate = 90, .hw_value = 0x05,}, | ||
125 | {.bitrate = 120, .hw_value = 0x06,}, | ||
126 | {.bitrate = 180, .hw_value = 0x07,}, | ||
127 | {.bitrate = 240, .hw_value = 0x08,}, | ||
128 | {.bitrate = 360, .hw_value = 0x09,}, | ||
129 | {.bitrate = 480, .hw_value = 0x0a,}, | ||
130 | {.bitrate = 540, .hw_value = 0x0b,}, | ||
131 | }; | ||
132 | |||
133 | static const struct ieee80211_supported_band rtl_band_2ghz = { | ||
134 | .band = IEEE80211_BAND_2GHZ, | ||
135 | |||
136 | .channels = rtl_channeltable_2g, | ||
137 | .n_channels = ARRAY_SIZE(rtl_channeltable_2g), | ||
138 | |||
139 | .bitrates = rtl_ratetable_2g, | ||
140 | .n_bitrates = ARRAY_SIZE(rtl_ratetable_2g), | ||
141 | |||
142 | .ht_cap = {0}, | ||
143 | }; | ||
144 | |||
145 | static struct ieee80211_supported_band rtl_band_5ghz = { | ||
146 | .band = IEEE80211_BAND_5GHZ, | ||
147 | |||
148 | .channels = rtl_channeltable_5g, | ||
149 | .n_channels = ARRAY_SIZE(rtl_channeltable_5g), | ||
150 | |||
151 | .bitrates = rtl_ratetable_5g, | ||
152 | .n_bitrates = ARRAY_SIZE(rtl_ratetable_5g), | ||
153 | |||
154 | .ht_cap = {0}, | ||
155 | }; | ||
156 | |||
157 | static const u8 tid_to_ac[] = { | ||
158 | 2, /* IEEE80211_AC_BE */ | ||
159 | 3, /* IEEE80211_AC_BK */ | ||
160 | 3, /* IEEE80211_AC_BK */ | ||
161 | 2, /* IEEE80211_AC_BE */ | ||
162 | 1, /* IEEE80211_AC_VI */ | ||
163 | 1, /* IEEE80211_AC_VI */ | ||
164 | 0, /* IEEE80211_AC_VO */ | ||
165 | 0, /* IEEE80211_AC_VO */ | ||
166 | }; | ||
167 | |||
168 | u8 rtl_tid_to_ac(struct ieee80211_hw *hw, u8 tid) | ||
169 | { | ||
170 | return tid_to_ac[tid]; | ||
171 | } | ||
172 | |||
173 | static void _rtl_init_hw_ht_capab(struct ieee80211_hw *hw, | ||
174 | struct ieee80211_sta_ht_cap *ht_cap) | ||
175 | { | ||
176 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
177 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
178 | |||
179 | ht_cap->ht_supported = true; | ||
180 | ht_cap->cap = IEEE80211_HT_CAP_SUP_WIDTH_20_40 | | ||
181 | IEEE80211_HT_CAP_SGI_40 | | ||
182 | IEEE80211_HT_CAP_SGI_20 | | ||
183 | IEEE80211_HT_CAP_DSSSCCK40 | IEEE80211_HT_CAP_MAX_AMSDU; | ||
184 | |||
185 | if (rtlpriv->rtlhal.disable_amsdu_8k) | ||
186 | ht_cap->cap &= ~IEEE80211_HT_CAP_MAX_AMSDU; | ||
187 | |||
188 | /* | ||
189 | *Maximum length of AMPDU that the STA can receive. | ||
190 | *Length = 2 ^ (13 + max_ampdu_length_exp) - 1 (octets) | ||
191 | */ | ||
192 | ht_cap->ampdu_factor = IEEE80211_HT_MAX_AMPDU_64K; | ||
193 | |||
194 | /*Minimum MPDU start spacing , */ | ||
195 | ht_cap->ampdu_density = IEEE80211_HT_MPDU_DENSITY_16; | ||
196 | |||
197 | ht_cap->mcs.tx_params = IEEE80211_HT_MCS_TX_DEFINED; | ||
198 | |||
199 | /* | ||
200 | *hw->wiphy->bands[IEEE80211_BAND_2GHZ] | ||
201 | *base on ant_num | ||
202 | *rx_mask: RX mask | ||
203 | *if rx_ant =1 rx_mask[0]=0xff;==>MCS0-MCS7 | ||
204 | *if rx_ant =2 rx_mask[1]=0xff;==>MCS8-MCS15 | ||
205 | *if rx_ant >=3 rx_mask[2]=0xff; | ||
206 | *if BW_40 rx_mask[4]=0x01; | ||
207 | *highest supported RX rate | ||
208 | */ | ||
209 | if (get_rf_type(rtlphy) == RF_1T2R || get_rf_type(rtlphy) == RF_2T2R) { | ||
210 | |||
211 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("1T2R or 2T2R\n")); | ||
212 | |||
213 | ht_cap->mcs.rx_mask[0] = 0xFF; | ||
214 | ht_cap->mcs.rx_mask[1] = 0xFF; | ||
215 | ht_cap->mcs.rx_mask[4] = 0x01; | ||
216 | |||
217 | ht_cap->mcs.rx_highest = cpu_to_le16(MAX_BIT_RATE_40MHZ_MCS15); | ||
218 | } else if (get_rf_type(rtlphy) == RF_1T1R) { | ||
219 | |||
220 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("1T1R\n")); | ||
221 | |||
222 | ht_cap->mcs.rx_mask[0] = 0xFF; | ||
223 | ht_cap->mcs.rx_mask[1] = 0x00; | ||
224 | ht_cap->mcs.rx_mask[4] = 0x01; | ||
225 | |||
226 | ht_cap->mcs.rx_highest = cpu_to_le16(MAX_BIT_RATE_40MHZ_MCS7); | ||
227 | } | ||
228 | } | ||
229 | |||
230 | static void _rtl_init_mac80211(struct ieee80211_hw *hw) | ||
231 | { | ||
232 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
233 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
234 | struct rtl_mac *rtlmac = rtl_mac(rtl_priv(hw)); | ||
235 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
236 | struct ieee80211_supported_band *sband; | ||
237 | |||
238 | |||
239 | if (rtlhal->macphymode == SINGLEMAC_SINGLEPHY && rtlhal->bandset == | ||
240 | BAND_ON_BOTH) { | ||
241 | /* 1: 2.4 G bands */ | ||
242 | /* <1> use mac->bands as mem for hw->wiphy->bands */ | ||
243 | sband = &(rtlmac->bands[IEEE80211_BAND_2GHZ]); | ||
244 | |||
245 | /* <2> set hw->wiphy->bands[IEEE80211_BAND_2GHZ] | ||
246 | * to default value(1T1R) */ | ||
247 | memcpy(&(rtlmac->bands[IEEE80211_BAND_2GHZ]), &rtl_band_2ghz, | ||
248 | sizeof(struct ieee80211_supported_band)); | ||
249 | |||
250 | /* <3> init ht cap base on ant_num */ | ||
251 | _rtl_init_hw_ht_capab(hw, &sband->ht_cap); | ||
252 | |||
253 | /* <4> set mac->sband to wiphy->sband */ | ||
254 | hw->wiphy->bands[IEEE80211_BAND_2GHZ] = sband; | ||
255 | |||
256 | /* 2: 5 G bands */ | ||
257 | /* <1> use mac->bands as mem for hw->wiphy->bands */ | ||
258 | sband = &(rtlmac->bands[IEEE80211_BAND_5GHZ]); | ||
259 | |||
260 | /* <2> set hw->wiphy->bands[IEEE80211_BAND_5GHZ] | ||
261 | * to default value(1T1R) */ | ||
262 | memcpy(&(rtlmac->bands[IEEE80211_BAND_5GHZ]), &rtl_band_5ghz, | ||
263 | sizeof(struct ieee80211_supported_band)); | ||
264 | |||
265 | /* <3> init ht cap base on ant_num */ | ||
266 | _rtl_init_hw_ht_capab(hw, &sband->ht_cap); | ||
267 | |||
268 | /* <4> set mac->sband to wiphy->sband */ | ||
269 | hw->wiphy->bands[IEEE80211_BAND_5GHZ] = sband; | ||
270 | } else { | ||
271 | if (rtlhal->current_bandtype == BAND_ON_2_4G) { | ||
272 | /* <1> use mac->bands as mem for hw->wiphy->bands */ | ||
273 | sband = &(rtlmac->bands[IEEE80211_BAND_2GHZ]); | ||
274 | |||
275 | /* <2> set hw->wiphy->bands[IEEE80211_BAND_2GHZ] | ||
276 | * to default value(1T1R) */ | ||
277 | memcpy(&(rtlmac->bands[IEEE80211_BAND_2GHZ]), | ||
278 | &rtl_band_2ghz, | ||
279 | sizeof(struct ieee80211_supported_band)); | ||
280 | |||
281 | /* <3> init ht cap base on ant_num */ | ||
282 | _rtl_init_hw_ht_capab(hw, &sband->ht_cap); | ||
283 | |||
284 | /* <4> set mac->sband to wiphy->sband */ | ||
285 | hw->wiphy->bands[IEEE80211_BAND_2GHZ] = sband; | ||
286 | } else if (rtlhal->current_bandtype == BAND_ON_5G) { | ||
287 | /* <1> use mac->bands as mem for hw->wiphy->bands */ | ||
288 | sband = &(rtlmac->bands[IEEE80211_BAND_5GHZ]); | ||
289 | |||
290 | /* <2> set hw->wiphy->bands[IEEE80211_BAND_5GHZ] | ||
291 | * to default value(1T1R) */ | ||
292 | memcpy(&(rtlmac->bands[IEEE80211_BAND_5GHZ]), | ||
293 | &rtl_band_5ghz, | ||
294 | sizeof(struct ieee80211_supported_band)); | ||
295 | |||
296 | /* <3> init ht cap base on ant_num */ | ||
297 | _rtl_init_hw_ht_capab(hw, &sband->ht_cap); | ||
298 | |||
299 | /* <4> set mac->sband to wiphy->sband */ | ||
300 | hw->wiphy->bands[IEEE80211_BAND_5GHZ] = sband; | ||
301 | } else { | ||
302 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
303 | ("Err BAND %d\n", | ||
304 | rtlhal->current_bandtype)); | ||
305 | } | ||
306 | } | ||
307 | /* <5> set hw caps */ | ||
308 | hw->flags = IEEE80211_HW_SIGNAL_DBM | | ||
309 | IEEE80211_HW_RX_INCLUDES_FCS | | ||
310 | IEEE80211_HW_BEACON_FILTER | | ||
311 | IEEE80211_HW_AMPDU_AGGREGATION | | ||
312 | IEEE80211_HW_REPORTS_TX_ACK_STATUS | 0; | ||
313 | |||
314 | /* swlps or hwlps has been set in diff chip in init_sw_vars */ | ||
315 | if (rtlpriv->psc.swctrl_lps) | ||
316 | hw->flags |= IEEE80211_HW_SUPPORTS_PS | | ||
317 | IEEE80211_HW_PS_NULLFUNC_STACK | | ||
318 | /* IEEE80211_HW_SUPPORTS_DYNAMIC_PS | */ | ||
319 | 0; | ||
320 | |||
321 | hw->wiphy->interface_modes = | ||
322 | BIT(NL80211_IFTYPE_AP) | | ||
323 | BIT(NL80211_IFTYPE_STATION) | | ||
324 | BIT(NL80211_IFTYPE_ADHOC); | ||
325 | |||
326 | hw->wiphy->rts_threshold = 2347; | ||
327 | |||
328 | hw->queues = AC_MAX; | ||
329 | hw->extra_tx_headroom = RTL_TX_HEADER_SIZE; | ||
330 | |||
331 | /* TODO: Correct this value for our hw */ | ||
332 | /* TODO: define these hard code value */ | ||
333 | hw->channel_change_time = 100; | ||
334 | hw->max_listen_interval = 10; | ||
335 | hw->max_rate_tries = 4; | ||
336 | /* hw->max_rates = 1; */ | ||
337 | hw->sta_data_size = sizeof(struct rtl_sta_info); | ||
338 | |||
339 | /* <6> mac address */ | ||
340 | if (is_valid_ether_addr(rtlefuse->dev_addr)) { | ||
341 | SET_IEEE80211_PERM_ADDR(hw, rtlefuse->dev_addr); | ||
342 | } else { | ||
343 | u8 rtlmac[] = { 0x00, 0xe0, 0x4c, 0x81, 0x92, 0x00 }; | ||
344 | get_random_bytes((rtlmac + (ETH_ALEN - 1)), 1); | ||
345 | SET_IEEE80211_PERM_ADDR(hw, rtlmac); | ||
346 | } | ||
347 | |||
348 | } | ||
349 | |||
350 | static void _rtl_init_deferred_work(struct ieee80211_hw *hw) | ||
351 | { | ||
352 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
353 | |||
354 | /* <1> timer */ | ||
355 | init_timer(&rtlpriv->works.watchdog_timer); | ||
356 | setup_timer(&rtlpriv->works.watchdog_timer, | ||
357 | rtl_watch_dog_timer_callback, (unsigned long)hw); | ||
358 | |||
359 | /* <2> work queue */ | ||
360 | rtlpriv->works.hw = hw; | ||
361 | rtlpriv->works.rtl_wq = alloc_workqueue(rtlpriv->cfg->name, 0, 0); | ||
362 | INIT_DELAYED_WORK(&rtlpriv->works.watchdog_wq, | ||
363 | (void *)rtl_watchdog_wq_callback); | ||
364 | INIT_DELAYED_WORK(&rtlpriv->works.ips_nic_off_wq, | ||
365 | (void *)rtl_ips_nic_off_wq_callback); | ||
366 | INIT_DELAYED_WORK(&rtlpriv->works.ps_work, | ||
367 | (void *)rtl_swlps_wq_callback); | ||
368 | INIT_DELAYED_WORK(&rtlpriv->works.ps_rfon_wq, | ||
369 | (void *)rtl_swlps_rfon_wq_callback); | ||
370 | |||
371 | } | ||
372 | |||
373 | void rtl_deinit_deferred_work(struct ieee80211_hw *hw) | ||
374 | { | ||
375 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
376 | |||
377 | del_timer_sync(&rtlpriv->works.watchdog_timer); | ||
378 | |||
379 | cancel_delayed_work(&rtlpriv->works.watchdog_wq); | ||
380 | cancel_delayed_work(&rtlpriv->works.ips_nic_off_wq); | ||
381 | cancel_delayed_work(&rtlpriv->works.ps_work); | ||
382 | cancel_delayed_work(&rtlpriv->works.ps_rfon_wq); | ||
383 | } | ||
384 | |||
385 | void rtl_init_rfkill(struct ieee80211_hw *hw) | ||
386 | { | ||
387 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
388 | |||
389 | bool radio_state; | ||
390 | bool blocked; | ||
391 | u8 valid = 0; | ||
392 | |||
393 | /*set init state to on */ | ||
394 | rtlpriv->rfkill.rfkill_state = 1; | ||
395 | wiphy_rfkill_set_hw_state(hw->wiphy, 0); | ||
396 | |||
397 | radio_state = rtlpriv->cfg->ops->radio_onoff_checking(hw, &valid); | ||
398 | |||
399 | if (valid) { | ||
400 | printk(KERN_INFO "rtlwifi: wireless switch is %s\n", | ||
401 | rtlpriv->rfkill.rfkill_state ? "on" : "off"); | ||
402 | |||
403 | rtlpriv->rfkill.rfkill_state = radio_state; | ||
404 | |||
405 | blocked = (rtlpriv->rfkill.rfkill_state == 1) ? 0 : 1; | ||
406 | wiphy_rfkill_set_hw_state(hw->wiphy, blocked); | ||
407 | } | ||
408 | |||
409 | wiphy_rfkill_start_polling(hw->wiphy); | ||
410 | } | ||
411 | |||
412 | void rtl_deinit_rfkill(struct ieee80211_hw *hw) | ||
413 | { | ||
414 | wiphy_rfkill_stop_polling(hw->wiphy); | ||
415 | } | ||
416 | |||
417 | int rtl_init_core(struct ieee80211_hw *hw) | ||
418 | { | ||
419 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
420 | struct rtl_mac *rtlmac = rtl_mac(rtl_priv(hw)); | ||
421 | |||
422 | /* <1> init mac80211 */ | ||
423 | _rtl_init_mac80211(hw); | ||
424 | rtlmac->hw = hw; | ||
425 | |||
426 | /* <2> rate control register */ | ||
427 | hw->rate_control_algorithm = "rtl_rc"; | ||
428 | |||
429 | /* | ||
430 | * <3> init CRDA must come after init | ||
431 | * mac80211 hw in _rtl_init_mac80211. | ||
432 | */ | ||
433 | if (rtl_regd_init(hw, rtl_reg_notifier)) { | ||
434 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("REGD init failed\n")); | ||
435 | return 1; | ||
436 | } else { | ||
437 | /* CRDA regd hint must after init CRDA */ | ||
438 | if (regulatory_hint(hw->wiphy, rtlpriv->regd.alpha2)) { | ||
439 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
440 | ("regulatory_hint fail\n")); | ||
441 | } | ||
442 | } | ||
443 | |||
444 | /* <4> locks */ | ||
445 | mutex_init(&rtlpriv->locks.conf_mutex); | ||
446 | spin_lock_init(&rtlpriv->locks.ips_lock); | ||
447 | spin_lock_init(&rtlpriv->locks.irq_th_lock); | ||
448 | spin_lock_init(&rtlpriv->locks.h2c_lock); | ||
449 | spin_lock_init(&rtlpriv->locks.rf_ps_lock); | ||
450 | spin_lock_init(&rtlpriv->locks.rf_lock); | ||
451 | spin_lock_init(&rtlpriv->locks.lps_lock); | ||
452 | spin_lock_init(&rtlpriv->locks.waitq_lock); | ||
453 | spin_lock_init(&rtlpriv->locks.cck_and_rw_pagea_lock); | ||
454 | |||
455 | rtlmac->link_state = MAC80211_NOLINK; | ||
456 | |||
457 | /* <5> init deferred work */ | ||
458 | _rtl_init_deferred_work(hw); | ||
459 | |||
460 | return 0; | ||
461 | } | ||
462 | |||
463 | void rtl_deinit_core(struct ieee80211_hw *hw) | ||
464 | { | ||
465 | } | ||
466 | |||
467 | void rtl_init_rx_config(struct ieee80211_hw *hw) | ||
468 | { | ||
469 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
470 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
471 | |||
472 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_RCR, (u8 *) (&mac->rx_conf)); | ||
473 | } | ||
474 | |||
475 | /********************************************************* | ||
476 | * | ||
477 | * tx information functions | ||
478 | * | ||
479 | *********************************************************/ | ||
480 | static void _rtl_qurey_shortpreamble_mode(struct ieee80211_hw *hw, | ||
481 | struct rtl_tcb_desc *tcb_desc, | ||
482 | struct ieee80211_tx_info *info) | ||
483 | { | ||
484 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
485 | u8 rate_flag = info->control.rates[0].flags; | ||
486 | |||
487 | tcb_desc->use_shortpreamble = false; | ||
488 | |||
489 | /* 1M can only use Long Preamble. 11B spec */ | ||
490 | if (tcb_desc->hw_rate == rtlpriv->cfg->maps[RTL_RC_CCK_RATE1M]) | ||
491 | return; | ||
492 | else if (rate_flag & IEEE80211_TX_RC_USE_SHORT_PREAMBLE) | ||
493 | tcb_desc->use_shortpreamble = true; | ||
494 | |||
495 | return; | ||
496 | } | ||
497 | |||
498 | static void _rtl_query_shortgi(struct ieee80211_hw *hw, | ||
499 | struct ieee80211_sta *sta, | ||
500 | struct rtl_tcb_desc *tcb_desc, | ||
501 | struct ieee80211_tx_info *info) | ||
502 | { | ||
503 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
504 | u8 rate_flag = info->control.rates[0].flags; | ||
505 | u8 sgi_40 = 0, sgi_20 = 0, bw_40 = 0; | ||
506 | tcb_desc->use_shortgi = false; | ||
507 | |||
508 | if (sta == NULL) | ||
509 | return; | ||
510 | |||
511 | sgi_40 = sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40; | ||
512 | sgi_20 = sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20; | ||
513 | |||
514 | if (!(sta->ht_cap.ht_supported)) | ||
515 | return; | ||
516 | |||
517 | if (!sgi_40 && !sgi_20) | ||
518 | return; | ||
519 | |||
520 | if (mac->opmode == NL80211_IFTYPE_STATION) | ||
521 | bw_40 = mac->bw_40; | ||
522 | else if (mac->opmode == NL80211_IFTYPE_AP || | ||
523 | mac->opmode == NL80211_IFTYPE_ADHOC) | ||
524 | bw_40 = sta->ht_cap.cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40; | ||
525 | |||
526 | if ((bw_40 == true) && sgi_40) | ||
527 | tcb_desc->use_shortgi = true; | ||
528 | else if ((bw_40 == false) && sgi_20) | ||
529 | tcb_desc->use_shortgi = true; | ||
530 | |||
531 | if (!(rate_flag & IEEE80211_TX_RC_SHORT_GI)) | ||
532 | tcb_desc->use_shortgi = false; | ||
533 | } | ||
534 | |||
535 | static void _rtl_query_protection_mode(struct ieee80211_hw *hw, | ||
536 | struct rtl_tcb_desc *tcb_desc, | ||
537 | struct ieee80211_tx_info *info) | ||
538 | { | ||
539 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
540 | u8 rate_flag = info->control.rates[0].flags; | ||
541 | |||
542 | /* Common Settings */ | ||
543 | tcb_desc->rts_stbc = false; | ||
544 | tcb_desc->cts_enable = false; | ||
545 | tcb_desc->rts_sc = 0; | ||
546 | tcb_desc->rts_bw = false; | ||
547 | tcb_desc->rts_use_shortpreamble = false; | ||
548 | tcb_desc->rts_use_shortgi = false; | ||
549 | |||
550 | if (rate_flag & IEEE80211_TX_RC_USE_CTS_PROTECT) { | ||
551 | /* Use CTS-to-SELF in protection mode. */ | ||
552 | tcb_desc->rts_enable = true; | ||
553 | tcb_desc->cts_enable = true; | ||
554 | tcb_desc->rts_rate = rtlpriv->cfg->maps[RTL_RC_OFDM_RATE24M]; | ||
555 | } else if (rate_flag & IEEE80211_TX_RC_USE_RTS_CTS) { | ||
556 | /* Use RTS-CTS in protection mode. */ | ||
557 | tcb_desc->rts_enable = true; | ||
558 | tcb_desc->rts_rate = rtlpriv->cfg->maps[RTL_RC_OFDM_RATE24M]; | ||
559 | } | ||
560 | } | ||
561 | |||
562 | static void _rtl_txrate_selectmode(struct ieee80211_hw *hw, | ||
563 | struct ieee80211_sta *sta, | ||
564 | struct rtl_tcb_desc *tcb_desc) | ||
565 | { | ||
566 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
567 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
568 | struct rtl_sta_info *sta_entry = NULL; | ||
569 | u8 ratr_index = 7; | ||
570 | |||
571 | if (sta) { | ||
572 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
573 | ratr_index = sta_entry->ratr_index; | ||
574 | } | ||
575 | if (!tcb_desc->disable_ratefallback || !tcb_desc->use_driver_rate) { | ||
576 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
577 | tcb_desc->ratr_index = 0; | ||
578 | } else if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
579 | if (tcb_desc->multicast || tcb_desc->broadcast) { | ||
580 | tcb_desc->hw_rate = | ||
581 | rtlpriv->cfg->maps[RTL_RC_CCK_RATE2M]; | ||
582 | tcb_desc->use_driver_rate = 1; | ||
583 | } else { | ||
584 | /* TODO */ | ||
585 | } | ||
586 | tcb_desc->ratr_index = ratr_index; | ||
587 | } else if (mac->opmode == NL80211_IFTYPE_AP) { | ||
588 | tcb_desc->ratr_index = ratr_index; | ||
589 | } | ||
590 | } | ||
591 | |||
592 | if (rtlpriv->dm.useramask) { | ||
593 | /* TODO we will differentiate adhoc and station futrue */ | ||
594 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
595 | tcb_desc->mac_id = 0; | ||
596 | |||
597 | if (mac->mode == WIRELESS_MODE_N_24G) | ||
598 | tcb_desc->ratr_index = RATR_INX_WIRELESS_NGB; | ||
599 | else if (mac->mode == WIRELESS_MODE_N_5G) | ||
600 | tcb_desc->ratr_index = RATR_INX_WIRELESS_NG; | ||
601 | else if (mac->mode & WIRELESS_MODE_G) | ||
602 | tcb_desc->ratr_index = RATR_INX_WIRELESS_GB; | ||
603 | else if (mac->mode & WIRELESS_MODE_B) | ||
604 | tcb_desc->ratr_index = RATR_INX_WIRELESS_B; | ||
605 | else if (mac->mode & WIRELESS_MODE_A) | ||
606 | tcb_desc->ratr_index = RATR_INX_WIRELESS_G; | ||
607 | } else if (mac->opmode == NL80211_IFTYPE_AP || | ||
608 | mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
609 | if (NULL != sta) { | ||
610 | if (sta->aid > 0) | ||
611 | tcb_desc->mac_id = sta->aid + 1; | ||
612 | else | ||
613 | tcb_desc->mac_id = 1; | ||
614 | } else { | ||
615 | tcb_desc->mac_id = 0; | ||
616 | } | ||
617 | } | ||
618 | } | ||
619 | |||
620 | } | ||
621 | |||
622 | static void _rtl_query_bandwidth_mode(struct ieee80211_hw *hw, | ||
623 | struct ieee80211_sta *sta, | ||
624 | struct rtl_tcb_desc *tcb_desc) | ||
625 | { | ||
626 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
627 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
628 | |||
629 | tcb_desc->packet_bw = false; | ||
630 | if (!sta) | ||
631 | return; | ||
632 | if (mac->opmode == NL80211_IFTYPE_AP || | ||
633 | mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
634 | if (!(sta->ht_cap.ht_supported) || | ||
635 | !(sta->ht_cap.cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40)) | ||
636 | return; | ||
637 | } else if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
638 | if (!mac->bw_40 || !(sta->ht_cap.ht_supported)) | ||
639 | return; | ||
640 | } | ||
641 | if (tcb_desc->multicast || tcb_desc->broadcast) | ||
642 | return; | ||
643 | |||
644 | /*use legency rate, shall use 20MHz */ | ||
645 | if (tcb_desc->hw_rate <= rtlpriv->cfg->maps[RTL_RC_OFDM_RATE54M]) | ||
646 | return; | ||
647 | |||
648 | tcb_desc->packet_bw = true; | ||
649 | } | ||
650 | |||
651 | static u8 _rtl_get_highest_n_rate(struct ieee80211_hw *hw) | ||
652 | { | ||
653 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
654 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
655 | u8 hw_rate; | ||
656 | |||
657 | if (get_rf_type(rtlphy) == RF_2T2R) | ||
658 | hw_rate = rtlpriv->cfg->maps[RTL_RC_HT_RATEMCS15]; | ||
659 | else | ||
660 | hw_rate = rtlpriv->cfg->maps[RTL_RC_HT_RATEMCS7]; | ||
661 | |||
662 | return hw_rate; | ||
663 | } | ||
664 | |||
665 | void rtl_get_tcb_desc(struct ieee80211_hw *hw, | ||
666 | struct ieee80211_tx_info *info, | ||
667 | struct ieee80211_sta *sta, | ||
668 | struct sk_buff *skb, struct rtl_tcb_desc *tcb_desc) | ||
669 | { | ||
670 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
671 | struct rtl_mac *rtlmac = rtl_mac(rtl_priv(hw)); | ||
672 | struct ieee80211_hdr *hdr = rtl_get_hdr(skb); | ||
673 | struct ieee80211_rate *txrate; | ||
674 | __le16 fc = hdr->frame_control; | ||
675 | |||
676 | txrate = ieee80211_get_tx_rate(hw, info); | ||
677 | tcb_desc->hw_rate = txrate->hw_value; | ||
678 | |||
679 | if (ieee80211_is_data(fc)) { | ||
680 | /* | ||
681 | *we set data rate INX 0 | ||
682 | *in rtl_rc.c if skb is special data or | ||
683 | *mgt which need low data rate. | ||
684 | */ | ||
685 | |||
686 | /* | ||
687 | *So tcb_desc->hw_rate is just used for | ||
688 | *special data and mgt frames | ||
689 | */ | ||
690 | if (info->control.rates[0].idx == 0 && | ||
691 | ieee80211_is_nullfunc(fc)) { | ||
692 | tcb_desc->use_driver_rate = true; | ||
693 | tcb_desc->ratr_index = RATR_INX_WIRELESS_MC; | ||
694 | |||
695 | tcb_desc->disable_ratefallback = 1; | ||
696 | } else { | ||
697 | /* | ||
698 | *because hw will nerver use hw_rate | ||
699 | *when tcb_desc->use_driver_rate = false | ||
700 | *so we never set highest N rate here, | ||
701 | *and N rate will all be controlled by FW | ||
702 | *when tcb_desc->use_driver_rate = false | ||
703 | */ | ||
704 | if (sta && (sta->ht_cap.ht_supported)) { | ||
705 | tcb_desc->hw_rate = _rtl_get_highest_n_rate(hw); | ||
706 | } else { | ||
707 | if (rtlmac->mode == WIRELESS_MODE_B) { | ||
708 | tcb_desc->hw_rate = | ||
709 | rtlpriv->cfg->maps[RTL_RC_CCK_RATE11M]; | ||
710 | } else { | ||
711 | tcb_desc->hw_rate = | ||
712 | rtlpriv->cfg->maps[RTL_RC_OFDM_RATE54M]; | ||
713 | } | ||
714 | } | ||
715 | } | ||
716 | |||
717 | if (is_multicast_ether_addr(ieee80211_get_DA(hdr))) | ||
718 | tcb_desc->multicast = 1; | ||
719 | else if (is_broadcast_ether_addr(ieee80211_get_DA(hdr))) | ||
720 | tcb_desc->broadcast = 1; | ||
721 | |||
722 | _rtl_txrate_selectmode(hw, sta, tcb_desc); | ||
723 | _rtl_query_bandwidth_mode(hw, sta, tcb_desc); | ||
724 | _rtl_qurey_shortpreamble_mode(hw, tcb_desc, info); | ||
725 | _rtl_query_shortgi(hw, sta, tcb_desc, info); | ||
726 | _rtl_query_protection_mode(hw, tcb_desc, info); | ||
727 | } else { | ||
728 | tcb_desc->use_driver_rate = true; | ||
729 | tcb_desc->ratr_index = RATR_INX_WIRELESS_MC; | ||
730 | tcb_desc->disable_ratefallback = 1; | ||
731 | tcb_desc->mac_id = 0; | ||
732 | tcb_desc->packet_bw = false; | ||
733 | } | ||
734 | } | ||
735 | EXPORT_SYMBOL(rtl_get_tcb_desc); | ||
736 | |||
737 | bool rtl_action_proc(struct ieee80211_hw *hw, struct sk_buff *skb, u8 is_tx) | ||
738 | { | ||
739 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
740 | struct ieee80211_hdr *hdr = rtl_get_hdr(skb); | ||
741 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
742 | __le16 fc = hdr->frame_control; | ||
743 | u8 *act = (u8 *) (((u8 *) skb->data + MAC80211_3ADDR_LEN)); | ||
744 | u8 category; | ||
745 | |||
746 | if (!ieee80211_is_action(fc)) | ||
747 | return true; | ||
748 | |||
749 | category = *act; | ||
750 | act++; | ||
751 | switch (category) { | ||
752 | case ACT_CAT_BA: | ||
753 | switch (*act) { | ||
754 | case ACT_ADDBAREQ: | ||
755 | if (mac->act_scanning) | ||
756 | return false; | ||
757 | |||
758 | RT_TRACE(rtlpriv, (COMP_SEND | COMP_RECV), DBG_DMESG, | ||
759 | ("%s ACT_ADDBAREQ From :" MAC_FMT "\n", | ||
760 | is_tx ? "Tx" : "Rx", MAC_ARG(hdr->addr2))); | ||
761 | break; | ||
762 | case ACT_ADDBARSP: | ||
763 | RT_TRACE(rtlpriv, (COMP_SEND | COMP_RECV), DBG_DMESG, | ||
764 | ("%s ACT_ADDBARSP From :" MAC_FMT "\n", | ||
765 | is_tx ? "Tx" : "Rx", MAC_ARG(hdr->addr2))); | ||
766 | break; | ||
767 | case ACT_DELBA: | ||
768 | RT_TRACE(rtlpriv, (COMP_SEND | COMP_RECV), DBG_DMESG, | ||
769 | ("ACT_ADDBADEL From :" MAC_FMT "\n", | ||
770 | MAC_ARG(hdr->addr2))); | ||
771 | break; | ||
772 | } | ||
773 | break; | ||
774 | default: | ||
775 | break; | ||
776 | } | ||
777 | |||
778 | return true; | ||
779 | } | ||
780 | |||
781 | /*should call before software enc*/ | ||
782 | u8 rtl_is_special_data(struct ieee80211_hw *hw, struct sk_buff *skb, u8 is_tx) | ||
783 | { | ||
784 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
785 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
786 | __le16 fc = rtl_get_fc(skb); | ||
787 | u16 ether_type; | ||
788 | u8 mac_hdr_len = ieee80211_get_hdrlen_from_skb(skb); | ||
789 | const struct iphdr *ip; | ||
790 | |||
791 | if (!ieee80211_is_data(fc)) | ||
792 | return false; | ||
793 | |||
794 | |||
795 | ip = (struct iphdr *)((u8 *) skb->data + mac_hdr_len + | ||
796 | SNAP_SIZE + PROTOC_TYPE_SIZE); | ||
797 | ether_type = *(u16 *) ((u8 *) skb->data + mac_hdr_len + SNAP_SIZE); | ||
798 | /* ether_type = ntohs(ether_type); */ | ||
799 | |||
800 | if (ETH_P_IP == ether_type) { | ||
801 | if (IPPROTO_UDP == ip->protocol) { | ||
802 | struct udphdr *udp = (struct udphdr *)((u8 *) ip + | ||
803 | (ip->ihl << 2)); | ||
804 | if (((((u8 *) udp)[1] == 68) && | ||
805 | (((u8 *) udp)[3] == 67)) || | ||
806 | ((((u8 *) udp)[1] == 67) && | ||
807 | (((u8 *) udp)[3] == 68))) { | ||
808 | /* | ||
809 | * 68 : UDP BOOTP client | ||
810 | * 67 : UDP BOOTP server | ||
811 | */ | ||
812 | RT_TRACE(rtlpriv, (COMP_SEND | COMP_RECV), | ||
813 | DBG_DMESG, ("dhcp %s !!\n", | ||
814 | (is_tx) ? "Tx" : "Rx")); | ||
815 | |||
816 | if (is_tx) { | ||
817 | rtl_lps_leave(hw); | ||
818 | ppsc->last_delaylps_stamp_jiffies = | ||
819 | jiffies; | ||
820 | } | ||
821 | |||
822 | return true; | ||
823 | } | ||
824 | } | ||
825 | } else if (ETH_P_ARP == ether_type) { | ||
826 | if (is_tx) { | ||
827 | rtl_lps_leave(hw); | ||
828 | ppsc->last_delaylps_stamp_jiffies = jiffies; | ||
829 | } | ||
830 | |||
831 | return true; | ||
832 | } else if (ETH_P_PAE == ether_type) { | ||
833 | RT_TRACE(rtlpriv, (COMP_SEND | COMP_RECV), DBG_DMESG, | ||
834 | ("802.1X %s EAPOL pkt!!\n", (is_tx) ? "Tx" : "Rx")); | ||
835 | |||
836 | if (is_tx) { | ||
837 | rtl_lps_leave(hw); | ||
838 | ppsc->last_delaylps_stamp_jiffies = jiffies; | ||
839 | } | ||
840 | |||
841 | return true; | ||
842 | } else if (ETH_P_IPV6 == ether_type) { | ||
843 | /* IPv6 */ | ||
844 | return true; | ||
845 | } | ||
846 | |||
847 | return false; | ||
848 | } | ||
849 | |||
850 | /********************************************************* | ||
851 | * | ||
852 | * functions called by core.c | ||
853 | * | ||
854 | *********************************************************/ | ||
855 | int rtl_tx_agg_start(struct ieee80211_hw *hw, | ||
856 | struct ieee80211_sta *sta, u16 tid, u16 *ssn) | ||
857 | { | ||
858 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
859 | struct rtl_tid_data *tid_data; | ||
860 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
861 | struct rtl_sta_info *sta_entry = NULL; | ||
862 | |||
863 | if (sta == NULL) | ||
864 | return -EINVAL; | ||
865 | |||
866 | if (unlikely(tid >= MAX_TID_COUNT)) | ||
867 | return -EINVAL; | ||
868 | |||
869 | sta_entry = (struct rtl_sta_info *)sta->drv_priv; | ||
870 | if (!sta_entry) | ||
871 | return -ENXIO; | ||
872 | tid_data = &sta_entry->tids[tid]; | ||
873 | |||
874 | RT_TRACE(rtlpriv, COMP_SEND, DBG_DMESG, | ||
875 | ("on ra = %pM tid = %d seq:%d\n", sta->addr, tid, | ||
876 | tid_data->seq_number)); | ||
877 | |||
878 | *ssn = tid_data->seq_number; | ||
879 | tid_data->agg.agg_state = RTL_AGG_START; | ||
880 | |||
881 | ieee80211_start_tx_ba_cb_irqsafe(mac->vif, sta->addr, tid); | ||
882 | |||
883 | return 0; | ||
884 | } | ||
885 | |||
886 | int rtl_tx_agg_stop(struct ieee80211_hw *hw, | ||
887 | struct ieee80211_sta *sta, u16 tid) | ||
888 | { | ||
889 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
890 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
891 | struct rtl_tid_data *tid_data; | ||
892 | struct rtl_sta_info *sta_entry = NULL; | ||
893 | |||
894 | if (sta == NULL) | ||
895 | return -EINVAL; | ||
896 | |||
897 | if (!sta->addr) { | ||
898 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("ra = NULL\n")); | ||
899 | return -EINVAL; | ||
900 | } | ||
901 | |||
902 | RT_TRACE(rtlpriv, COMP_SEND, DBG_DMESG, | ||
903 | ("on ra = %pM tid = %d\n", sta->addr, tid)); | ||
904 | |||
905 | if (unlikely(tid >= MAX_TID_COUNT)) | ||
906 | return -EINVAL; | ||
907 | |||
908 | sta_entry = (struct rtl_sta_info *)sta->drv_priv; | ||
909 | tid_data = &sta_entry->tids[tid]; | ||
910 | sta_entry->tids[tid].agg.agg_state = RTL_AGG_STOP; | ||
911 | |||
912 | ieee80211_stop_tx_ba_cb_irqsafe(mac->vif, sta->addr, tid); | ||
913 | |||
914 | return 0; | ||
915 | } | ||
916 | |||
917 | int rtl_tx_agg_oper(struct ieee80211_hw *hw, | ||
918 | struct ieee80211_sta *sta, u16 tid) | ||
919 | { | ||
920 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
921 | struct rtl_tid_data *tid_data; | ||
922 | struct rtl_sta_info *sta_entry = NULL; | ||
923 | |||
924 | if (sta == NULL) | ||
925 | return -EINVAL; | ||
926 | |||
927 | if (!sta->addr) { | ||
928 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("ra = NULL\n")); | ||
929 | return -EINVAL; | ||
930 | } | ||
931 | |||
932 | RT_TRACE(rtlpriv, COMP_SEND, DBG_DMESG, | ||
933 | ("on ra = %pM tid = %d\n", sta->addr, tid)); | ||
934 | |||
935 | if (unlikely(tid >= MAX_TID_COUNT)) | ||
936 | return -EINVAL; | ||
937 | |||
938 | sta_entry = (struct rtl_sta_info *)sta->drv_priv; | ||
939 | tid_data = &sta_entry->tids[tid]; | ||
940 | sta_entry->tids[tid].agg.agg_state = RTL_AGG_OPERATIONAL; | ||
941 | |||
942 | return 0; | ||
943 | } | ||
944 | |||
945 | /********************************************************* | ||
946 | * | ||
947 | * wq & timer callback functions | ||
948 | * | ||
949 | *********************************************************/ | ||
950 | void rtl_watchdog_wq_callback(void *data) | ||
951 | { | ||
952 | struct rtl_works *rtlworks = container_of_dwork_rtl(data, | ||
953 | struct rtl_works, | ||
954 | watchdog_wq); | ||
955 | struct ieee80211_hw *hw = rtlworks->hw; | ||
956 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
957 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
958 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
959 | bool busytraffic = false; | ||
960 | bool higher_busytraffic = false; | ||
961 | bool higher_busyrxtraffic = false; | ||
962 | u8 idx, tid; | ||
963 | u32 rx_cnt_inp4eriod = 0; | ||
964 | u32 tx_cnt_inp4eriod = 0; | ||
965 | u32 aver_rx_cnt_inperiod = 0; | ||
966 | u32 aver_tx_cnt_inperiod = 0; | ||
967 | u32 aver_tidtx_inperiod[MAX_TID_COUNT] = {0}; | ||
968 | u32 tidtx_inp4eriod[MAX_TID_COUNT] = {0}; | ||
969 | bool enter_ps = false; | ||
970 | |||
971 | if (is_hal_stop(rtlhal)) | ||
972 | return; | ||
973 | |||
974 | /* <1> Determine if action frame is allowed */ | ||
975 | if (mac->link_state > MAC80211_NOLINK) { | ||
976 | if (mac->cnt_after_linked < 20) | ||
977 | mac->cnt_after_linked++; | ||
978 | } else { | ||
979 | mac->cnt_after_linked = 0; | ||
980 | } | ||
981 | |||
982 | /* | ||
983 | *<3> to check if traffic busy, if | ||
984 | * busytraffic we don't change channel | ||
985 | */ | ||
986 | if (mac->link_state >= MAC80211_LINKED) { | ||
987 | |||
988 | /* (1) get aver_rx_cnt_inperiod & aver_tx_cnt_inperiod */ | ||
989 | for (idx = 0; idx <= 2; idx++) { | ||
990 | rtlpriv->link_info.num_rx_in4period[idx] = | ||
991 | rtlpriv->link_info.num_rx_in4period[idx + 1]; | ||
992 | rtlpriv->link_info.num_tx_in4period[idx] = | ||
993 | rtlpriv->link_info.num_tx_in4period[idx + 1]; | ||
994 | } | ||
995 | rtlpriv->link_info.num_rx_in4period[3] = | ||
996 | rtlpriv->link_info.num_rx_inperiod; | ||
997 | rtlpriv->link_info.num_tx_in4period[3] = | ||
998 | rtlpriv->link_info.num_tx_inperiod; | ||
999 | for (idx = 0; idx <= 3; idx++) { | ||
1000 | rx_cnt_inp4eriod += | ||
1001 | rtlpriv->link_info.num_rx_in4period[idx]; | ||
1002 | tx_cnt_inp4eriod += | ||
1003 | rtlpriv->link_info.num_tx_in4period[idx]; | ||
1004 | } | ||
1005 | aver_rx_cnt_inperiod = rx_cnt_inp4eriod / 4; | ||
1006 | aver_tx_cnt_inperiod = tx_cnt_inp4eriod / 4; | ||
1007 | |||
1008 | /* (2) check traffic busy */ | ||
1009 | if (aver_rx_cnt_inperiod > 100 || aver_tx_cnt_inperiod > 100) | ||
1010 | busytraffic = true; | ||
1011 | |||
1012 | /* Higher Tx/Rx data. */ | ||
1013 | if (aver_rx_cnt_inperiod > 4000 || | ||
1014 | aver_tx_cnt_inperiod > 4000) { | ||
1015 | higher_busytraffic = true; | ||
1016 | |||
1017 | /* Extremely high Rx data. */ | ||
1018 | if (aver_rx_cnt_inperiod > 5000) | ||
1019 | higher_busyrxtraffic = true; | ||
1020 | } | ||
1021 | |||
1022 | /* check every tid's tx traffic */ | ||
1023 | for (tid = 0; tid <= 7; tid++) { | ||
1024 | for (idx = 0; idx <= 2; idx++) | ||
1025 | rtlpriv->link_info.tidtx_in4period[tid][idx] = | ||
1026 | rtlpriv->link_info.tidtx_in4period[tid] | ||
1027 | [idx + 1]; | ||
1028 | rtlpriv->link_info.tidtx_in4period[tid][3] = | ||
1029 | rtlpriv->link_info.tidtx_inperiod[tid]; | ||
1030 | |||
1031 | for (idx = 0; idx <= 3; idx++) | ||
1032 | tidtx_inp4eriod[tid] += | ||
1033 | rtlpriv->link_info.tidtx_in4period[tid][idx]; | ||
1034 | aver_tidtx_inperiod[tid] = tidtx_inp4eriod[tid] / 4; | ||
1035 | if (aver_tidtx_inperiod[tid] > 5000) | ||
1036 | rtlpriv->link_info.higher_busytxtraffic[tid] = | ||
1037 | true; | ||
1038 | else | ||
1039 | rtlpriv->link_info.higher_busytxtraffic[tid] = | ||
1040 | false; | ||
1041 | } | ||
1042 | |||
1043 | if (((rtlpriv->link_info.num_rx_inperiod + | ||
1044 | rtlpriv->link_info.num_tx_inperiod) > 8) || | ||
1045 | (rtlpriv->link_info.num_rx_inperiod > 2)) | ||
1046 | enter_ps = false; | ||
1047 | else | ||
1048 | enter_ps = true; | ||
1049 | |||
1050 | /* LeisurePS only work in infra mode. */ | ||
1051 | if (enter_ps) | ||
1052 | rtl_lps_enter(hw); | ||
1053 | else | ||
1054 | rtl_lps_leave(hw); | ||
1055 | } | ||
1056 | |||
1057 | rtlpriv->link_info.num_rx_inperiod = 0; | ||
1058 | rtlpriv->link_info.num_tx_inperiod = 0; | ||
1059 | for (tid = 0; tid <= 7; tid++) | ||
1060 | rtlpriv->link_info.tidtx_inperiod[tid] = 0; | ||
1061 | |||
1062 | rtlpriv->link_info.busytraffic = busytraffic; | ||
1063 | rtlpriv->link_info.higher_busytraffic = higher_busytraffic; | ||
1064 | rtlpriv->link_info.higher_busyrxtraffic = higher_busyrxtraffic; | ||
1065 | |||
1066 | /* <3> DM */ | ||
1067 | rtlpriv->cfg->ops->dm_watchdog(hw); | ||
1068 | } | ||
1069 | |||
1070 | void rtl_watch_dog_timer_callback(unsigned long data) | ||
1071 | { | ||
1072 | struct ieee80211_hw *hw = (struct ieee80211_hw *)data; | ||
1073 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1074 | |||
1075 | queue_delayed_work(rtlpriv->works.rtl_wq, | ||
1076 | &rtlpriv->works.watchdog_wq, 0); | ||
1077 | |||
1078 | mod_timer(&rtlpriv->works.watchdog_timer, | ||
1079 | jiffies + MSECS(RTL_WATCH_DOG_TIME)); | ||
1080 | } | ||
1081 | |||
1082 | /********************************************************* | ||
1083 | * | ||
1084 | * frame process functions | ||
1085 | * | ||
1086 | *********************************************************/ | ||
1087 | u8 *rtl_find_ie(u8 *data, unsigned int len, u8 ie) | ||
1088 | { | ||
1089 | struct ieee80211_mgmt *mgmt = (void *)data; | ||
1090 | u8 *pos, *end; | ||
1091 | |||
1092 | pos = (u8 *)mgmt->u.beacon.variable; | ||
1093 | end = data + len; | ||
1094 | while (pos < end) { | ||
1095 | if (pos + 2 + pos[1] > end) | ||
1096 | return NULL; | ||
1097 | |||
1098 | if (pos[0] == ie) | ||
1099 | return pos; | ||
1100 | |||
1101 | pos += 2 + pos[1]; | ||
1102 | } | ||
1103 | return NULL; | ||
1104 | } | ||
1105 | |||
1106 | /* when we use 2 rx ants we send IEEE80211_SMPS_OFF */ | ||
1107 | /* when we use 1 rx ant we send IEEE80211_SMPS_STATIC */ | ||
1108 | static struct sk_buff *rtl_make_smps_action(struct ieee80211_hw *hw, | ||
1109 | enum ieee80211_smps_mode smps, u8 *da, u8 *bssid) | ||
1110 | { | ||
1111 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1112 | struct sk_buff *skb; | ||
1113 | struct ieee80211_mgmt *action_frame; | ||
1114 | |||
1115 | /* 27 = header + category + action + smps mode */ | ||
1116 | skb = dev_alloc_skb(27 + hw->extra_tx_headroom); | ||
1117 | if (!skb) | ||
1118 | return NULL; | ||
1119 | |||
1120 | skb_reserve(skb, hw->extra_tx_headroom); | ||
1121 | action_frame = (void *)skb_put(skb, 27); | ||
1122 | memset(action_frame, 0, 27); | ||
1123 | memcpy(action_frame->da, da, ETH_ALEN); | ||
1124 | memcpy(action_frame->sa, rtlefuse->dev_addr, ETH_ALEN); | ||
1125 | memcpy(action_frame->bssid, bssid, ETH_ALEN); | ||
1126 | action_frame->frame_control = cpu_to_le16(IEEE80211_FTYPE_MGMT | | ||
1127 | IEEE80211_STYPE_ACTION); | ||
1128 | action_frame->u.action.category = WLAN_CATEGORY_HT; | ||
1129 | action_frame->u.action.u.ht_smps.action = WLAN_HT_ACTION_SMPS; | ||
1130 | switch (smps) { | ||
1131 | case IEEE80211_SMPS_AUTOMATIC:/* 0 */ | ||
1132 | case IEEE80211_SMPS_NUM_MODES:/* 4 */ | ||
1133 | WARN_ON(1); | ||
1134 | case IEEE80211_SMPS_OFF:/* 1 */ /*MIMO_PS_NOLIMIT*/ | ||
1135 | action_frame->u.action.u.ht_smps.smps_control = | ||
1136 | WLAN_HT_SMPS_CONTROL_DISABLED;/* 0 */ | ||
1137 | break; | ||
1138 | case IEEE80211_SMPS_STATIC:/* 2 */ /*MIMO_PS_STATIC*/ | ||
1139 | action_frame->u.action.u.ht_smps.smps_control = | ||
1140 | WLAN_HT_SMPS_CONTROL_STATIC;/* 1 */ | ||
1141 | break; | ||
1142 | case IEEE80211_SMPS_DYNAMIC:/* 3 */ /*MIMO_PS_DYNAMIC*/ | ||
1143 | action_frame->u.action.u.ht_smps.smps_control = | ||
1144 | WLAN_HT_SMPS_CONTROL_DYNAMIC;/* 3 */ | ||
1145 | break; | ||
1146 | } | ||
1147 | |||
1148 | return skb; | ||
1149 | } | ||
1150 | |||
1151 | int rtl_send_smps_action(struct ieee80211_hw *hw, | ||
1152 | struct ieee80211_sta *sta, u8 *da, u8 *bssid, | ||
1153 | enum ieee80211_smps_mode smps) | ||
1154 | { | ||
1155 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1156 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1157 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1158 | struct sk_buff *skb = rtl_make_smps_action(hw, smps, da, bssid); | ||
1159 | struct rtl_tcb_desc tcb_desc; | ||
1160 | memset(&tcb_desc, 0, sizeof(struct rtl_tcb_desc)); | ||
1161 | |||
1162 | if (rtlpriv->mac80211.act_scanning) | ||
1163 | goto err_free; | ||
1164 | |||
1165 | if (!sta) | ||
1166 | goto err_free; | ||
1167 | |||
1168 | if (unlikely(is_hal_stop(rtlhal) || ppsc->rfpwr_state != ERFON)) | ||
1169 | goto err_free; | ||
1170 | |||
1171 | if (!test_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status)) | ||
1172 | goto err_free; | ||
1173 | |||
1174 | /* this is a type = mgmt * stype = action frame */ | ||
1175 | if (skb) { | ||
1176 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
1177 | struct rtl_sta_info *sta_entry = | ||
1178 | (struct rtl_sta_info *) sta->drv_priv; | ||
1179 | sta_entry->mimo_ps = smps; | ||
1180 | rtlpriv->cfg->ops->update_rate_tbl(hw, sta, 0); | ||
1181 | |||
1182 | info->control.rates[0].idx = 0; | ||
1183 | info->control.sta = sta; | ||
1184 | info->band = hw->conf.channel->band; | ||
1185 | rtlpriv->intf_ops->adapter_tx(hw, skb, &tcb_desc); | ||
1186 | } | ||
1187 | err_free: | ||
1188 | return 0; | ||
1189 | } | ||
1190 | |||
1191 | /********************************************************* | ||
1192 | * | ||
1193 | * IOT functions | ||
1194 | * | ||
1195 | *********************************************************/ | ||
1196 | static bool rtl_chk_vendor_ouisub(struct ieee80211_hw *hw, | ||
1197 | struct octet_string vendor_ie) | ||
1198 | { | ||
1199 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1200 | bool matched = false; | ||
1201 | static u8 athcap_1[] = { 0x00, 0x03, 0x7F }; | ||
1202 | static u8 athcap_2[] = { 0x00, 0x13, 0x74 }; | ||
1203 | static u8 broadcap_1[] = { 0x00, 0x10, 0x18 }; | ||
1204 | static u8 broadcap_2[] = { 0x00, 0x0a, 0xf7 }; | ||
1205 | static u8 broadcap_3[] = { 0x00, 0x05, 0xb5 }; | ||
1206 | static u8 racap[] = { 0x00, 0x0c, 0x43 }; | ||
1207 | static u8 ciscocap[] = { 0x00, 0x40, 0x96 }; | ||
1208 | static u8 marvcap[] = { 0x00, 0x50, 0x43 }; | ||
1209 | |||
1210 | if (memcmp(vendor_ie.octet, athcap_1, 3) == 0 || | ||
1211 | memcmp(vendor_ie.octet, athcap_2, 3) == 0) { | ||
1212 | rtlpriv->mac80211.vendor = PEER_ATH; | ||
1213 | matched = true; | ||
1214 | } else if (memcmp(vendor_ie.octet, broadcap_1, 3) == 0 || | ||
1215 | memcmp(vendor_ie.octet, broadcap_2, 3) == 0 || | ||
1216 | memcmp(vendor_ie.octet, broadcap_3, 3) == 0) { | ||
1217 | rtlpriv->mac80211.vendor = PEER_BROAD; | ||
1218 | matched = true; | ||
1219 | } else if (memcmp(vendor_ie.octet, racap, 3) == 0) { | ||
1220 | rtlpriv->mac80211.vendor = PEER_RAL; | ||
1221 | matched = true; | ||
1222 | } else if (memcmp(vendor_ie.octet, ciscocap, 3) == 0) { | ||
1223 | rtlpriv->mac80211.vendor = PEER_CISCO; | ||
1224 | matched = true; | ||
1225 | } else if (memcmp(vendor_ie.octet, marvcap, 3) == 0) { | ||
1226 | rtlpriv->mac80211.vendor = PEER_MARV; | ||
1227 | matched = true; | ||
1228 | } | ||
1229 | |||
1230 | return matched; | ||
1231 | } | ||
1232 | |||
1233 | static bool rtl_find_221_ie(struct ieee80211_hw *hw, u8 *data, | ||
1234 | unsigned int len) | ||
1235 | { | ||
1236 | struct ieee80211_mgmt *mgmt = (void *)data; | ||
1237 | struct octet_string vendor_ie; | ||
1238 | u8 *pos, *end; | ||
1239 | |||
1240 | pos = (u8 *)mgmt->u.beacon.variable; | ||
1241 | end = data + len; | ||
1242 | while (pos < end) { | ||
1243 | if (pos[0] == 221) { | ||
1244 | vendor_ie.length = pos[1]; | ||
1245 | vendor_ie.octet = &pos[2]; | ||
1246 | if (rtl_chk_vendor_ouisub(hw, vendor_ie)) | ||
1247 | return true; | ||
1248 | } | ||
1249 | |||
1250 | if (pos + 2 + pos[1] > end) | ||
1251 | return false; | ||
1252 | |||
1253 | pos += 2 + pos[1]; | ||
1254 | } | ||
1255 | return false; | ||
1256 | } | ||
1257 | |||
1258 | void rtl_recognize_peer(struct ieee80211_hw *hw, u8 *data, unsigned int len) | ||
1259 | { | ||
1260 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1261 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1262 | struct ieee80211_hdr *hdr = (void *)data; | ||
1263 | u32 vendor = PEER_UNKNOWN; | ||
1264 | |||
1265 | static u8 ap3_1[3] = { 0x00, 0x14, 0xbf }; | ||
1266 | static u8 ap3_2[3] = { 0x00, 0x1a, 0x70 }; | ||
1267 | static u8 ap3_3[3] = { 0x00, 0x1d, 0x7e }; | ||
1268 | static u8 ap4_1[3] = { 0x00, 0x90, 0xcc }; | ||
1269 | static u8 ap4_2[3] = { 0x00, 0x0e, 0x2e }; | ||
1270 | static u8 ap4_3[3] = { 0x00, 0x18, 0x02 }; | ||
1271 | static u8 ap4_4[3] = { 0x00, 0x17, 0x3f }; | ||
1272 | static u8 ap4_5[3] = { 0x00, 0x1c, 0xdf }; | ||
1273 | static u8 ap5_1[3] = { 0x00, 0x1c, 0xf0 }; | ||
1274 | static u8 ap5_2[3] = { 0x00, 0x21, 0x91 }; | ||
1275 | static u8 ap5_3[3] = { 0x00, 0x24, 0x01 }; | ||
1276 | static u8 ap5_4[3] = { 0x00, 0x15, 0xe9 }; | ||
1277 | static u8 ap5_5[3] = { 0x00, 0x17, 0x9A }; | ||
1278 | static u8 ap5_6[3] = { 0x00, 0x18, 0xE7 }; | ||
1279 | static u8 ap6_1[3] = { 0x00, 0x17, 0x94 }; | ||
1280 | static u8 ap7_1[3] = { 0x00, 0x14, 0xa4 }; | ||
1281 | |||
1282 | if (mac->opmode != NL80211_IFTYPE_STATION) | ||
1283 | return; | ||
1284 | |||
1285 | if (mac->link_state == MAC80211_NOLINK) { | ||
1286 | mac->vendor = PEER_UNKNOWN; | ||
1287 | return; | ||
1288 | } | ||
1289 | |||
1290 | if (mac->cnt_after_linked > 2) | ||
1291 | return; | ||
1292 | |||
1293 | /* check if this really is a beacon */ | ||
1294 | if (!ieee80211_is_beacon(hdr->frame_control)) | ||
1295 | return; | ||
1296 | |||
1297 | /* min. beacon length + FCS_LEN */ | ||
1298 | if (len <= 40 + FCS_LEN) | ||
1299 | return; | ||
1300 | |||
1301 | /* and only beacons from the associated BSSID, please */ | ||
1302 | if (compare_ether_addr(hdr->addr3, rtlpriv->mac80211.bssid)) | ||
1303 | return; | ||
1304 | |||
1305 | if (rtl_find_221_ie(hw, data, len)) | ||
1306 | vendor = mac->vendor; | ||
1307 | |||
1308 | if ((memcmp(mac->bssid, ap5_1, 3) == 0) || | ||
1309 | (memcmp(mac->bssid, ap5_2, 3) == 0) || | ||
1310 | (memcmp(mac->bssid, ap5_3, 3) == 0) || | ||
1311 | (memcmp(mac->bssid, ap5_4, 3) == 0) || | ||
1312 | (memcmp(mac->bssid, ap5_5, 3) == 0) || | ||
1313 | (memcmp(mac->bssid, ap5_6, 3) == 0) || | ||
1314 | vendor == PEER_ATH) { | ||
1315 | vendor = PEER_ATH; | ||
1316 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("=>ath find\n")); | ||
1317 | } else if ((memcmp(mac->bssid, ap4_4, 3) == 0) || | ||
1318 | (memcmp(mac->bssid, ap4_5, 3) == 0) || | ||
1319 | (memcmp(mac->bssid, ap4_1, 3) == 0) || | ||
1320 | (memcmp(mac->bssid, ap4_2, 3) == 0) || | ||
1321 | (memcmp(mac->bssid, ap4_3, 3) == 0) || | ||
1322 | vendor == PEER_RAL) { | ||
1323 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("=>ral findn\n")); | ||
1324 | vendor = PEER_RAL; | ||
1325 | } else if (memcmp(mac->bssid, ap6_1, 3) == 0 || | ||
1326 | vendor == PEER_CISCO) { | ||
1327 | vendor = PEER_CISCO; | ||
1328 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("=>cisco find\n")); | ||
1329 | } else if ((memcmp(mac->bssid, ap3_1, 3) == 0) || | ||
1330 | (memcmp(mac->bssid, ap3_2, 3) == 0) || | ||
1331 | (memcmp(mac->bssid, ap3_3, 3) == 0) || | ||
1332 | vendor == PEER_BROAD) { | ||
1333 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("=>broad find\n")); | ||
1334 | vendor = PEER_BROAD; | ||
1335 | } else if (memcmp(mac->bssid, ap7_1, 3) == 0 || | ||
1336 | vendor == PEER_MARV) { | ||
1337 | vendor = PEER_MARV; | ||
1338 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("=>marv find\n")); | ||
1339 | } | ||
1340 | |||
1341 | mac->vendor = vendor; | ||
1342 | } | ||
1343 | |||
1344 | /********************************************************* | ||
1345 | * | ||
1346 | * sysfs functions | ||
1347 | * | ||
1348 | *********************************************************/ | ||
1349 | static ssize_t rtl_show_debug_level(struct device *d, | ||
1350 | struct device_attribute *attr, char *buf) | ||
1351 | { | ||
1352 | struct ieee80211_hw *hw = dev_get_drvdata(d); | ||
1353 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1354 | |||
1355 | return sprintf(buf, "0x%08X\n", rtlpriv->dbg.global_debuglevel); | ||
1356 | } | ||
1357 | |||
1358 | static ssize_t rtl_store_debug_level(struct device *d, | ||
1359 | struct device_attribute *attr, | ||
1360 | const char *buf, size_t count) | ||
1361 | { | ||
1362 | struct ieee80211_hw *hw = dev_get_drvdata(d); | ||
1363 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1364 | unsigned long val; | ||
1365 | int ret; | ||
1366 | |||
1367 | ret = strict_strtoul(buf, 0, &val); | ||
1368 | if (ret) { | ||
1369 | printk(KERN_DEBUG "%s is not in hex or decimal form.\n", buf); | ||
1370 | } else { | ||
1371 | rtlpriv->dbg.global_debuglevel = val; | ||
1372 | printk(KERN_DEBUG "debuglevel:%x\n", | ||
1373 | rtlpriv->dbg.global_debuglevel); | ||
1374 | } | ||
1375 | |||
1376 | return strnlen(buf, count); | ||
1377 | } | ||
1378 | |||
1379 | static DEVICE_ATTR(debug_level, S_IWUSR | S_IRUGO, | ||
1380 | rtl_show_debug_level, rtl_store_debug_level); | ||
1381 | |||
1382 | static struct attribute *rtl_sysfs_entries[] = { | ||
1383 | |||
1384 | &dev_attr_debug_level.attr, | ||
1385 | |||
1386 | NULL | ||
1387 | }; | ||
1388 | |||
1389 | /* | ||
1390 | * "name" is folder name witch will be | ||
1391 | * put in device directory like : | ||
1392 | * sys/devices/pci0000:00/0000:00:1c.4/ | ||
1393 | * 0000:06:00.0/rtl_sysfs | ||
1394 | */ | ||
1395 | struct attribute_group rtl_attribute_group = { | ||
1396 | .name = "rtlsysfs", | ||
1397 | .attrs = rtl_sysfs_entries, | ||
1398 | }; | ||
1399 | |||
1400 | MODULE_AUTHOR("lizhaoming <chaoming_li@realsil.com.cn>"); | ||
1401 | MODULE_AUTHOR("Realtek WlanFAE <wlanfae@realtek.com>"); | ||
1402 | MODULE_AUTHOR("Larry Finger <Larry.FInger@lwfinger.net>"); | ||
1403 | MODULE_LICENSE("GPL"); | ||
1404 | MODULE_DESCRIPTION("Realtek 802.11n PCI wireless core"); | ||
1405 | |||
1406 | static int __init rtl_core_module_init(void) | ||
1407 | { | ||
1408 | if (rtl_rate_control_register()) | ||
1409 | printk(KERN_ERR "rtlwifi: Unable to register rtl_rc," | ||
1410 | "use default RC !!\n"); | ||
1411 | |||
1412 | return 0; | ||
1413 | } | ||
1414 | |||
1415 | static void __exit rtl_core_module_exit(void) | ||
1416 | { | ||
1417 | /*RC*/ | ||
1418 | rtl_rate_control_unregister(); | ||
1419 | } | ||
1420 | |||
1421 | module_init(rtl_core_module_init); | ||
1422 | module_exit(rtl_core_module_exit); | ||
diff --git a/drivers/net/wireless/rtlwifi/base.h b/drivers/net/wireless/rtlwifi/base.h new file mode 100644 index 000000000000..a91f3eee59c8 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/base.h | |||
@@ -0,0 +1,143 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_BASE_H__ | ||
31 | #define __RTL_BASE_H__ | ||
32 | |||
33 | enum ap_peer { | ||
34 | PEER_UNKNOWN = 0, | ||
35 | PEER_RTL = 1, | ||
36 | PEER_RTL_92SE = 2, | ||
37 | PEER_BROAD = 3, | ||
38 | PEER_RAL = 4, | ||
39 | PEER_ATH = 5, | ||
40 | PEER_CISCO = 6, | ||
41 | PEER_MARV = 7, | ||
42 | PEER_AIRGO = 9, | ||
43 | PEER_MAX = 10, | ||
44 | } ; | ||
45 | |||
46 | #define RTL_DUMMY_OFFSET 0 | ||
47 | #define RTL_DUMMY_UNIT 8 | ||
48 | #define RTL_TX_DUMMY_SIZE (RTL_DUMMY_OFFSET * RTL_DUMMY_UNIT) | ||
49 | #define RTL_TX_DESC_SIZE 32 | ||
50 | #define RTL_TX_HEADER_SIZE (RTL_TX_DESC_SIZE + RTL_TX_DUMMY_SIZE) | ||
51 | |||
52 | #define HT_AMSDU_SIZE_4K 3839 | ||
53 | #define HT_AMSDU_SIZE_8K 7935 | ||
54 | |||
55 | #define MAX_BIT_RATE_40MHZ_MCS15 300 /* Mbps */ | ||
56 | #define MAX_BIT_RATE_40MHZ_MCS7 150 /* Mbps */ | ||
57 | |||
58 | #define RTL_RATE_COUNT_LEGACY 12 | ||
59 | #define RTL_CHANNEL_COUNT 14 | ||
60 | |||
61 | #define FRAME_OFFSET_FRAME_CONTROL 0 | ||
62 | #define FRAME_OFFSET_DURATION 2 | ||
63 | #define FRAME_OFFSET_ADDRESS1 4 | ||
64 | #define FRAME_OFFSET_ADDRESS2 10 | ||
65 | #define FRAME_OFFSET_ADDRESS3 16 | ||
66 | #define FRAME_OFFSET_SEQUENCE 22 | ||
67 | #define FRAME_OFFSET_ADDRESS4 24 | ||
68 | |||
69 | #define SET_80211_HDR_FRAME_CONTROL(_hdr, _val) \ | ||
70 | WRITEEF2BYTE(_hdr, _val) | ||
71 | #define SET_80211_HDR_TYPE_AND_SUBTYPE(_hdr, _val) \ | ||
72 | WRITEEF1BYTE(_hdr, _val) | ||
73 | #define SET_80211_HDR_PWR_MGNT(_hdr, _val) \ | ||
74 | SET_BITS_TO_LE_2BYTE(_hdr, 12, 1, _val) | ||
75 | #define SET_80211_HDR_TO_DS(_hdr, _val) \ | ||
76 | SET_BITS_TO_LE_2BYTE(_hdr, 8, 1, _val) | ||
77 | |||
78 | #define SET_80211_PS_POLL_AID(_hdr, _val) \ | ||
79 | (*(u16 *)((u8 *)(_hdr) + 2) = le16_to_cpu(_val)) | ||
80 | #define SET_80211_PS_POLL_BSSID(_hdr, _val) \ | ||
81 | memcpy(((u8 *)(_hdr)) + 4, (u8 *)(_val), ETH_ALEN) | ||
82 | #define SET_80211_PS_POLL_TA(_hdr, _val) \ | ||
83 | memcpy(((u8 *)(_hdr)) + 10, (u8 *)(_val), ETH_ALEN) | ||
84 | |||
85 | #define SET_80211_HDR_DURATION(_hdr, _val) \ | ||
86 | (*(u16 *)((u8 *)(_hdr) + FRAME_OFFSET_DURATION) = le16_to_cpu(_val)) | ||
87 | #define SET_80211_HDR_ADDRESS1(_hdr, _val) \ | ||
88 | CP_MACADDR((u8 *)(_hdr)+FRAME_OFFSET_ADDRESS1, (u8 *)(_val)) | ||
89 | #define SET_80211_HDR_ADDRESS2(_hdr, _val) \ | ||
90 | CP_MACADDR((u8 *)(_hdr)+FRAME_OFFSET_ADDRESS2, (u8 *)(_val)) | ||
91 | #define SET_80211_HDR_ADDRESS3(_hdr, _val) \ | ||
92 | CP_MACADDR((u8 *)(_hdr)+FRAME_OFFSET_ADDRESS3, (u8 *)(_val)) | ||
93 | #define SET_80211_HDR_FRAGMENT_SEQUENCE(_hdr, _val) \ | ||
94 | WRITEEF2BYTE((u8 *)(_hdr)+FRAME_OFFSET_SEQUENCE, _val) | ||
95 | |||
96 | #define SET_BEACON_PROBE_RSP_TIME_STAMP_LOW(__phdr, __val) \ | ||
97 | WRITEEF4BYTE(((u8 *)(__phdr)) + 24, __val) | ||
98 | #define SET_BEACON_PROBE_RSP_TIME_STAMP_HIGH(__phdr, __val) \ | ||
99 | WRITEEF4BYTE(((u8 *)(__phdr)) + 28, __val) | ||
100 | #define SET_BEACON_PROBE_RSP_BEACON_INTERVAL(__phdr, __val) \ | ||
101 | WRITEEF2BYTE(((u8 *)(__phdr)) + 32, __val) | ||
102 | #define GET_BEACON_PROBE_RSP_CAPABILITY_INFO(__phdr) \ | ||
103 | READEF2BYTE(((u8 *)(__phdr)) + 34) | ||
104 | #define SET_BEACON_PROBE_RSP_CAPABILITY_INFO(__phdr, __val) \ | ||
105 | WRITEEF2BYTE(((u8 *)(__phdr)) + 34, __val) | ||
106 | #define MASK_BEACON_PROBE_RSP_CAPABILITY_INFO(__phdr, __val) \ | ||
107 | SET_BEACON_PROBE_RSP_CAPABILITY_INFO(__phdr, \ | ||
108 | (GET_BEACON_PROBE_RSP_CAPABILITY_INFO(__phdr) & (~(__val)))) | ||
109 | |||
110 | int rtl_init_core(struct ieee80211_hw *hw); | ||
111 | void rtl_deinit_core(struct ieee80211_hw *hw); | ||
112 | void rtl_init_rx_config(struct ieee80211_hw *hw); | ||
113 | void rtl_init_rfkill(struct ieee80211_hw *hw); | ||
114 | void rtl_deinit_rfkill(struct ieee80211_hw *hw); | ||
115 | |||
116 | void rtl_watch_dog_timer_callback(unsigned long data); | ||
117 | void rtl_deinit_deferred_work(struct ieee80211_hw *hw); | ||
118 | |||
119 | bool rtl_action_proc(struct ieee80211_hw *hw, struct sk_buff *skb, u8 is_tx); | ||
120 | u8 rtl_is_special_data(struct ieee80211_hw *hw, struct sk_buff *skb, u8 is_tx); | ||
121 | |||
122 | void rtl_watch_dog_timer_callback(unsigned long data); | ||
123 | int rtl_tx_agg_start(struct ieee80211_hw *hw, struct ieee80211_sta *sta, | ||
124 | u16 tid, u16 *ssn); | ||
125 | int rtl_tx_agg_stop(struct ieee80211_hw *hw, struct ieee80211_sta *sta, | ||
126 | u16 tid); | ||
127 | int rtl_tx_agg_oper(struct ieee80211_hw *hw, struct ieee80211_sta *sta, | ||
128 | u16 tid); | ||
129 | void rtl_watchdog_wq_callback(void *data); | ||
130 | |||
131 | void rtl_get_tcb_desc(struct ieee80211_hw *hw, | ||
132 | struct ieee80211_tx_info *info, | ||
133 | struct ieee80211_sta *sta, | ||
134 | struct sk_buff *skb, struct rtl_tcb_desc *tcb_desc); | ||
135 | |||
136 | int rtl_send_smps_action(struct ieee80211_hw *hw, | ||
137 | struct ieee80211_sta *sta, u8 *da, u8 *bssid, | ||
138 | enum ieee80211_smps_mode smps); | ||
139 | u8 *rtl_find_ie(u8 *data, unsigned int len, u8 ie); | ||
140 | void rtl_recognize_peer(struct ieee80211_hw *hw, u8 *data, unsigned int len); | ||
141 | u8 rtl_tid_to_ac(struct ieee80211_hw *hw, u8 tid); | ||
142 | extern struct attribute_group rtl_attribute_group; | ||
143 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/cam.c b/drivers/net/wireless/rtlwifi/cam.c new file mode 100644 index 000000000000..7295af0536b7 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/cam.c | |||
@@ -0,0 +1,355 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "cam.h" | ||
32 | |||
33 | void rtl_cam_reset_sec_info(struct ieee80211_hw *hw) | ||
34 | { | ||
35 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
36 | |||
37 | rtlpriv->sec.use_defaultkey = false; | ||
38 | rtlpriv->sec.pairwise_enc_algorithm = NO_ENCRYPTION; | ||
39 | rtlpriv->sec.group_enc_algorithm = NO_ENCRYPTION; | ||
40 | memset(rtlpriv->sec.key_buf, 0, KEY_BUF_SIZE * MAX_KEY_LEN); | ||
41 | memset(rtlpriv->sec.key_len, 0, KEY_BUF_SIZE); | ||
42 | rtlpriv->sec.pairwise_key = NULL; | ||
43 | } | ||
44 | |||
45 | static void rtl_cam_program_entry(struct ieee80211_hw *hw, u32 entry_no, | ||
46 | u8 *mac_addr, u8 *key_cont_128, u16 us_config) | ||
47 | { | ||
48 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
49 | |||
50 | u32 target_command; | ||
51 | u32 target_content = 0; | ||
52 | u8 entry_i; | ||
53 | |||
54 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
55 | ("key_cont_128:\n %x:%x:%x:%x:%x:%x\n", | ||
56 | key_cont_128[0], key_cont_128[1], | ||
57 | key_cont_128[2], key_cont_128[3], | ||
58 | key_cont_128[4], key_cont_128[5])); | ||
59 | |||
60 | for (entry_i = 0; entry_i < CAM_CONTENT_COUNT; entry_i++) { | ||
61 | target_command = entry_i + CAM_CONTENT_COUNT * entry_no; | ||
62 | target_command = target_command | BIT(31) | BIT(16); | ||
63 | |||
64 | if (entry_i == 0) { | ||
65 | target_content = (u32) (*(mac_addr + 0)) << 16 | | ||
66 | (u32) (*(mac_addr + 1)) << 24 | (u32) us_config; | ||
67 | |||
68 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], | ||
69 | target_content); | ||
70 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], | ||
71 | target_command); | ||
72 | |||
73 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
74 | ("WRITE %x: %x\n", | ||
75 | rtlpriv->cfg->maps[WCAMI], target_content)); | ||
76 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
77 | ("The Key ID is %d\n", entry_no)); | ||
78 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
79 | ("WRITE %x: %x\n", | ||
80 | rtlpriv->cfg->maps[RWCAM], target_command)); | ||
81 | |||
82 | } else if (entry_i == 1) { | ||
83 | |||
84 | target_content = (u32) (*(mac_addr + 5)) << 24 | | ||
85 | (u32) (*(mac_addr + 4)) << 16 | | ||
86 | (u32) (*(mac_addr + 3)) << 8 | | ||
87 | (u32) (*(mac_addr + 2)); | ||
88 | |||
89 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], | ||
90 | target_content); | ||
91 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], | ||
92 | target_command); | ||
93 | |||
94 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
95 | ("WRITE A4: %x\n", target_content)); | ||
96 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
97 | ("WRITE A0: %x\n", target_command)); | ||
98 | |||
99 | } else { | ||
100 | |||
101 | target_content = | ||
102 | (u32) (*(key_cont_128 + (entry_i * 4 - 8) + 3)) << | ||
103 | 24 | (u32) (*(key_cont_128 + (entry_i * 4 - 8) + 2)) | ||
104 | << 16 | | ||
105 | (u32) (*(key_cont_128 + (entry_i * 4 - 8) + 1)) << 8 | ||
106 | | (u32) (*(key_cont_128 + (entry_i * 4 - 8) + 0)); | ||
107 | |||
108 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], | ||
109 | target_content); | ||
110 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], | ||
111 | target_command); | ||
112 | udelay(100); | ||
113 | |||
114 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
115 | ("WRITE A4: %x\n", target_content)); | ||
116 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
117 | ("WRITE A0: %x\n", target_command)); | ||
118 | } | ||
119 | } | ||
120 | |||
121 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
122 | ("after set key, usconfig:%x\n", us_config)); | ||
123 | } | ||
124 | |||
125 | u8 rtl_cam_add_one_entry(struct ieee80211_hw *hw, u8 *mac_addr, | ||
126 | u32 ul_key_id, u32 ul_entry_idx, u32 ul_enc_alg, | ||
127 | u32 ul_default_key, u8 *key_content) | ||
128 | { | ||
129 | u32 us_config; | ||
130 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
131 | |||
132 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
133 | ("EntryNo:%x, ulKeyId=%x, ulEncAlg=%x, " | ||
134 | "ulUseDK=%x MacAddr" MAC_FMT "\n", | ||
135 | ul_entry_idx, ul_key_id, ul_enc_alg, | ||
136 | ul_default_key, MAC_ARG(mac_addr))); | ||
137 | |||
138 | if (ul_key_id == TOTAL_CAM_ENTRY) { | ||
139 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
140 | ("<=== ulKeyId exceed!\n")); | ||
141 | return 0; | ||
142 | } | ||
143 | |||
144 | if (ul_default_key == 1) { | ||
145 | us_config = CFG_VALID | ((u16) (ul_enc_alg) << 2); | ||
146 | } else { | ||
147 | us_config = CFG_VALID | ((ul_enc_alg) << 2) | ul_key_id; | ||
148 | } | ||
149 | |||
150 | rtl_cam_program_entry(hw, ul_entry_idx, mac_addr, | ||
151 | (u8 *) key_content, us_config); | ||
152 | |||
153 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("<===\n")); | ||
154 | |||
155 | return 1; | ||
156 | |||
157 | } | ||
158 | EXPORT_SYMBOL(rtl_cam_add_one_entry); | ||
159 | |||
160 | int rtl_cam_delete_one_entry(struct ieee80211_hw *hw, | ||
161 | u8 *mac_addr, u32 ul_key_id) | ||
162 | { | ||
163 | u32 ul_command; | ||
164 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
165 | |||
166 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("key_idx:%d\n", ul_key_id)); | ||
167 | |||
168 | ul_command = ul_key_id * CAM_CONTENT_COUNT; | ||
169 | ul_command = ul_command | BIT(31) | BIT(16); | ||
170 | |||
171 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], 0); | ||
172 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], ul_command); | ||
173 | |||
174 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
175 | ("rtl_cam_delete_one_entry(): WRITE A4: %x\n", 0)); | ||
176 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
177 | ("rtl_cam_delete_one_entry(): WRITE A0: %x\n", ul_command)); | ||
178 | |||
179 | return 0; | ||
180 | |||
181 | } | ||
182 | EXPORT_SYMBOL(rtl_cam_delete_one_entry); | ||
183 | |||
184 | void rtl_cam_reset_all_entry(struct ieee80211_hw *hw) | ||
185 | { | ||
186 | u32 ul_command; | ||
187 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
188 | |||
189 | ul_command = BIT(31) | BIT(30); | ||
190 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], ul_command); | ||
191 | } | ||
192 | EXPORT_SYMBOL(rtl_cam_reset_all_entry); | ||
193 | |||
194 | void rtl_cam_mark_invalid(struct ieee80211_hw *hw, u8 uc_index) | ||
195 | { | ||
196 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
197 | |||
198 | u32 ul_command; | ||
199 | u32 ul_content; | ||
200 | u32 ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
201 | |||
202 | switch (rtlpriv->sec.pairwise_enc_algorithm) { | ||
203 | case WEP40_ENCRYPTION: | ||
204 | ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_WEP40]; | ||
205 | break; | ||
206 | case WEP104_ENCRYPTION: | ||
207 | ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_WEP104]; | ||
208 | break; | ||
209 | case TKIP_ENCRYPTION: | ||
210 | ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_TKIP]; | ||
211 | break; | ||
212 | case AESCCMP_ENCRYPTION: | ||
213 | ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
214 | break; | ||
215 | default: | ||
216 | ul_enc_algo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
217 | } | ||
218 | |||
219 | ul_content = (uc_index & 3) | ((u16) (ul_enc_algo) << 2); | ||
220 | |||
221 | ul_content |= BIT(15); | ||
222 | ul_command = CAM_CONTENT_COUNT * uc_index; | ||
223 | ul_command = ul_command | BIT(31) | BIT(16); | ||
224 | |||
225 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], ul_content); | ||
226 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], ul_command); | ||
227 | |||
228 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
229 | ("rtl_cam_mark_invalid(): WRITE A4: %x\n", ul_content)); | ||
230 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
231 | ("rtl_cam_mark_invalid(): WRITE A0: %x\n", ul_command)); | ||
232 | } | ||
233 | EXPORT_SYMBOL(rtl_cam_mark_invalid); | ||
234 | |||
235 | void rtl_cam_empty_entry(struct ieee80211_hw *hw, u8 uc_index) | ||
236 | { | ||
237 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
238 | |||
239 | u32 ul_command; | ||
240 | u32 ul_content; | ||
241 | u32 ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
242 | u8 entry_i; | ||
243 | |||
244 | switch (rtlpriv->sec.pairwise_enc_algorithm) { | ||
245 | case WEP40_ENCRYPTION: | ||
246 | ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_WEP40]; | ||
247 | break; | ||
248 | case WEP104_ENCRYPTION: | ||
249 | ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_WEP104]; | ||
250 | break; | ||
251 | case TKIP_ENCRYPTION: | ||
252 | ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_TKIP]; | ||
253 | break; | ||
254 | case AESCCMP_ENCRYPTION: | ||
255 | ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
256 | break; | ||
257 | default: | ||
258 | ul_encalgo = rtlpriv->cfg->maps[SEC_CAM_AES]; | ||
259 | } | ||
260 | |||
261 | for (entry_i = 0; entry_i < CAM_CONTENT_COUNT; entry_i++) { | ||
262 | |||
263 | if (entry_i == 0) { | ||
264 | ul_content = | ||
265 | (uc_index & 0x03) | ((u16) (ul_encalgo) << 2); | ||
266 | ul_content |= BIT(15); | ||
267 | |||
268 | } else { | ||
269 | ul_content = 0; | ||
270 | } | ||
271 | |||
272 | ul_command = CAM_CONTENT_COUNT * uc_index + entry_i; | ||
273 | ul_command = ul_command | BIT(31) | BIT(16); | ||
274 | |||
275 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[WCAMI], ul_content); | ||
276 | rtl_write_dword(rtlpriv, rtlpriv->cfg->maps[RWCAM], ul_command); | ||
277 | |||
278 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
279 | ("rtl_cam_empty_entry(): WRITE A4: %x\n", | ||
280 | ul_content)); | ||
281 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
282 | ("rtl_cam_empty_entry(): WRITE A0: %x\n", | ||
283 | ul_command)); | ||
284 | } | ||
285 | |||
286 | } | ||
287 | EXPORT_SYMBOL(rtl_cam_empty_entry); | ||
288 | |||
289 | u8 rtl_cam_get_free_entry(struct ieee80211_hw *hw, u8 *sta_addr) | ||
290 | { | ||
291 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
292 | u32 bitmap = (rtlpriv->sec.hwsec_cam_bitmap) >> 4; | ||
293 | u8 entry_idx = 0; | ||
294 | u8 i, *addr; | ||
295 | |||
296 | if (NULL == sta_addr) { | ||
297 | RT_TRACE(rtlpriv, COMP_SEC, DBG_EMERG, | ||
298 | ("sta_addr is NULL.\n")); | ||
299 | return TOTAL_CAM_ENTRY; | ||
300 | } | ||
301 | /* Does STA already exist? */ | ||
302 | for (i = 4; i < TOTAL_CAM_ENTRY; i++) { | ||
303 | addr = rtlpriv->sec.hwsec_cam_sta_addr[i]; | ||
304 | if (memcmp(addr, sta_addr, ETH_ALEN) == 0) | ||
305 | return i; | ||
306 | } | ||
307 | /* Get a free CAM entry. */ | ||
308 | for (entry_idx = 4; entry_idx < TOTAL_CAM_ENTRY; entry_idx++) { | ||
309 | if ((bitmap & BIT(0)) == 0) { | ||
310 | RT_TRACE(rtlpriv, COMP_SEC, DBG_EMERG, | ||
311 | ("-----hwsec_cam_bitmap: 0x%x entry_idx=%d\n", | ||
312 | rtlpriv->sec.hwsec_cam_bitmap, entry_idx)); | ||
313 | rtlpriv->sec.hwsec_cam_bitmap |= BIT(0) << entry_idx; | ||
314 | memcpy(rtlpriv->sec.hwsec_cam_sta_addr[entry_idx], | ||
315 | sta_addr, ETH_ALEN); | ||
316 | return entry_idx; | ||
317 | } | ||
318 | bitmap = bitmap >> 1; | ||
319 | } | ||
320 | return TOTAL_CAM_ENTRY; | ||
321 | } | ||
322 | EXPORT_SYMBOL(rtl_cam_get_free_entry); | ||
323 | |||
324 | void rtl_cam_del_entry(struct ieee80211_hw *hw, u8 *sta_addr) | ||
325 | { | ||
326 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
327 | u32 bitmap; | ||
328 | u8 i, *addr; | ||
329 | |||
330 | if (NULL == sta_addr) { | ||
331 | RT_TRACE(rtlpriv, COMP_SEC, DBG_EMERG, | ||
332 | ("sta_addr is NULL.\n")); | ||
333 | } | ||
334 | |||
335 | if ((sta_addr[0]|sta_addr[1]|sta_addr[2]|sta_addr[3]|\ | ||
336 | sta_addr[4]|sta_addr[5]) == 0) { | ||
337 | RT_TRACE(rtlpriv, COMP_SEC, DBG_EMERG, | ||
338 | ("sta_addr is 00:00:00:00:00:00.\n")); | ||
339 | return; | ||
340 | } | ||
341 | /* Does STA already exist? */ | ||
342 | for (i = 4; i < TOTAL_CAM_ENTRY; i++) { | ||
343 | addr = rtlpriv->sec.hwsec_cam_sta_addr[i]; | ||
344 | bitmap = (rtlpriv->sec.hwsec_cam_bitmap) >> i; | ||
345 | if (((bitmap & BIT(0)) == BIT(0)) && | ||
346 | (memcmp(addr, sta_addr, ETH_ALEN) == 0)) { | ||
347 | /* Remove from HW Security CAM */ | ||
348 | memset(rtlpriv->sec.hwsec_cam_sta_addr[i], 0, ETH_ALEN); | ||
349 | rtlpriv->sec.hwsec_cam_bitmap &= ~(BIT(0) << i); | ||
350 | printk(KERN_INFO "&&&&&&&&&del entry %d\n", i); | ||
351 | } | ||
352 | } | ||
353 | return; | ||
354 | } | ||
355 | EXPORT_SYMBOL(rtl_cam_del_entry); | ||
diff --git a/drivers/net/wireless/rtlwifi/cam.h b/drivers/net/wireless/rtlwifi/cam.h new file mode 100644 index 000000000000..c62da4eefc75 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/cam.h | |||
@@ -0,0 +1,56 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_CAM_H_ | ||
31 | #define __RTL_CAM_H_ | ||
32 | |||
33 | #define CAM_CONTENT_COUNT 8 | ||
34 | |||
35 | #define CFG_DEFAULT_KEY BIT(5) | ||
36 | #define CFG_VALID BIT(15) | ||
37 | |||
38 | #define PAIRWISE_KEYIDX 0 | ||
39 | #define CAM_PAIRWISE_KEY_POSITION 4 | ||
40 | |||
41 | #define CAM_CONFIG_USEDK 1 | ||
42 | #define CAM_CONFIG_NO_USEDK 0 | ||
43 | |||
44 | extern void rtl_cam_reset_all_entry(struct ieee80211_hw *hw); | ||
45 | extern u8 rtl_cam_add_one_entry(struct ieee80211_hw *hw, u8 *mac_addr, | ||
46 | u32 ul_key_id, u32 ul_entry_idx, u32 ul_enc_alg, | ||
47 | u32 ul_default_key, u8 *key_content); | ||
48 | int rtl_cam_delete_one_entry(struct ieee80211_hw *hw, u8 *mac_addr, | ||
49 | u32 ul_key_id); | ||
50 | void rtl_cam_mark_invalid(struct ieee80211_hw *hw, u8 uc_index); | ||
51 | void rtl_cam_empty_entry(struct ieee80211_hw *hw, u8 uc_index); | ||
52 | void rtl_cam_reset_sec_info(struct ieee80211_hw *hw); | ||
53 | u8 rtl_cam_get_free_entry(struct ieee80211_hw *hw, u8 *sta_addr); | ||
54 | void rtl_cam_del_entry(struct ieee80211_hw *hw, u8 *sta_addr); | ||
55 | |||
56 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/core.c b/drivers/net/wireless/rtlwifi/core.c new file mode 100644 index 000000000000..d2ec2535aa3c --- /dev/null +++ b/drivers/net/wireless/rtlwifi/core.c | |||
@@ -0,0 +1,1143 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "core.h" | ||
32 | #include "cam.h" | ||
33 | #include "base.h" | ||
34 | #include "ps.h" | ||
35 | |||
36 | /*mutex for start & stop is must here. */ | ||
37 | static int rtl_op_start(struct ieee80211_hw *hw) | ||
38 | { | ||
39 | int err; | ||
40 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
41 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
42 | |||
43 | if (!is_hal_stop(rtlhal)) | ||
44 | return 0; | ||
45 | if (!test_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status)) | ||
46 | return 0; | ||
47 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
48 | err = rtlpriv->intf_ops->adapter_start(hw); | ||
49 | if (!err) | ||
50 | rtl_watch_dog_timer_callback((unsigned long)hw); | ||
51 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
52 | return err; | ||
53 | } | ||
54 | |||
55 | static void rtl_op_stop(struct ieee80211_hw *hw) | ||
56 | { | ||
57 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
58 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
59 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
60 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
61 | |||
62 | if (is_hal_stop(rtlhal)) | ||
63 | return; | ||
64 | |||
65 | if (unlikely(ppsc->rfpwr_state == ERFOFF)) { | ||
66 | rtl_ips_nic_on(hw); | ||
67 | mdelay(1); | ||
68 | } | ||
69 | |||
70 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
71 | |||
72 | mac->link_state = MAC80211_NOLINK; | ||
73 | memset(mac->bssid, 0, 6); | ||
74 | mac->vendor = PEER_UNKNOWN; | ||
75 | |||
76 | /*reset sec info */ | ||
77 | rtl_cam_reset_sec_info(hw); | ||
78 | |||
79 | rtl_deinit_deferred_work(hw); | ||
80 | rtlpriv->intf_ops->adapter_stop(hw); | ||
81 | |||
82 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
83 | } | ||
84 | |||
85 | static void rtl_op_tx(struct ieee80211_hw *hw, struct sk_buff *skb) | ||
86 | { | ||
87 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
88 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
89 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
90 | struct rtl_tcb_desc tcb_desc; | ||
91 | memset(&tcb_desc, 0, sizeof(struct rtl_tcb_desc)); | ||
92 | |||
93 | if (unlikely(is_hal_stop(rtlhal) || ppsc->rfpwr_state != ERFON)) | ||
94 | goto err_free; | ||
95 | |||
96 | if (!test_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status)) | ||
97 | goto err_free; | ||
98 | |||
99 | if (!rtlpriv->intf_ops->waitq_insert(hw, skb)) | ||
100 | rtlpriv->intf_ops->adapter_tx(hw, skb, &tcb_desc); | ||
101 | |||
102 | return; | ||
103 | |||
104 | err_free: | ||
105 | dev_kfree_skb_any(skb); | ||
106 | } | ||
107 | |||
108 | static int rtl_op_add_interface(struct ieee80211_hw *hw, | ||
109 | struct ieee80211_vif *vif) | ||
110 | { | ||
111 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
112 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
113 | int err = 0; | ||
114 | |||
115 | if (mac->vif) { | ||
116 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
117 | ("vif has been set!! mac->vif = 0x%p\n", mac->vif)); | ||
118 | return -EOPNOTSUPP; | ||
119 | } | ||
120 | |||
121 | rtl_ips_nic_on(hw); | ||
122 | |||
123 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
124 | switch (vif->type) { | ||
125 | case NL80211_IFTYPE_STATION: | ||
126 | if (mac->beacon_enabled == 1) { | ||
127 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
128 | ("NL80211_IFTYPE_STATION\n")); | ||
129 | mac->beacon_enabled = 0; | ||
130 | rtlpriv->cfg->ops->update_interrupt_mask(hw, 0, | ||
131 | rtlpriv->cfg->maps | ||
132 | [RTL_IBSS_INT_MASKS]); | ||
133 | } | ||
134 | break; | ||
135 | case NL80211_IFTYPE_ADHOC: | ||
136 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
137 | ("NL80211_IFTYPE_ADHOC\n")); | ||
138 | |||
139 | mac->link_state = MAC80211_LINKED; | ||
140 | rtlpriv->cfg->ops->set_bcn_reg(hw); | ||
141 | if (rtlpriv->rtlhal.current_bandtype == BAND_ON_2_4G) | ||
142 | mac->basic_rates = 0xfff; | ||
143 | else | ||
144 | mac->basic_rates = 0xff0; | ||
145 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_BASIC_RATE, | ||
146 | (u8 *) (&mac->basic_rates)); | ||
147 | |||
148 | break; | ||
149 | case NL80211_IFTYPE_AP: | ||
150 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
151 | ("NL80211_IFTYPE_AP\n")); | ||
152 | |||
153 | mac->link_state = MAC80211_LINKED; | ||
154 | rtlpriv->cfg->ops->set_bcn_reg(hw); | ||
155 | if (rtlpriv->rtlhal.current_bandtype == BAND_ON_2_4G) | ||
156 | mac->basic_rates = 0xfff; | ||
157 | else | ||
158 | mac->basic_rates = 0xff0; | ||
159 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_BASIC_RATE, | ||
160 | (u8 *) (&mac->basic_rates)); | ||
161 | break; | ||
162 | default: | ||
163 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
164 | ("operation mode %d is not support!\n", vif->type)); | ||
165 | err = -EOPNOTSUPP; | ||
166 | goto out; | ||
167 | } | ||
168 | |||
169 | mac->vif = vif; | ||
170 | mac->opmode = vif->type; | ||
171 | rtlpriv->cfg->ops->set_network_type(hw, vif->type); | ||
172 | memcpy(mac->mac_addr, vif->addr, ETH_ALEN); | ||
173 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_ETHER_ADDR, mac->mac_addr); | ||
174 | |||
175 | out: | ||
176 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
177 | return err; | ||
178 | } | ||
179 | |||
180 | static void rtl_op_remove_interface(struct ieee80211_hw *hw, | ||
181 | struct ieee80211_vif *vif) | ||
182 | { | ||
183 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
184 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
185 | |||
186 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
187 | |||
188 | /* Free beacon resources */ | ||
189 | if ((mac->opmode == NL80211_IFTYPE_AP) || | ||
190 | (mac->opmode == NL80211_IFTYPE_ADHOC) || | ||
191 | (mac->opmode == NL80211_IFTYPE_MESH_POINT)) { | ||
192 | if (mac->beacon_enabled == 1) { | ||
193 | mac->beacon_enabled = 0; | ||
194 | rtlpriv->cfg->ops->update_interrupt_mask(hw, 0, | ||
195 | rtlpriv->cfg->maps | ||
196 | [RTL_IBSS_INT_MASKS]); | ||
197 | } | ||
198 | } | ||
199 | |||
200 | /* | ||
201 | *Note: We assume NL80211_IFTYPE_UNSPECIFIED as | ||
202 | *NO LINK for our hardware. | ||
203 | */ | ||
204 | mac->vif = NULL; | ||
205 | mac->link_state = MAC80211_NOLINK; | ||
206 | memset(mac->bssid, 0, 6); | ||
207 | mac->vendor = PEER_UNKNOWN; | ||
208 | mac->opmode = NL80211_IFTYPE_UNSPECIFIED; | ||
209 | rtlpriv->cfg->ops->set_network_type(hw, mac->opmode); | ||
210 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
211 | } | ||
212 | |||
213 | static int rtl_op_config(struct ieee80211_hw *hw, u32 changed) | ||
214 | { | ||
215 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
216 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
217 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
218 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
219 | struct ieee80211_conf *conf = &hw->conf; | ||
220 | |||
221 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
222 | if (changed & IEEE80211_CONF_CHANGE_LISTEN_INTERVAL) { /*BIT(2)*/ | ||
223 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
224 | ("IEEE80211_CONF_CHANGE_LISTEN_INTERVAL\n")); | ||
225 | } | ||
226 | |||
227 | /*For IPS */ | ||
228 | if (changed & IEEE80211_CONF_CHANGE_IDLE) { | ||
229 | if (hw->conf.flags & IEEE80211_CONF_IDLE) | ||
230 | rtl_ips_nic_off(hw); | ||
231 | else | ||
232 | rtl_ips_nic_on(hw); | ||
233 | } else { | ||
234 | /* | ||
235 | *although rfoff may not cause by ips, but we will | ||
236 | *check the reason in set_rf_power_state function | ||
237 | */ | ||
238 | if (unlikely(ppsc->rfpwr_state == ERFOFF)) | ||
239 | rtl_ips_nic_on(hw); | ||
240 | } | ||
241 | |||
242 | /*For LPS */ | ||
243 | if (changed & IEEE80211_CONF_CHANGE_PS) { | ||
244 | cancel_delayed_work(&rtlpriv->works.ps_work); | ||
245 | cancel_delayed_work(&rtlpriv->works.ps_rfon_wq); | ||
246 | if (conf->flags & IEEE80211_CONF_PS) { | ||
247 | rtlpriv->psc.sw_ps_enabled = true; | ||
248 | /* sleep here is must, or we may recv the beacon and | ||
249 | * cause mac80211 into wrong ps state, this will cause | ||
250 | * power save nullfunc send fail, and further cause | ||
251 | * pkt loss, So sleep must quickly but not immediatly | ||
252 | * because that will cause nullfunc send by mac80211 | ||
253 | * fail, and cause pkt loss, we have tested that 5mA | ||
254 | * is worked very well */ | ||
255 | if (!rtlpriv->psc.multi_buffered) | ||
256 | queue_delayed_work(rtlpriv->works.rtl_wq, | ||
257 | &rtlpriv->works.ps_work, | ||
258 | MSECS(5)); | ||
259 | } else { | ||
260 | rtl_swlps_rf_awake(hw); | ||
261 | rtlpriv->psc.sw_ps_enabled = false; | ||
262 | } | ||
263 | } | ||
264 | |||
265 | if (changed & IEEE80211_CONF_CHANGE_RETRY_LIMITS) { | ||
266 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
267 | ("IEEE80211_CONF_CHANGE_RETRY_LIMITS %x\n", | ||
268 | hw->conf.long_frame_max_tx_count)); | ||
269 | mac->retry_long = hw->conf.long_frame_max_tx_count; | ||
270 | mac->retry_short = hw->conf.long_frame_max_tx_count; | ||
271 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RETRY_LIMIT, | ||
272 | (u8 *) (&hw->conf. | ||
273 | long_frame_max_tx_count)); | ||
274 | } | ||
275 | |||
276 | if (changed & IEEE80211_CONF_CHANGE_CHANNEL) { | ||
277 | struct ieee80211_channel *channel = hw->conf.channel; | ||
278 | u8 wide_chan = (u8) channel->hw_value; | ||
279 | |||
280 | /* | ||
281 | *because we should back channel to | ||
282 | *current_network.chan in in scanning, | ||
283 | *So if set_chan == current_network.chan | ||
284 | *we should set it. | ||
285 | *because mac80211 tell us wrong bw40 | ||
286 | *info for cisco1253 bw20, so we modify | ||
287 | *it here based on UPPER & LOWER | ||
288 | */ | ||
289 | switch (hw->conf.channel_type) { | ||
290 | case NL80211_CHAN_HT20: | ||
291 | case NL80211_CHAN_NO_HT: | ||
292 | /* SC */ | ||
293 | mac->cur_40_prime_sc = | ||
294 | PRIME_CHNL_OFFSET_DONT_CARE; | ||
295 | rtlphy->current_chan_bw = HT_CHANNEL_WIDTH_20; | ||
296 | mac->bw_40 = false; | ||
297 | break; | ||
298 | case NL80211_CHAN_HT40MINUS: | ||
299 | /* SC */ | ||
300 | mac->cur_40_prime_sc = PRIME_CHNL_OFFSET_UPPER; | ||
301 | rtlphy->current_chan_bw = | ||
302 | HT_CHANNEL_WIDTH_20_40; | ||
303 | mac->bw_40 = true; | ||
304 | |||
305 | /*wide channel */ | ||
306 | wide_chan -= 2; | ||
307 | |||
308 | break; | ||
309 | case NL80211_CHAN_HT40PLUS: | ||
310 | /* SC */ | ||
311 | mac->cur_40_prime_sc = PRIME_CHNL_OFFSET_LOWER; | ||
312 | rtlphy->current_chan_bw = | ||
313 | HT_CHANNEL_WIDTH_20_40; | ||
314 | mac->bw_40 = true; | ||
315 | |||
316 | /*wide channel */ | ||
317 | wide_chan += 2; | ||
318 | |||
319 | break; | ||
320 | default: | ||
321 | mac->bw_40 = false; | ||
322 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
323 | ("switch case not processed\n")); | ||
324 | break; | ||
325 | } | ||
326 | |||
327 | if (wide_chan <= 0) | ||
328 | wide_chan = 1; | ||
329 | |||
330 | /* In scanning, before we go offchannel we may send a ps=1 null | ||
331 | * to AP, and then we may send a ps = 0 null to AP quickly, but | ||
332 | * first null may have caused AP to put lots of packet to hw tx | ||
333 | * buffer. These packets must be tx'd before we go off channel | ||
334 | * so we must delay more time to let AP flush these packets | ||
335 | * before going offchannel, or dis-association or delete BA will | ||
336 | * happen by AP | ||
337 | */ | ||
338 | if (rtlpriv->mac80211.offchan_deley) { | ||
339 | rtlpriv->mac80211.offchan_deley = false; | ||
340 | mdelay(50); | ||
341 | } | ||
342 | rtlphy->current_channel = wide_chan; | ||
343 | |||
344 | rtlpriv->cfg->ops->switch_channel(hw); | ||
345 | rtlpriv->cfg->ops->set_channel_access(hw); | ||
346 | rtlpriv->cfg->ops->set_bw_mode(hw, | ||
347 | hw->conf.channel_type); | ||
348 | } | ||
349 | |||
350 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
351 | |||
352 | return 0; | ||
353 | } | ||
354 | |||
355 | static void rtl_op_configure_filter(struct ieee80211_hw *hw, | ||
356 | unsigned int changed_flags, | ||
357 | unsigned int *new_flags, u64 multicast) | ||
358 | { | ||
359 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
360 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
361 | |||
362 | *new_flags &= RTL_SUPPORTED_FILTERS; | ||
363 | if (!changed_flags) | ||
364 | return; | ||
365 | |||
366 | /*TODO: we disable broadcase now, so enable here */ | ||
367 | if (changed_flags & FIF_ALLMULTI) { | ||
368 | if (*new_flags & FIF_ALLMULTI) { | ||
369 | mac->rx_conf |= rtlpriv->cfg->maps[MAC_RCR_AM] | | ||
370 | rtlpriv->cfg->maps[MAC_RCR_AB]; | ||
371 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
372 | ("Enable receive multicast frame.\n")); | ||
373 | } else { | ||
374 | mac->rx_conf &= ~(rtlpriv->cfg->maps[MAC_RCR_AM] | | ||
375 | rtlpriv->cfg->maps[MAC_RCR_AB]); | ||
376 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
377 | ("Disable receive multicast frame.\n")); | ||
378 | } | ||
379 | } | ||
380 | |||
381 | if (changed_flags & FIF_FCSFAIL) { | ||
382 | if (*new_flags & FIF_FCSFAIL) { | ||
383 | mac->rx_conf |= rtlpriv->cfg->maps[MAC_RCR_ACRC32]; | ||
384 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
385 | ("Enable receive FCS error frame.\n")); | ||
386 | } else { | ||
387 | mac->rx_conf &= ~rtlpriv->cfg->maps[MAC_RCR_ACRC32]; | ||
388 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
389 | ("Disable receive FCS error frame.\n")); | ||
390 | } | ||
391 | } | ||
392 | |||
393 | /* if ssid not set to hw don't check bssid | ||
394 | * here just used for linked scanning, & linked | ||
395 | * and nolink check bssid is set in set network_type */ | ||
396 | if ((changed_flags & FIF_BCN_PRBRESP_PROMISC) && | ||
397 | (mac->link_state >= MAC80211_LINKED)) { | ||
398 | if (mac->opmode != NL80211_IFTYPE_AP) { | ||
399 | if (*new_flags & FIF_BCN_PRBRESP_PROMISC) { | ||
400 | rtlpriv->cfg->ops->set_chk_bssid(hw, false); | ||
401 | } else { | ||
402 | rtlpriv->cfg->ops->set_chk_bssid(hw, true); | ||
403 | } | ||
404 | } | ||
405 | } | ||
406 | |||
407 | if (changed_flags & FIF_CONTROL) { | ||
408 | if (*new_flags & FIF_CONTROL) { | ||
409 | mac->rx_conf |= rtlpriv->cfg->maps[MAC_RCR_ACF]; | ||
410 | |||
411 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
412 | ("Enable receive control frame.\n")); | ||
413 | } else { | ||
414 | mac->rx_conf &= ~rtlpriv->cfg->maps[MAC_RCR_ACF]; | ||
415 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
416 | ("Disable receive control frame.\n")); | ||
417 | } | ||
418 | } | ||
419 | |||
420 | if (changed_flags & FIF_OTHER_BSS) { | ||
421 | if (*new_flags & FIF_OTHER_BSS) { | ||
422 | mac->rx_conf |= rtlpriv->cfg->maps[MAC_RCR_AAP]; | ||
423 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
424 | ("Enable receive other BSS's frame.\n")); | ||
425 | } else { | ||
426 | mac->rx_conf &= ~rtlpriv->cfg->maps[MAC_RCR_AAP]; | ||
427 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
428 | ("Disable receive other BSS's frame.\n")); | ||
429 | } | ||
430 | } | ||
431 | } | ||
432 | static int rtl_op_sta_add(struct ieee80211_hw *hw, | ||
433 | struct ieee80211_vif *vif, | ||
434 | struct ieee80211_sta *sta) | ||
435 | { | ||
436 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
437 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
438 | struct rtl_sta_info *sta_entry; | ||
439 | |||
440 | if (sta) { | ||
441 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
442 | if (rtlhal->current_bandtype == BAND_ON_2_4G) { | ||
443 | sta_entry->wireless_mode = WIRELESS_MODE_G; | ||
444 | if (sta->supp_rates[0] <= 0xf) | ||
445 | sta_entry->wireless_mode = WIRELESS_MODE_B; | ||
446 | if (sta->ht_cap.ht_supported == true) | ||
447 | sta_entry->wireless_mode = WIRELESS_MODE_N_24G; | ||
448 | } else if (rtlhal->current_bandtype == BAND_ON_5G) { | ||
449 | sta_entry->wireless_mode = WIRELESS_MODE_A; | ||
450 | if (sta->ht_cap.ht_supported == true) | ||
451 | sta_entry->wireless_mode = WIRELESS_MODE_N_24G; | ||
452 | } | ||
453 | |||
454 | /* I found some times mac80211 give wrong supp_rates for adhoc*/ | ||
455 | if (rtlpriv->mac80211.opmode == NL80211_IFTYPE_ADHOC) | ||
456 | sta_entry->wireless_mode = WIRELESS_MODE_G; | ||
457 | |||
458 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
459 | ("Add sta addr is "MAC_FMT"\n", MAC_ARG(sta->addr))); | ||
460 | rtlpriv->cfg->ops->update_rate_tbl(hw, sta, 0); | ||
461 | } | ||
462 | return 0; | ||
463 | } | ||
464 | static int rtl_op_sta_remove(struct ieee80211_hw *hw, | ||
465 | struct ieee80211_vif *vif, | ||
466 | struct ieee80211_sta *sta) | ||
467 | { | ||
468 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
469 | struct rtl_sta_info *sta_entry; | ||
470 | if (sta) { | ||
471 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
472 | ("Remove sta addr is "MAC_FMT"\n", MAC_ARG(sta->addr))); | ||
473 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
474 | sta_entry->wireless_mode = 0; | ||
475 | sta_entry->ratr_index = 0; | ||
476 | } | ||
477 | return 0; | ||
478 | } | ||
479 | static int _rtl_get_hal_qnum(u16 queue) | ||
480 | { | ||
481 | int qnum; | ||
482 | |||
483 | switch (queue) { | ||
484 | case 0: | ||
485 | qnum = AC3_VO; | ||
486 | break; | ||
487 | case 1: | ||
488 | qnum = AC2_VI; | ||
489 | break; | ||
490 | case 2: | ||
491 | qnum = AC0_BE; | ||
492 | break; | ||
493 | case 3: | ||
494 | qnum = AC1_BK; | ||
495 | break; | ||
496 | default: | ||
497 | qnum = AC0_BE; | ||
498 | break; | ||
499 | } | ||
500 | return qnum; | ||
501 | } | ||
502 | |||
503 | /* | ||
504 | *for mac80211 VO=0, VI=1, BE=2, BK=3 | ||
505 | *for rtl819x BE=0, BK=1, VI=2, VO=3 | ||
506 | */ | ||
507 | static int rtl_op_conf_tx(struct ieee80211_hw *hw, u16 queue, | ||
508 | const struct ieee80211_tx_queue_params *param) | ||
509 | { | ||
510 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
511 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
512 | int aci; | ||
513 | |||
514 | if (queue >= AC_MAX) { | ||
515 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
516 | ("queue number %d is incorrect!\n", queue)); | ||
517 | return -EINVAL; | ||
518 | } | ||
519 | |||
520 | aci = _rtl_get_hal_qnum(queue); | ||
521 | mac->ac[aci].aifs = param->aifs; | ||
522 | mac->ac[aci].cw_min = cpu_to_le16(param->cw_min); | ||
523 | mac->ac[aci].cw_max = cpu_to_le16(param->cw_max); | ||
524 | mac->ac[aci].tx_op = cpu_to_le16(param->txop); | ||
525 | memcpy(&mac->edca_param[aci], param, sizeof(*param)); | ||
526 | rtlpriv->cfg->ops->set_qos(hw, aci); | ||
527 | return 0; | ||
528 | } | ||
529 | |||
530 | static void rtl_op_bss_info_changed(struct ieee80211_hw *hw, | ||
531 | struct ieee80211_vif *vif, | ||
532 | struct ieee80211_bss_conf *bss_conf, u32 changed) | ||
533 | { | ||
534 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
535 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
536 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
537 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
538 | struct ieee80211_sta *sta = NULL; | ||
539 | |||
540 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
541 | if ((vif->type == NL80211_IFTYPE_ADHOC) || | ||
542 | (vif->type == NL80211_IFTYPE_AP) || | ||
543 | (vif->type == NL80211_IFTYPE_MESH_POINT)) { | ||
544 | if ((changed & BSS_CHANGED_BEACON) || | ||
545 | (changed & BSS_CHANGED_BEACON_ENABLED && | ||
546 | bss_conf->enable_beacon)) { | ||
547 | if (mac->beacon_enabled == 0) { | ||
548 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
549 | ("BSS_CHANGED_BEACON_ENABLED\n")); | ||
550 | |||
551 | /*start hw beacon interrupt. */ | ||
552 | /*rtlpriv->cfg->ops->set_bcn_reg(hw); */ | ||
553 | mac->beacon_enabled = 1; | ||
554 | rtlpriv->cfg->ops->update_interrupt_mask(hw, | ||
555 | rtlpriv->cfg->maps | ||
556 | [RTL_IBSS_INT_MASKS], | ||
557 | 0); | ||
558 | |||
559 | if (rtlpriv->cfg->ops->linked_set_reg) | ||
560 | rtlpriv->cfg->ops->linked_set_reg(hw); | ||
561 | } | ||
562 | } | ||
563 | if ((changed & BSS_CHANGED_BEACON_ENABLED && | ||
564 | !bss_conf->enable_beacon)) { | ||
565 | if (mac->beacon_enabled == 1) { | ||
566 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
567 | ("ADHOC DISABLE BEACON\n")); | ||
568 | |||
569 | mac->beacon_enabled = 0; | ||
570 | rtlpriv->cfg->ops->update_interrupt_mask(hw, 0, | ||
571 | rtlpriv->cfg->maps | ||
572 | [RTL_IBSS_INT_MASKS]); | ||
573 | } | ||
574 | } | ||
575 | if (changed & BSS_CHANGED_BEACON_INT) { | ||
576 | RT_TRACE(rtlpriv, COMP_BEACON, DBG_TRACE, | ||
577 | ("BSS_CHANGED_BEACON_INT\n")); | ||
578 | mac->beacon_interval = bss_conf->beacon_int; | ||
579 | rtlpriv->cfg->ops->set_bcn_intv(hw); | ||
580 | } | ||
581 | } | ||
582 | |||
583 | /*TODO: reference to enum ieee80211_bss_change */ | ||
584 | if (changed & BSS_CHANGED_ASSOC) { | ||
585 | if (bss_conf->assoc) { | ||
586 | /* we should reset all sec info & cam | ||
587 | * before set cam after linked, we should not | ||
588 | * reset in disassoc, that will cause tkip->wep | ||
589 | * fail because some flag will be wrong */ | ||
590 | /* reset sec info */ | ||
591 | rtl_cam_reset_sec_info(hw); | ||
592 | /* reset cam to fix wep fail issue | ||
593 | * when change from wpa to wep */ | ||
594 | rtl_cam_reset_all_entry(hw); | ||
595 | |||
596 | mac->link_state = MAC80211_LINKED; | ||
597 | mac->cnt_after_linked = 0; | ||
598 | mac->assoc_id = bss_conf->aid; | ||
599 | memcpy(mac->bssid, bss_conf->bssid, 6); | ||
600 | |||
601 | if (rtlpriv->cfg->ops->linked_set_reg) | ||
602 | rtlpriv->cfg->ops->linked_set_reg(hw); | ||
603 | if (mac->opmode == NL80211_IFTYPE_STATION && sta) | ||
604 | rtlpriv->cfg->ops->update_rate_tbl(hw, sta, 0); | ||
605 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
606 | ("BSS_CHANGED_ASSOC\n")); | ||
607 | } else { | ||
608 | if (mac->link_state == MAC80211_LINKED) | ||
609 | rtl_lps_leave(hw); | ||
610 | |||
611 | mac->link_state = MAC80211_NOLINK; | ||
612 | memset(mac->bssid, 0, 6); | ||
613 | mac->vendor = PEER_UNKNOWN; | ||
614 | |||
615 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
616 | ("BSS_CHANGED_UN_ASSOC\n")); | ||
617 | } | ||
618 | } | ||
619 | |||
620 | if (changed & BSS_CHANGED_ERP_CTS_PROT) { | ||
621 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
622 | ("BSS_CHANGED_ERP_CTS_PROT\n")); | ||
623 | mac->use_cts_protect = bss_conf->use_cts_prot; | ||
624 | } | ||
625 | |||
626 | if (changed & BSS_CHANGED_ERP_PREAMBLE) { | ||
627 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, | ||
628 | ("BSS_CHANGED_ERP_PREAMBLE use short preamble:%x\n", | ||
629 | bss_conf->use_short_preamble)); | ||
630 | |||
631 | mac->short_preamble = bss_conf->use_short_preamble; | ||
632 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_ACK_PREAMBLE, | ||
633 | (u8 *) (&mac->short_preamble)); | ||
634 | } | ||
635 | |||
636 | if (changed & BSS_CHANGED_ERP_SLOT) { | ||
637 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
638 | ("BSS_CHANGED_ERP_SLOT\n")); | ||
639 | |||
640 | if (bss_conf->use_short_slot) | ||
641 | mac->slot_time = RTL_SLOT_TIME_9; | ||
642 | else | ||
643 | mac->slot_time = RTL_SLOT_TIME_20; | ||
644 | |||
645 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SLOT_TIME, | ||
646 | (u8 *) (&mac->slot_time)); | ||
647 | } | ||
648 | |||
649 | if (changed & BSS_CHANGED_HT) { | ||
650 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
651 | ("BSS_CHANGED_HT\n")); | ||
652 | rcu_read_lock(); | ||
653 | sta = get_sta(hw, vif, (u8 *)bss_conf->bssid); | ||
654 | if (sta) { | ||
655 | if (sta->ht_cap.ampdu_density > | ||
656 | mac->current_ampdu_density) | ||
657 | mac->current_ampdu_density = | ||
658 | sta->ht_cap.ampdu_density; | ||
659 | if (sta->ht_cap.ampdu_factor < | ||
660 | mac->current_ampdu_factor) | ||
661 | mac->current_ampdu_factor = | ||
662 | sta->ht_cap.ampdu_factor; | ||
663 | } | ||
664 | rcu_read_unlock(); | ||
665 | |||
666 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SHORTGI_DENSITY, | ||
667 | (u8 *) (&mac->max_mss_density)); | ||
668 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_AMPDU_FACTOR, | ||
669 | &mac->current_ampdu_factor); | ||
670 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_AMPDU_MIN_SPACE, | ||
671 | &mac->current_ampdu_density); | ||
672 | } | ||
673 | |||
674 | if (changed & BSS_CHANGED_BSSID) { | ||
675 | u32 basic_rates; | ||
676 | |||
677 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_BSSID, | ||
678 | (u8 *) bss_conf->bssid); | ||
679 | |||
680 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_DMESG, | ||
681 | (MAC_FMT "\n", MAC_ARG(bss_conf->bssid))); | ||
682 | |||
683 | mac->vendor = PEER_UNKNOWN; | ||
684 | memcpy(mac->bssid, bss_conf->bssid, 6); | ||
685 | rtlpriv->cfg->ops->set_network_type(hw, vif->type); | ||
686 | |||
687 | rcu_read_lock(); | ||
688 | sta = get_sta(hw, vif, (u8 *)bss_conf->bssid); | ||
689 | if (!sta) { | ||
690 | rcu_read_unlock(); | ||
691 | goto out; | ||
692 | } | ||
693 | |||
694 | if (rtlhal->current_bandtype == BAND_ON_5G) { | ||
695 | mac->mode = WIRELESS_MODE_A; | ||
696 | } else { | ||
697 | if (sta->supp_rates[0] <= 0xf) | ||
698 | mac->mode = WIRELESS_MODE_B; | ||
699 | else | ||
700 | mac->mode = WIRELESS_MODE_G; | ||
701 | } | ||
702 | |||
703 | if (sta->ht_cap.ht_supported) { | ||
704 | if (rtlhal->current_bandtype == BAND_ON_2_4G) | ||
705 | mac->mode = WIRELESS_MODE_N_24G; | ||
706 | else | ||
707 | mac->mode = WIRELESS_MODE_N_5G; | ||
708 | } | ||
709 | |||
710 | /* just station need it, because ibss & ap mode will | ||
711 | * set in sta_add, and will be NULL here */ | ||
712 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
713 | struct rtl_sta_info *sta_entry; | ||
714 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
715 | sta_entry->wireless_mode = mac->mode; | ||
716 | } | ||
717 | |||
718 | if (sta->ht_cap.ht_supported) { | ||
719 | mac->ht_enable = true; | ||
720 | |||
721 | /* | ||
722 | * for cisco 1252 bw20 it's wrong | ||
723 | * if (ht_cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40) { | ||
724 | * mac->bw_40 = true; | ||
725 | * } | ||
726 | * */ | ||
727 | } | ||
728 | |||
729 | if (changed & BSS_CHANGED_BASIC_RATES) { | ||
730 | /* for 5G must << RATE_6M_INDEX=4, | ||
731 | * because 5G have no cck rate*/ | ||
732 | if (rtlhal->current_bandtype == BAND_ON_5G) | ||
733 | basic_rates = sta->supp_rates[1] << 4; | ||
734 | else | ||
735 | basic_rates = sta->supp_rates[0]; | ||
736 | |||
737 | mac->basic_rates = basic_rates; | ||
738 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_BASIC_RATE, | ||
739 | (u8 *) (&basic_rates)); | ||
740 | } | ||
741 | rcu_read_unlock(); | ||
742 | } | ||
743 | |||
744 | /* | ||
745 | * For FW LPS: | ||
746 | * To tell firmware we have connected | ||
747 | * to an AP. For 92SE/CE power save v2. | ||
748 | */ | ||
749 | if (changed & BSS_CHANGED_ASSOC) { | ||
750 | if (bss_conf->assoc) { | ||
751 | if (ppsc->fwctrl_lps) { | ||
752 | u8 mstatus = RT_MEDIA_CONNECT; | ||
753 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
754 | HW_VAR_H2C_FW_JOINBSSRPT, | ||
755 | (u8 *) (&mstatus)); | ||
756 | ppsc->report_linked = true; | ||
757 | } | ||
758 | } else { | ||
759 | if (ppsc->fwctrl_lps) { | ||
760 | u8 mstatus = RT_MEDIA_DISCONNECT; | ||
761 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
762 | HW_VAR_H2C_FW_JOINBSSRPT, | ||
763 | (u8 *)(&mstatus)); | ||
764 | ppsc->report_linked = false; | ||
765 | } | ||
766 | } | ||
767 | } | ||
768 | |||
769 | out: | ||
770 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
771 | } | ||
772 | |||
773 | static u64 rtl_op_get_tsf(struct ieee80211_hw *hw) | ||
774 | { | ||
775 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
776 | u64 tsf; | ||
777 | |||
778 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_CORRECT_TSF, (u8 *) (&tsf)); | ||
779 | return tsf; | ||
780 | } | ||
781 | |||
782 | static void rtl_op_set_tsf(struct ieee80211_hw *hw, u64 tsf) | ||
783 | { | ||
784 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
785 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
786 | u8 bibss = (mac->opmode == NL80211_IFTYPE_ADHOC) ? 1 : 0; | ||
787 | |||
788 | mac->tsf = tsf; | ||
789 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_CORRECT_TSF, (u8 *) (&bibss)); | ||
790 | } | ||
791 | |||
792 | static void rtl_op_reset_tsf(struct ieee80211_hw *hw) | ||
793 | { | ||
794 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
795 | u8 tmp = 0; | ||
796 | |||
797 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_DUAL_TSF_RST, (u8 *) (&tmp)); | ||
798 | } | ||
799 | |||
800 | static void rtl_op_sta_notify(struct ieee80211_hw *hw, | ||
801 | struct ieee80211_vif *vif, | ||
802 | enum sta_notify_cmd cmd, | ||
803 | struct ieee80211_sta *sta) | ||
804 | { | ||
805 | switch (cmd) { | ||
806 | case STA_NOTIFY_SLEEP: | ||
807 | break; | ||
808 | case STA_NOTIFY_AWAKE: | ||
809 | break; | ||
810 | default: | ||
811 | break; | ||
812 | } | ||
813 | } | ||
814 | |||
815 | static int rtl_op_ampdu_action(struct ieee80211_hw *hw, | ||
816 | struct ieee80211_vif *vif, | ||
817 | enum ieee80211_ampdu_mlme_action action, | ||
818 | struct ieee80211_sta *sta, u16 tid, u16 *ssn, | ||
819 | u8 buf_size) | ||
820 | { | ||
821 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
822 | |||
823 | switch (action) { | ||
824 | case IEEE80211_AMPDU_TX_START: | ||
825 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
826 | ("IEEE80211_AMPDU_TX_START: TID:%d\n", tid)); | ||
827 | return rtl_tx_agg_start(hw, sta, tid, ssn); | ||
828 | break; | ||
829 | case IEEE80211_AMPDU_TX_STOP: | ||
830 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
831 | ("IEEE80211_AMPDU_TX_STOP: TID:%d\n", tid)); | ||
832 | return rtl_tx_agg_stop(hw, sta, tid); | ||
833 | break; | ||
834 | case IEEE80211_AMPDU_TX_OPERATIONAL: | ||
835 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
836 | ("IEEE80211_AMPDU_TX_OPERATIONAL:TID:%d\n", tid)); | ||
837 | rtl_tx_agg_oper(hw, sta, tid); | ||
838 | break; | ||
839 | case IEEE80211_AMPDU_RX_START: | ||
840 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
841 | ("IEEE80211_AMPDU_RX_START:TID:%d\n", tid)); | ||
842 | break; | ||
843 | case IEEE80211_AMPDU_RX_STOP: | ||
844 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_TRACE, | ||
845 | ("IEEE80211_AMPDU_RX_STOP:TID:%d\n", tid)); | ||
846 | break; | ||
847 | default: | ||
848 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
849 | ("IEEE80211_AMPDU_ERR!!!!:\n")); | ||
850 | return -EOPNOTSUPP; | ||
851 | } | ||
852 | return 0; | ||
853 | } | ||
854 | |||
855 | static void rtl_op_sw_scan_start(struct ieee80211_hw *hw) | ||
856 | { | ||
857 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
858 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
859 | |||
860 | mac->act_scanning = true; | ||
861 | |||
862 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("\n")); | ||
863 | |||
864 | if (mac->link_state == MAC80211_LINKED) { | ||
865 | rtl_lps_leave(hw); | ||
866 | mac->link_state = MAC80211_LINKED_SCANNING; | ||
867 | } else { | ||
868 | rtl_ips_nic_on(hw); | ||
869 | } | ||
870 | |||
871 | /* Dual mac */ | ||
872 | rtlpriv->rtlhal.load_imrandiqk_setting_for2g = false; | ||
873 | |||
874 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_SITE_SURVEY); | ||
875 | rtlpriv->cfg->ops->scan_operation_backup(hw, SCAN_OPT_BACKUP); | ||
876 | } | ||
877 | |||
878 | static void rtl_op_sw_scan_complete(struct ieee80211_hw *hw) | ||
879 | { | ||
880 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
881 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
882 | |||
883 | RT_TRACE(rtlpriv, COMP_MAC80211, DBG_LOUD, ("\n")); | ||
884 | mac->act_scanning = false; | ||
885 | /* Dual mac */ | ||
886 | rtlpriv->rtlhal.load_imrandiqk_setting_for2g = false; | ||
887 | |||
888 | if (mac->link_state == MAC80211_LINKED_SCANNING) { | ||
889 | mac->link_state = MAC80211_LINKED; | ||
890 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
891 | /* fix fwlps issue */ | ||
892 | rtlpriv->cfg->ops->set_network_type(hw, mac->opmode); | ||
893 | } | ||
894 | } | ||
895 | |||
896 | rtlpriv->cfg->ops->scan_operation_backup(hw, SCAN_OPT_RESTORE); | ||
897 | } | ||
898 | |||
899 | static int rtl_op_set_key(struct ieee80211_hw *hw, enum set_key_cmd cmd, | ||
900 | struct ieee80211_vif *vif, struct ieee80211_sta *sta, | ||
901 | struct ieee80211_key_conf *key) | ||
902 | { | ||
903 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
904 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
905 | u8 key_type = NO_ENCRYPTION; | ||
906 | u8 key_idx; | ||
907 | bool group_key = false; | ||
908 | bool wep_only = false; | ||
909 | int err = 0; | ||
910 | u8 mac_addr[ETH_ALEN]; | ||
911 | u8 bcast_addr[ETH_ALEN] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; | ||
912 | u8 zero_addr[ETH_ALEN] = { 0 }; | ||
913 | |||
914 | if (rtlpriv->cfg->mod_params->sw_crypto || rtlpriv->sec.use_sw_sec) { | ||
915 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
916 | ("not open hw encryption\n")); | ||
917 | return -ENOSPC; /*User disabled HW-crypto */ | ||
918 | } | ||
919 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
920 | ("%s hardware based encryption for keyidx: %d, mac: %pM\n", | ||
921 | cmd == SET_KEY ? "Using" : "Disabling", key->keyidx, | ||
922 | sta ? sta->addr : bcast_addr)); | ||
923 | rtlpriv->sec.being_setkey = true; | ||
924 | rtl_ips_nic_on(hw); | ||
925 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
926 | /* <1> get encryption alg */ | ||
927 | |||
928 | switch (key->cipher) { | ||
929 | case WLAN_CIPHER_SUITE_WEP40: | ||
930 | key_type = WEP40_ENCRYPTION; | ||
931 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("alg:WEP40\n")); | ||
932 | break; | ||
933 | case WLAN_CIPHER_SUITE_WEP104: | ||
934 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
935 | ("alg:WEP104\n")); | ||
936 | key_type = WEP104_ENCRYPTION; | ||
937 | break; | ||
938 | case WLAN_CIPHER_SUITE_TKIP: | ||
939 | key_type = TKIP_ENCRYPTION; | ||
940 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("alg:TKIP\n")); | ||
941 | break; | ||
942 | case WLAN_CIPHER_SUITE_CCMP: | ||
943 | key_type = AESCCMP_ENCRYPTION; | ||
944 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("alg:CCMP\n")); | ||
945 | break; | ||
946 | default: | ||
947 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
948 | ("alg_err:%x!!!!:\n", key->cipher)); | ||
949 | goto out_unlock; | ||
950 | } | ||
951 | if (key_type == WEP40_ENCRYPTION || | ||
952 | key_type == WEP104_ENCRYPTION || | ||
953 | mac->opmode == NL80211_IFTYPE_ADHOC) | ||
954 | rtlpriv->sec.use_defaultkey = true; | ||
955 | |||
956 | /* <2> get key_idx */ | ||
957 | key_idx = (u8) (key->keyidx); | ||
958 | if (key_idx > 3) | ||
959 | goto out_unlock; | ||
960 | /* <3> if pairwise key enable_hw_sec */ | ||
961 | group_key = !(key->flags & IEEE80211_KEY_FLAG_PAIRWISE); | ||
962 | |||
963 | /* wep always be group key, but there are two conditions: | ||
964 | * 1) wep only: is just for wep enc, in this condition | ||
965 | * rtlpriv->sec.pairwise_enc_algorithm == NO_ENCRYPTION | ||
966 | * will be true & enable_hw_sec will be set when wep | ||
967 | * ke setting. | ||
968 | * 2) wep(group) + AES(pairwise): some AP like cisco | ||
969 | * may use it, in this condition enable_hw_sec will not | ||
970 | * be set when wep key setting */ | ||
971 | /* we must reset sec_info after lingked before set key, | ||
972 | * or some flag will be wrong*/ | ||
973 | if (mac->opmode == NL80211_IFTYPE_AP) { | ||
974 | if (!group_key || key_type == WEP40_ENCRYPTION || | ||
975 | key_type == WEP104_ENCRYPTION) { | ||
976 | if (group_key) | ||
977 | wep_only = true; | ||
978 | rtlpriv->cfg->ops->enable_hw_sec(hw); | ||
979 | } | ||
980 | } else { | ||
981 | if ((!group_key) || (mac->opmode == NL80211_IFTYPE_ADHOC) || | ||
982 | rtlpriv->sec.pairwise_enc_algorithm == NO_ENCRYPTION) { | ||
983 | if (rtlpriv->sec.pairwise_enc_algorithm == | ||
984 | NO_ENCRYPTION && | ||
985 | (key_type == WEP40_ENCRYPTION || | ||
986 | key_type == WEP104_ENCRYPTION)) | ||
987 | wep_only = true; | ||
988 | rtlpriv->sec.pairwise_enc_algorithm = key_type; | ||
989 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
990 | ("set enable_hw_sec, key_type:%x(OPEN:0 WEP40:1" | ||
991 | " TKIP:2 AES:4 WEP104:5)\n", key_type)); | ||
992 | rtlpriv->cfg->ops->enable_hw_sec(hw); | ||
993 | } | ||
994 | } | ||
995 | /* <4> set key based on cmd */ | ||
996 | switch (cmd) { | ||
997 | case SET_KEY: | ||
998 | if (wep_only) { | ||
999 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
1000 | ("set WEP(group/pairwise) key\n")); | ||
1001 | /* Pairwise key with an assigned MAC address. */ | ||
1002 | rtlpriv->sec.pairwise_enc_algorithm = key_type; | ||
1003 | rtlpriv->sec.group_enc_algorithm = key_type; | ||
1004 | /*set local buf about wep key. */ | ||
1005 | memcpy(rtlpriv->sec.key_buf[key_idx], | ||
1006 | key->key, key->keylen); | ||
1007 | rtlpriv->sec.key_len[key_idx] = key->keylen; | ||
1008 | memcpy(mac_addr, zero_addr, ETH_ALEN); | ||
1009 | } else if (group_key) { /* group key */ | ||
1010 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
1011 | ("set group key\n")); | ||
1012 | /* group key */ | ||
1013 | rtlpriv->sec.group_enc_algorithm = key_type; | ||
1014 | /*set local buf about group key. */ | ||
1015 | memcpy(rtlpriv->sec.key_buf[key_idx], | ||
1016 | key->key, key->keylen); | ||
1017 | rtlpriv->sec.key_len[key_idx] = key->keylen; | ||
1018 | memcpy(mac_addr, bcast_addr, ETH_ALEN); | ||
1019 | } else { /* pairwise key */ | ||
1020 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
1021 | ("set pairwise key\n")); | ||
1022 | if (!sta) { | ||
1023 | RT_ASSERT(false, ("pairwise key withnot" | ||
1024 | "mac_addr\n")); | ||
1025 | |||
1026 | err = -EOPNOTSUPP; | ||
1027 | goto out_unlock; | ||
1028 | } | ||
1029 | /* Pairwise key with an assigned MAC address. */ | ||
1030 | rtlpriv->sec.pairwise_enc_algorithm = key_type; | ||
1031 | /*set local buf about pairwise key. */ | ||
1032 | memcpy(rtlpriv->sec.key_buf[PAIRWISE_KEYIDX], | ||
1033 | key->key, key->keylen); | ||
1034 | rtlpriv->sec.key_len[PAIRWISE_KEYIDX] = key->keylen; | ||
1035 | rtlpriv->sec.pairwise_key = | ||
1036 | rtlpriv->sec.key_buf[PAIRWISE_KEYIDX]; | ||
1037 | memcpy(mac_addr, sta->addr, ETH_ALEN); | ||
1038 | } | ||
1039 | rtlpriv->cfg->ops->set_key(hw, key_idx, mac_addr, | ||
1040 | group_key, key_type, wep_only, | ||
1041 | false); | ||
1042 | /* <5> tell mac80211 do something: */ | ||
1043 | /*must use sw generate IV, or can not work !!!!. */ | ||
1044 | key->flags |= IEEE80211_KEY_FLAG_GENERATE_IV; | ||
1045 | key->hw_key_idx = key_idx; | ||
1046 | if (key_type == TKIP_ENCRYPTION) | ||
1047 | key->flags |= IEEE80211_KEY_FLAG_GENERATE_MMIC; | ||
1048 | break; | ||
1049 | case DISABLE_KEY: | ||
1050 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
1051 | ("disable key delete one entry\n")); | ||
1052 | /*set local buf about wep key. */ | ||
1053 | if (mac->opmode == NL80211_IFTYPE_AP) { | ||
1054 | if (sta) | ||
1055 | rtl_cam_del_entry(hw, sta->addr); | ||
1056 | } | ||
1057 | memset(rtlpriv->sec.key_buf[key_idx], 0, key->keylen); | ||
1058 | rtlpriv->sec.key_len[key_idx] = 0; | ||
1059 | memcpy(mac_addr, zero_addr, ETH_ALEN); | ||
1060 | /* | ||
1061 | *mac80211 will delete entrys one by one, | ||
1062 | *so don't use rtl_cam_reset_all_entry | ||
1063 | *or clear all entry here. | ||
1064 | */ | ||
1065 | rtl_cam_delete_one_entry(hw, mac_addr, key_idx); | ||
1066 | break; | ||
1067 | default: | ||
1068 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1069 | ("cmd_err:%x!!!!:\n", cmd)); | ||
1070 | } | ||
1071 | out_unlock: | ||
1072 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
1073 | rtlpriv->sec.being_setkey = false; | ||
1074 | return err; | ||
1075 | } | ||
1076 | |||
1077 | static void rtl_op_rfkill_poll(struct ieee80211_hw *hw) | ||
1078 | { | ||
1079 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1080 | |||
1081 | bool radio_state; | ||
1082 | bool blocked; | ||
1083 | u8 valid = 0; | ||
1084 | |||
1085 | if (!test_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status)) | ||
1086 | return; | ||
1087 | |||
1088 | mutex_lock(&rtlpriv->locks.conf_mutex); | ||
1089 | |||
1090 | /*if Radio On return true here */ | ||
1091 | radio_state = rtlpriv->cfg->ops->radio_onoff_checking(hw, &valid); | ||
1092 | |||
1093 | if (valid) { | ||
1094 | if (unlikely(radio_state != rtlpriv->rfkill.rfkill_state)) { | ||
1095 | rtlpriv->rfkill.rfkill_state = radio_state; | ||
1096 | |||
1097 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
1098 | (KERN_INFO "wireless radio switch turned %s\n", | ||
1099 | radio_state ? "on" : "off")); | ||
1100 | |||
1101 | blocked = (rtlpriv->rfkill.rfkill_state == 1) ? 0 : 1; | ||
1102 | wiphy_rfkill_set_hw_state(hw->wiphy, blocked); | ||
1103 | } | ||
1104 | } | ||
1105 | |||
1106 | mutex_unlock(&rtlpriv->locks.conf_mutex); | ||
1107 | } | ||
1108 | |||
1109 | /* this function is called by mac80211 to flush tx buffer | ||
1110 | * before switch channle or power save, or tx buffer packet | ||
1111 | * maybe send after offchannel or rf sleep, this may cause | ||
1112 | * dis-association by AP */ | ||
1113 | static void rtl_op_flush(struct ieee80211_hw *hw, bool drop) | ||
1114 | { | ||
1115 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1116 | |||
1117 | if (rtlpriv->intf_ops->flush) | ||
1118 | rtlpriv->intf_ops->flush(hw, drop); | ||
1119 | } | ||
1120 | |||
1121 | const struct ieee80211_ops rtl_ops = { | ||
1122 | .start = rtl_op_start, | ||
1123 | .stop = rtl_op_stop, | ||
1124 | .tx = rtl_op_tx, | ||
1125 | .add_interface = rtl_op_add_interface, | ||
1126 | .remove_interface = rtl_op_remove_interface, | ||
1127 | .config = rtl_op_config, | ||
1128 | .configure_filter = rtl_op_configure_filter, | ||
1129 | .sta_add = rtl_op_sta_add, | ||
1130 | .sta_remove = rtl_op_sta_remove, | ||
1131 | .set_key = rtl_op_set_key, | ||
1132 | .conf_tx = rtl_op_conf_tx, | ||
1133 | .bss_info_changed = rtl_op_bss_info_changed, | ||
1134 | .get_tsf = rtl_op_get_tsf, | ||
1135 | .set_tsf = rtl_op_set_tsf, | ||
1136 | .reset_tsf = rtl_op_reset_tsf, | ||
1137 | .sta_notify = rtl_op_sta_notify, | ||
1138 | .ampdu_action = rtl_op_ampdu_action, | ||
1139 | .sw_scan_start = rtl_op_sw_scan_start, | ||
1140 | .sw_scan_complete = rtl_op_sw_scan_complete, | ||
1141 | .rfkill_poll = rtl_op_rfkill_poll, | ||
1142 | .flush = rtl_op_flush, | ||
1143 | }; | ||
diff --git a/drivers/net/wireless/rtlwifi/core.h b/drivers/net/wireless/rtlwifi/core.h new file mode 100644 index 000000000000..4b247db2861d --- /dev/null +++ b/drivers/net/wireless/rtlwifi/core.h | |||
@@ -0,0 +1,43 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * Tmis program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * Tmis program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * tmis program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * Tme full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_CORE_H__ | ||
31 | #define __RTL_CORE_H__ | ||
32 | |||
33 | #define RTL_SUPPORTED_FILTERS \ | ||
34 | (FIF_PROMISC_IN_BSS | \ | ||
35 | FIF_ALLMULTI | FIF_CONTROL | \ | ||
36 | FIF_OTHER_BSS | \ | ||
37 | FIF_FCSFAIL | \ | ||
38 | FIF_BCN_PRBRESP_PROMISC) | ||
39 | |||
40 | #define RTL_SUPPORTED_CTRL_FILTER 0xFF | ||
41 | |||
42 | extern const struct ieee80211_ops rtl_ops; | ||
43 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/debug.c b/drivers/net/wireless/rtlwifi/debug.c new file mode 100644 index 000000000000..5fa73852cb66 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/debug.c | |||
@@ -0,0 +1,50 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * Tmis program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * Tmis program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * tmis program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * Tme full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | *****************************************************************************/ | ||
28 | |||
29 | #include "wifi.h" | ||
30 | |||
31 | void rtl_dbgp_flag_init(struct ieee80211_hw *hw) | ||
32 | { | ||
33 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
34 | u8 i; | ||
35 | |||
36 | rtlpriv->dbg.global_debuglevel = DBG_EMERG; | ||
37 | |||
38 | rtlpriv->dbg.global_debugcomponents = | ||
39 | COMP_ERR | COMP_FW | COMP_INIT | COMP_RECV | COMP_SEND | | ||
40 | COMP_MLME | COMP_SCAN | COMP_INTR | COMP_LED | COMP_SEC | | ||
41 | COMP_BEACON | COMP_RATE | COMP_RXDESC | COMP_DIG | COMP_TXAGC | | ||
42 | COMP_POWER | COMP_POWER_TRACKING | COMP_BB_POWERSAVING | COMP_SWAS | | ||
43 | COMP_RF | COMP_TURBO | COMP_RATR | COMP_CMD | | ||
44 | COMP_EFUSE | COMP_QOS | COMP_MAC80211 | COMP_REGD | COMP_CHAN; | ||
45 | |||
46 | for (i = 0; i < DBGP_TYPE_MAX; i++) | ||
47 | rtlpriv->dbg.dbgp_type[i] = 0; | ||
48 | |||
49 | /*Init Debug flag enable condition */ | ||
50 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/debug.h b/drivers/net/wireless/rtlwifi/debug.h new file mode 100644 index 000000000000..e4aa8687408c --- /dev/null +++ b/drivers/net/wireless/rtlwifi/debug.h | |||
@@ -0,0 +1,213 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * Tmis program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * Tmis program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * tmis program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * Tme full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | *****************************************************************************/ | ||
28 | |||
29 | #ifndef __RTL_DEBUG_H__ | ||
30 | #define __RTL_DEBUG_H__ | ||
31 | |||
32 | /*-------------------------------------------------------------- | ||
33 | Debug level | ||
34 | --------------------------------------------------------------*/ | ||
35 | /* | ||
36 | *Fatal bug. | ||
37 | *For example, Tx/Rx/IO locked up, | ||
38 | *memory access violation, | ||
39 | *resource allocation failed, | ||
40 | *unexpected HW behavior, HW BUG | ||
41 | *and so on. | ||
42 | */ | ||
43 | #define DBG_EMERG 0 | ||
44 | |||
45 | /* | ||
46 | *Abnormal, rare, or unexpeted cases. | ||
47 | *For example, Packet/IO Ctl canceled, | ||
48 | *device suprisely unremoved and so on. | ||
49 | */ | ||
50 | #define DBG_WARNING 2 | ||
51 | |||
52 | /* | ||
53 | *Normal case driver developer should | ||
54 | *open, we can see link status like | ||
55 | *assoc/AddBA/DHCP/adapter start and | ||
56 | *so on basic and useful infromations. | ||
57 | */ | ||
58 | #define DBG_DMESG 3 | ||
59 | |||
60 | /* | ||
61 | *Normal case with useful information | ||
62 | *about current SW or HW state. | ||
63 | *For example, Tx/Rx descriptor to fill, | ||
64 | *Tx/Rx descriptor completed status, | ||
65 | *SW protocol state change, dynamic | ||
66 | *mechanism state change and so on. | ||
67 | */ | ||
68 | #define DBG_LOUD 4 | ||
69 | |||
70 | /* | ||
71 | *Normal case with detail execution | ||
72 | *flow or information. | ||
73 | */ | ||
74 | #define DBG_TRACE 5 | ||
75 | |||
76 | /*-------------------------------------------------------------- | ||
77 | Define the rt_trace components | ||
78 | --------------------------------------------------------------*/ | ||
79 | #define COMP_ERR BIT(0) | ||
80 | #define COMP_FW BIT(1) | ||
81 | #define COMP_INIT BIT(2) /*For init/deinit */ | ||
82 | #define COMP_RECV BIT(3) /*For Rx. */ | ||
83 | #define COMP_SEND BIT(4) /*For Tx. */ | ||
84 | #define COMP_MLME BIT(5) /*For MLME. */ | ||
85 | #define COMP_SCAN BIT(6) /*For Scan. */ | ||
86 | #define COMP_INTR BIT(7) /*For interrupt Related. */ | ||
87 | #define COMP_LED BIT(8) /*For LED. */ | ||
88 | #define COMP_SEC BIT(9) /*For sec. */ | ||
89 | #define COMP_BEACON BIT(10) /*For beacon. */ | ||
90 | #define COMP_RATE BIT(11) /*For rate. */ | ||
91 | #define COMP_RXDESC BIT(12) /*For rx desc. */ | ||
92 | #define COMP_DIG BIT(13) /*For DIG */ | ||
93 | #define COMP_TXAGC BIT(14) /*For Tx power */ | ||
94 | #define COMP_HIPWR BIT(15) /*For High Power Mechanism */ | ||
95 | #define COMP_POWER BIT(16) /*For lps/ips/aspm. */ | ||
96 | #define COMP_POWER_TRACKING BIT(17) /*For TX POWER TRACKING */ | ||
97 | #define COMP_BB_POWERSAVING BIT(18) | ||
98 | #define COMP_SWAS BIT(19) /*For SW Antenna Switch */ | ||
99 | #define COMP_RF BIT(20) /*For RF. */ | ||
100 | #define COMP_TURBO BIT(21) /*For EDCA TURBO. */ | ||
101 | #define COMP_RATR BIT(22) | ||
102 | #define COMP_CMD BIT(23) | ||
103 | #define COMP_EFUSE BIT(24) | ||
104 | #define COMP_QOS BIT(25) | ||
105 | #define COMP_MAC80211 BIT(26) | ||
106 | #define COMP_REGD BIT(27) | ||
107 | #define COMP_CHAN BIT(28) | ||
108 | #define COMP_USB BIT(29) | ||
109 | |||
110 | /*-------------------------------------------------------------- | ||
111 | Define the rt_print components | ||
112 | --------------------------------------------------------------*/ | ||
113 | /* Define EEPROM and EFUSE check module bit*/ | ||
114 | #define EEPROM_W BIT(0) | ||
115 | #define EFUSE_PG BIT(1) | ||
116 | #define EFUSE_READ_ALL BIT(2) | ||
117 | |||
118 | /* Define init check for module bit*/ | ||
119 | #define INIT_EEPROM BIT(0) | ||
120 | #define INIT_TxPower BIT(1) | ||
121 | #define INIT_IQK BIT(2) | ||
122 | #define INIT_RF BIT(3) | ||
123 | |||
124 | /* Define PHY-BB/RF/MAC check module bit */ | ||
125 | #define PHY_BBR BIT(0) | ||
126 | #define PHY_BBW BIT(1) | ||
127 | #define PHY_RFR BIT(2) | ||
128 | #define PHY_RFW BIT(3) | ||
129 | #define PHY_MACR BIT(4) | ||
130 | #define PHY_MACW BIT(5) | ||
131 | #define PHY_ALLR BIT(6) | ||
132 | #define PHY_ALLW BIT(7) | ||
133 | #define PHY_TXPWR BIT(8) | ||
134 | #define PHY_PWRDIFF BIT(9) | ||
135 | |||
136 | enum dbgp_flag_e { | ||
137 | FQOS = 0, | ||
138 | FTX = 1, | ||
139 | FRX = 2, | ||
140 | FSEC = 3, | ||
141 | FMGNT = 4, | ||
142 | FMLME = 5, | ||
143 | FRESOURCE = 6, | ||
144 | FBEACON = 7, | ||
145 | FISR = 8, | ||
146 | FPHY = 9, | ||
147 | FMP = 10, | ||
148 | FEEPROM = 11, | ||
149 | FPWR = 12, | ||
150 | FDM = 13, | ||
151 | FDBGCtrl = 14, | ||
152 | FC2H = 15, | ||
153 | FBT = 16, | ||
154 | FINIT = 17, | ||
155 | FIOCTL = 18, | ||
156 | DBGP_TYPE_MAX | ||
157 | }; | ||
158 | |||
159 | #define RT_ASSERT(_exp, fmt) \ | ||
160 | do { \ | ||
161 | if (!(_exp)) { \ | ||
162 | printk(KERN_DEBUG "%s:%s(): ", KBUILD_MODNAME, \ | ||
163 | __func__); \ | ||
164 | printk fmt; \ | ||
165 | } \ | ||
166 | } while (0); | ||
167 | |||
168 | #define RT_TRACE(rtlpriv, comp, level, fmt)\ | ||
169 | do { \ | ||
170 | if (unlikely(((comp) & rtlpriv->dbg.global_debugcomponents) && \ | ||
171 | ((level) <= rtlpriv->dbg.global_debuglevel))) {\ | ||
172 | printk(KERN_DEBUG "%s:%s():<%lx-%x> ", KBUILD_MODNAME, \ | ||
173 | __func__, in_interrupt(), in_atomic()); \ | ||
174 | printk fmt; \ | ||
175 | } \ | ||
176 | } while (0); | ||
177 | |||
178 | #define RTPRINT(rtlpriv, dbgtype, dbgflag, printstr) \ | ||
179 | do { \ | ||
180 | if (unlikely(rtlpriv->dbg.dbgp_type[dbgtype] & dbgflag)) { \ | ||
181 | printk(KERN_DEBUG "%s: ", KBUILD_MODNAME); \ | ||
182 | printk printstr; \ | ||
183 | } \ | ||
184 | } while (0); | ||
185 | |||
186 | #define RT_PRINT_DATA(rtlpriv, _comp, _level, _titlestring, _hexdata, \ | ||
187 | _hexdatalen) \ | ||
188 | do {\ | ||
189 | if (unlikely(((_comp) & rtlpriv->dbg.global_debugcomponents) &&\ | ||
190 | (_level <= rtlpriv->dbg.global_debuglevel))) { \ | ||
191 | int __i; \ | ||
192 | u8* ptr = (u8 *)_hexdata; \ | ||
193 | printk(KERN_DEBUG "%s: ", KBUILD_MODNAME); \ | ||
194 | printk("In process \"%s\" (pid %i):", current->comm,\ | ||
195 | current->pid); \ | ||
196 | printk(_titlestring); \ | ||
197 | for (__i = 0; __i < (int)_hexdatalen; __i++) { \ | ||
198 | printk("%02X%s", ptr[__i], (((__i + 1) % 4)\ | ||
199 | == 0) ? " " : " ");\ | ||
200 | if (((__i + 1) % 16) == 0) \ | ||
201 | printk("\n"); \ | ||
202 | } \ | ||
203 | printk(KERN_DEBUG "\n"); \ | ||
204 | } \ | ||
205 | } while (0); | ||
206 | |||
207 | #define MAC_FMT "%02x:%02x:%02x:%02x:%02x:%02x" | ||
208 | #define MAC_ARG(x) \ | ||
209 | ((u8 *)(x))[0], ((u8 *)(x))[1], ((u8 *)(x))[2],\ | ||
210 | ((u8 *)(x))[3], ((u8 *)(x))[4], ((u8 *)(x))[5] | ||
211 | |||
212 | void rtl_dbgp_flag_init(struct ieee80211_hw *hw); | ||
213 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/efuse.c b/drivers/net/wireless/rtlwifi/efuse.c new file mode 100644 index 000000000000..50de6f5d8a56 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/efuse.c | |||
@@ -0,0 +1,1210 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * Tmis program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * Tmis program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * tmis program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * Tme full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "efuse.h" | ||
32 | |||
33 | static const u8 MAX_PGPKT_SIZE = 9; | ||
34 | static const u8 PGPKT_DATA_SIZE = 8; | ||
35 | static const int EFUSE_MAX_SIZE = 512; | ||
36 | |||
37 | static const u8 EFUSE_OOB_PROTECT_BYTES = 15; | ||
38 | |||
39 | static const struct efuse_map RTL8712_SDIO_EFUSE_TABLE[] = { | ||
40 | {0, 0, 0, 2}, | ||
41 | {0, 1, 0, 2}, | ||
42 | {0, 2, 0, 2}, | ||
43 | {1, 0, 0, 1}, | ||
44 | {1, 0, 1, 1}, | ||
45 | {1, 1, 0, 1}, | ||
46 | {1, 1, 1, 3}, | ||
47 | {1, 3, 0, 17}, | ||
48 | {3, 3, 1, 48}, | ||
49 | {10, 0, 0, 6}, | ||
50 | {10, 3, 0, 1}, | ||
51 | {10, 3, 1, 1}, | ||
52 | {11, 0, 0, 28} | ||
53 | }; | ||
54 | |||
55 | static void efuse_shadow_read_1byte(struct ieee80211_hw *hw, u16 offset, | ||
56 | u8 *value); | ||
57 | static void efuse_shadow_read_2byte(struct ieee80211_hw *hw, u16 offset, | ||
58 | u16 *value); | ||
59 | static void efuse_shadow_read_4byte(struct ieee80211_hw *hw, u16 offset, | ||
60 | u32 *value); | ||
61 | static void efuse_shadow_write_1byte(struct ieee80211_hw *hw, u16 offset, | ||
62 | u8 value); | ||
63 | static void efuse_shadow_write_2byte(struct ieee80211_hw *hw, u16 offset, | ||
64 | u16 value); | ||
65 | static void efuse_shadow_write_4byte(struct ieee80211_hw *hw, u16 offset, | ||
66 | u32 value); | ||
67 | static int efuse_one_byte_read(struct ieee80211_hw *hw, u16 addr, | ||
68 | u8 *data); | ||
69 | static int efuse_one_byte_write(struct ieee80211_hw *hw, u16 addr, | ||
70 | u8 data); | ||
71 | static void efuse_read_all_map(struct ieee80211_hw *hw, u8 *efuse); | ||
72 | static int efuse_pg_packet_read(struct ieee80211_hw *hw, u8 offset, | ||
73 | u8 *data); | ||
74 | static int efuse_pg_packet_write(struct ieee80211_hw *hw, u8 offset, | ||
75 | u8 word_en, u8 *data); | ||
76 | static void efuse_word_enable_data_read(u8 word_en, u8 *sourdata, | ||
77 | u8 *targetdata); | ||
78 | static u8 efuse_word_enable_data_write(struct ieee80211_hw *hw, | ||
79 | u16 efuse_addr, u8 word_en, u8 *data); | ||
80 | static void efuse_power_switch(struct ieee80211_hw *hw, u8 write, | ||
81 | u8 pwrstate); | ||
82 | static u16 efuse_get_current_size(struct ieee80211_hw *hw); | ||
83 | static u8 efuse_calculate_word_cnts(u8 word_en); | ||
84 | |||
85 | void efuse_initialize(struct ieee80211_hw *hw) | ||
86 | { | ||
87 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
88 | u8 bytetemp; | ||
89 | u8 temp; | ||
90 | |||
91 | bytetemp = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[SYS_FUNC_EN] + 1); | ||
92 | temp = bytetemp | 0x20; | ||
93 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[SYS_FUNC_EN] + 1, temp); | ||
94 | |||
95 | bytetemp = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[SYS_ISO_CTRL] + 1); | ||
96 | temp = bytetemp & 0xFE; | ||
97 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[SYS_ISO_CTRL] + 1, temp); | ||
98 | |||
99 | bytetemp = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_TEST] + 3); | ||
100 | temp = bytetemp | 0x80; | ||
101 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_TEST] + 3, temp); | ||
102 | |||
103 | rtl_write_byte(rtlpriv, 0x2F8, 0x3); | ||
104 | |||
105 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3, 0x72); | ||
106 | |||
107 | } | ||
108 | |||
109 | u8 efuse_read_1byte(struct ieee80211_hw *hw, u16 address) | ||
110 | { | ||
111 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
112 | u8 data; | ||
113 | u8 bytetemp; | ||
114 | u8 temp; | ||
115 | u32 k = 0; | ||
116 | const u32 efuse_len = | ||
117 | rtlpriv->cfg->maps[EFUSE_REAL_CONTENT_SIZE]; | ||
118 | |||
119 | if (address < efuse_len) { | ||
120 | temp = address & 0xFF; | ||
121 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 1, | ||
122 | temp); | ||
123 | bytetemp = rtl_read_byte(rtlpriv, | ||
124 | rtlpriv->cfg->maps[EFUSE_CTRL] + 2); | ||
125 | temp = ((address >> 8) & 0x03) | (bytetemp & 0xFC); | ||
126 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 2, | ||
127 | temp); | ||
128 | |||
129 | bytetemp = rtl_read_byte(rtlpriv, | ||
130 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3); | ||
131 | temp = bytetemp & 0x7F; | ||
132 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3, | ||
133 | temp); | ||
134 | |||
135 | bytetemp = rtl_read_byte(rtlpriv, | ||
136 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3); | ||
137 | while (!(bytetemp & 0x80)) { | ||
138 | bytetemp = rtl_read_byte(rtlpriv, | ||
139 | rtlpriv->cfg-> | ||
140 | maps[EFUSE_CTRL] + 3); | ||
141 | k++; | ||
142 | if (k == 1000) { | ||
143 | k = 0; | ||
144 | break; | ||
145 | } | ||
146 | } | ||
147 | data = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL]); | ||
148 | return data; | ||
149 | } else | ||
150 | return 0xFF; | ||
151 | |||
152 | } | ||
153 | EXPORT_SYMBOL(efuse_read_1byte); | ||
154 | |||
155 | void efuse_write_1byte(struct ieee80211_hw *hw, u16 address, u8 value) | ||
156 | { | ||
157 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
158 | u8 bytetemp; | ||
159 | u8 temp; | ||
160 | u32 k = 0; | ||
161 | const u32 efuse_len = | ||
162 | rtlpriv->cfg->maps[EFUSE_REAL_CONTENT_SIZE]; | ||
163 | |||
164 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
165 | ("Addr=%x Data =%x\n", address, value)); | ||
166 | |||
167 | if (address < efuse_len) { | ||
168 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL], value); | ||
169 | |||
170 | temp = address & 0xFF; | ||
171 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 1, | ||
172 | temp); | ||
173 | bytetemp = rtl_read_byte(rtlpriv, | ||
174 | rtlpriv->cfg->maps[EFUSE_CTRL] + 2); | ||
175 | |||
176 | temp = ((address >> 8) & 0x03) | (bytetemp & 0xFC); | ||
177 | rtl_write_byte(rtlpriv, | ||
178 | rtlpriv->cfg->maps[EFUSE_CTRL] + 2, temp); | ||
179 | |||
180 | bytetemp = rtl_read_byte(rtlpriv, | ||
181 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3); | ||
182 | temp = bytetemp | 0x80; | ||
183 | rtl_write_byte(rtlpriv, | ||
184 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3, temp); | ||
185 | |||
186 | bytetemp = rtl_read_byte(rtlpriv, | ||
187 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3); | ||
188 | |||
189 | while (bytetemp & 0x80) { | ||
190 | bytetemp = rtl_read_byte(rtlpriv, | ||
191 | rtlpriv->cfg-> | ||
192 | maps[EFUSE_CTRL] + 3); | ||
193 | k++; | ||
194 | if (k == 100) { | ||
195 | k = 0; | ||
196 | break; | ||
197 | } | ||
198 | } | ||
199 | } | ||
200 | |||
201 | } | ||
202 | |||
203 | void read_efuse_byte(struct ieee80211_hw *hw, u16 _offset, u8 *pbuf) | ||
204 | { | ||
205 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
206 | u32 value32; | ||
207 | u8 readbyte; | ||
208 | u16 retry; | ||
209 | |||
210 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 1, | ||
211 | (_offset & 0xff)); | ||
212 | readbyte = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 2); | ||
213 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 2, | ||
214 | ((_offset >> 8) & 0x03) | (readbyte & 0xfc)); | ||
215 | |||
216 | readbyte = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3); | ||
217 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3, | ||
218 | (readbyte & 0x7f)); | ||
219 | |||
220 | retry = 0; | ||
221 | value32 = rtl_read_dword(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL]); | ||
222 | while (!(((value32 >> 24) & 0xff) & 0x80) && (retry < 10000)) { | ||
223 | value32 = rtl_read_dword(rtlpriv, | ||
224 | rtlpriv->cfg->maps[EFUSE_CTRL]); | ||
225 | retry++; | ||
226 | } | ||
227 | |||
228 | udelay(50); | ||
229 | value32 = rtl_read_dword(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL]); | ||
230 | |||
231 | *pbuf = (u8) (value32 & 0xff); | ||
232 | } | ||
233 | |||
234 | void read_efuse(struct ieee80211_hw *hw, u16 _offset, u16 _size_byte, u8 *pbuf) | ||
235 | { | ||
236 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
237 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
238 | u8 *efuse_tbl; | ||
239 | u8 rtemp8[1]; | ||
240 | u16 efuse_addr = 0; | ||
241 | u8 offset, wren; | ||
242 | u16 i; | ||
243 | u16 j; | ||
244 | const u16 efuse_max_section = | ||
245 | rtlpriv->cfg->maps[EFUSE_MAX_SECTION_MAP]; | ||
246 | const u32 efuse_len = | ||
247 | rtlpriv->cfg->maps[EFUSE_REAL_CONTENT_SIZE]; | ||
248 | u16 **efuse_word; | ||
249 | u16 efuse_utilized = 0; | ||
250 | u8 efuse_usage; | ||
251 | |||
252 | if ((_offset + _size_byte) > rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE]) { | ||
253 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
254 | ("read_efuse(): Invalid offset(%#x) with read " | ||
255 | "bytes(%#x)!!\n", _offset, _size_byte)); | ||
256 | return; | ||
257 | } | ||
258 | |||
259 | /* allocate memory for efuse_tbl and efuse_word */ | ||
260 | efuse_tbl = kmalloc(rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE] * | ||
261 | sizeof(u8), GFP_ATOMIC); | ||
262 | if (!efuse_tbl) | ||
263 | return; | ||
264 | efuse_word = kmalloc(EFUSE_MAX_WORD_UNIT * sizeof(u16 *), GFP_ATOMIC); | ||
265 | if (!efuse_word) | ||
266 | goto done; | ||
267 | for (i = 0; i < EFUSE_MAX_WORD_UNIT; i++) { | ||
268 | efuse_word[i] = kmalloc(efuse_max_section * sizeof(u16), | ||
269 | GFP_ATOMIC); | ||
270 | if (!efuse_word[i]) | ||
271 | goto done; | ||
272 | } | ||
273 | |||
274 | for (i = 0; i < efuse_max_section; i++) | ||
275 | for (j = 0; j < EFUSE_MAX_WORD_UNIT; j++) | ||
276 | efuse_word[j][i] = 0xFFFF; | ||
277 | |||
278 | read_efuse_byte(hw, efuse_addr, rtemp8); | ||
279 | if (*rtemp8 != 0xFF) { | ||
280 | efuse_utilized++; | ||
281 | RTPRINT(rtlpriv, FEEPROM, EFUSE_READ_ALL, | ||
282 | ("Addr=%d\n", efuse_addr)); | ||
283 | efuse_addr++; | ||
284 | } | ||
285 | |||
286 | while ((*rtemp8 != 0xFF) && (efuse_addr < efuse_len)) { | ||
287 | offset = ((*rtemp8 >> 4) & 0x0f); | ||
288 | |||
289 | if (offset < efuse_max_section) { | ||
290 | wren = (*rtemp8 & 0x0f); | ||
291 | RTPRINT(rtlpriv, FEEPROM, EFUSE_READ_ALL, | ||
292 | ("offset-%d Worden=%x\n", offset, wren)); | ||
293 | |||
294 | for (i = 0; i < EFUSE_MAX_WORD_UNIT; i++) { | ||
295 | if (!(wren & 0x01)) { | ||
296 | RTPRINT(rtlpriv, FEEPROM, | ||
297 | EFUSE_READ_ALL, ("Addr=%d\n", | ||
298 | efuse_addr)); | ||
299 | |||
300 | read_efuse_byte(hw, efuse_addr, rtemp8); | ||
301 | efuse_addr++; | ||
302 | efuse_utilized++; | ||
303 | efuse_word[i][offset] = | ||
304 | (*rtemp8 & 0xff); | ||
305 | |||
306 | if (efuse_addr >= efuse_len) | ||
307 | break; | ||
308 | |||
309 | RTPRINT(rtlpriv, FEEPROM, | ||
310 | EFUSE_READ_ALL, ("Addr=%d\n", | ||
311 | efuse_addr)); | ||
312 | |||
313 | read_efuse_byte(hw, efuse_addr, rtemp8); | ||
314 | efuse_addr++; | ||
315 | efuse_utilized++; | ||
316 | efuse_word[i][offset] |= | ||
317 | (((u16)*rtemp8 << 8) & 0xff00); | ||
318 | |||
319 | if (efuse_addr >= efuse_len) | ||
320 | break; | ||
321 | } | ||
322 | |||
323 | wren >>= 1; | ||
324 | } | ||
325 | } | ||
326 | |||
327 | RTPRINT(rtlpriv, FEEPROM, EFUSE_READ_ALL, | ||
328 | ("Addr=%d\n", efuse_addr)); | ||
329 | read_efuse_byte(hw, efuse_addr, rtemp8); | ||
330 | if (*rtemp8 != 0xFF && (efuse_addr < efuse_len)) { | ||
331 | efuse_utilized++; | ||
332 | efuse_addr++; | ||
333 | } | ||
334 | } | ||
335 | |||
336 | for (i = 0; i < efuse_max_section; i++) { | ||
337 | for (j = 0; j < EFUSE_MAX_WORD_UNIT; j++) { | ||
338 | efuse_tbl[(i * 8) + (j * 2)] = | ||
339 | (efuse_word[j][i] & 0xff); | ||
340 | efuse_tbl[(i * 8) + ((j * 2) + 1)] = | ||
341 | ((efuse_word[j][i] >> 8) & 0xff); | ||
342 | } | ||
343 | } | ||
344 | |||
345 | for (i = 0; i < _size_byte; i++) | ||
346 | pbuf[i] = efuse_tbl[_offset + i]; | ||
347 | |||
348 | rtlefuse->efuse_usedbytes = efuse_utilized; | ||
349 | efuse_usage = (u8) ((efuse_utilized * 100) / efuse_len); | ||
350 | rtlefuse->efuse_usedpercentage = efuse_usage; | ||
351 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_EFUSE_BYTES, | ||
352 | (u8 *)&efuse_utilized); | ||
353 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_EFUSE_USAGE, | ||
354 | (u8 *)&efuse_usage); | ||
355 | done: | ||
356 | for (i = 0; i < EFUSE_MAX_WORD_UNIT; i++) | ||
357 | kfree(efuse_word[i]); | ||
358 | kfree(efuse_word); | ||
359 | kfree(efuse_tbl); | ||
360 | } | ||
361 | |||
362 | bool efuse_shadow_update_chk(struct ieee80211_hw *hw) | ||
363 | { | ||
364 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
365 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
366 | u8 section_idx, i, Base; | ||
367 | u16 words_need = 0, hdr_num = 0, totalbytes, efuse_used; | ||
368 | bool wordchanged, result = true; | ||
369 | |||
370 | for (section_idx = 0; section_idx < 16; section_idx++) { | ||
371 | Base = section_idx * 8; | ||
372 | wordchanged = false; | ||
373 | |||
374 | for (i = 0; i < 8; i = i + 2) { | ||
375 | if ((rtlefuse->efuse_map[EFUSE_INIT_MAP][Base + i] != | ||
376 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][Base + i]) || | ||
377 | (rtlefuse->efuse_map[EFUSE_INIT_MAP][Base + i + 1] != | ||
378 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][Base + i + | ||
379 | 1])) { | ||
380 | words_need++; | ||
381 | wordchanged = true; | ||
382 | } | ||
383 | } | ||
384 | |||
385 | if (wordchanged == true) | ||
386 | hdr_num++; | ||
387 | } | ||
388 | |||
389 | totalbytes = hdr_num + words_need * 2; | ||
390 | efuse_used = rtlefuse->efuse_usedbytes; | ||
391 | |||
392 | if ((totalbytes + efuse_used) >= | ||
393 | (EFUSE_MAX_SIZE - EFUSE_OOB_PROTECT_BYTES)) | ||
394 | result = false; | ||
395 | |||
396 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
397 | ("efuse_shadow_update_chk(): totalbytes(%#x), " | ||
398 | "hdr_num(%#x), words_need(%#x), efuse_used(%d)\n", | ||
399 | totalbytes, hdr_num, words_need, efuse_used)); | ||
400 | |||
401 | return result; | ||
402 | } | ||
403 | |||
404 | void efuse_shadow_read(struct ieee80211_hw *hw, u8 type, | ||
405 | u16 offset, u32 *value) | ||
406 | { | ||
407 | if (type == 1) | ||
408 | efuse_shadow_read_1byte(hw, offset, (u8 *) value); | ||
409 | else if (type == 2) | ||
410 | efuse_shadow_read_2byte(hw, offset, (u16 *) value); | ||
411 | else if (type == 4) | ||
412 | efuse_shadow_read_4byte(hw, offset, (u32 *) value); | ||
413 | |||
414 | } | ||
415 | |||
416 | void efuse_shadow_write(struct ieee80211_hw *hw, u8 type, u16 offset, | ||
417 | u32 value) | ||
418 | { | ||
419 | if (type == 1) | ||
420 | efuse_shadow_write_1byte(hw, offset, (u8) value); | ||
421 | else if (type == 2) | ||
422 | efuse_shadow_write_2byte(hw, offset, (u16) value); | ||
423 | else if (type == 4) | ||
424 | efuse_shadow_write_4byte(hw, offset, value); | ||
425 | |||
426 | } | ||
427 | |||
428 | bool efuse_shadow_update(struct ieee80211_hw *hw) | ||
429 | { | ||
430 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
431 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
432 | u16 i, offset, base; | ||
433 | u8 word_en = 0x0F; | ||
434 | u8 first_pg = false; | ||
435 | |||
436 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, ("--->\n")); | ||
437 | |||
438 | if (!efuse_shadow_update_chk(hw)) { | ||
439 | efuse_read_all_map(hw, &rtlefuse->efuse_map[EFUSE_INIT_MAP][0]); | ||
440 | memcpy(&rtlefuse->efuse_map[EFUSE_MODIFY_MAP][0], | ||
441 | &rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
442 | rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE]); | ||
443 | |||
444 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
445 | ("<---efuse out of capacity!!\n")); | ||
446 | return false; | ||
447 | } | ||
448 | efuse_power_switch(hw, true, true); | ||
449 | |||
450 | for (offset = 0; offset < 16; offset++) { | ||
451 | |||
452 | word_en = 0x0F; | ||
453 | base = offset * 8; | ||
454 | |||
455 | for (i = 0; i < 8; i++) { | ||
456 | if (first_pg == true) { | ||
457 | |||
458 | word_en &= ~(BIT(i / 2)); | ||
459 | |||
460 | rtlefuse->efuse_map[EFUSE_INIT_MAP][base + i] = | ||
461 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][base + i]; | ||
462 | } else { | ||
463 | |||
464 | if (rtlefuse->efuse_map[EFUSE_INIT_MAP][base + i] != | ||
465 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][base + i]) { | ||
466 | word_en &= ~(BIT(i / 2)); | ||
467 | |||
468 | rtlefuse->efuse_map[EFUSE_INIT_MAP][base + i] = | ||
469 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][base + i]; | ||
470 | } | ||
471 | } | ||
472 | } | ||
473 | |||
474 | if (word_en != 0x0F) { | ||
475 | u8 tmpdata[8]; | ||
476 | memcpy(tmpdata, | ||
477 | &rtlefuse->efuse_map[EFUSE_MODIFY_MAP][base], | ||
478 | 8); | ||
479 | RT_PRINT_DATA(rtlpriv, COMP_INIT, DBG_LOUD, | ||
480 | ("U-efuse\n"), tmpdata, 8); | ||
481 | |||
482 | if (!efuse_pg_packet_write(hw, (u8) offset, word_en, | ||
483 | tmpdata)) { | ||
484 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
485 | ("PG section(%#x) fail!!\n", offset)); | ||
486 | break; | ||
487 | } | ||
488 | } | ||
489 | |||
490 | } | ||
491 | |||
492 | efuse_power_switch(hw, true, false); | ||
493 | efuse_read_all_map(hw, &rtlefuse->efuse_map[EFUSE_INIT_MAP][0]); | ||
494 | |||
495 | memcpy(&rtlefuse->efuse_map[EFUSE_MODIFY_MAP][0], | ||
496 | &rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
497 | rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE]); | ||
498 | |||
499 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, ("<---\n")); | ||
500 | return true; | ||
501 | } | ||
502 | |||
503 | void rtl_efuse_shadow_map_update(struct ieee80211_hw *hw) | ||
504 | { | ||
505 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
506 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
507 | |||
508 | if (rtlefuse->autoload_failflag == true) | ||
509 | memset(&rtlefuse->efuse_map[EFUSE_INIT_MAP][0], 0xFF, | ||
510 | rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE]); | ||
511 | else | ||
512 | efuse_read_all_map(hw, &rtlefuse->efuse_map[EFUSE_INIT_MAP][0]); | ||
513 | |||
514 | memcpy(&rtlefuse->efuse_map[EFUSE_MODIFY_MAP][0], | ||
515 | &rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
516 | rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE]); | ||
517 | |||
518 | } | ||
519 | EXPORT_SYMBOL(rtl_efuse_shadow_map_update); | ||
520 | |||
521 | void efuse_force_write_vendor_Id(struct ieee80211_hw *hw) | ||
522 | { | ||
523 | u8 tmpdata[8] = { 0xFF, 0xFF, 0xEC, 0x10, 0xFF, 0xFF, 0xFF, 0xFF }; | ||
524 | |||
525 | efuse_power_switch(hw, true, true); | ||
526 | |||
527 | efuse_pg_packet_write(hw, 1, 0xD, tmpdata); | ||
528 | |||
529 | efuse_power_switch(hw, true, false); | ||
530 | |||
531 | } | ||
532 | |||
533 | void efuse_re_pg_section(struct ieee80211_hw *hw, u8 section_idx) | ||
534 | { | ||
535 | } | ||
536 | |||
537 | static void efuse_shadow_read_1byte(struct ieee80211_hw *hw, | ||
538 | u16 offset, u8 *value) | ||
539 | { | ||
540 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
541 | *value = rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset]; | ||
542 | } | ||
543 | |||
544 | static void efuse_shadow_read_2byte(struct ieee80211_hw *hw, | ||
545 | u16 offset, u16 *value) | ||
546 | { | ||
547 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
548 | |||
549 | *value = rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset]; | ||
550 | *value |= rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 1] << 8; | ||
551 | |||
552 | } | ||
553 | |||
554 | static void efuse_shadow_read_4byte(struct ieee80211_hw *hw, | ||
555 | u16 offset, u32 *value) | ||
556 | { | ||
557 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
558 | |||
559 | *value = rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset]; | ||
560 | *value |= rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 1] << 8; | ||
561 | *value |= rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 2] << 16; | ||
562 | *value |= rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 3] << 24; | ||
563 | } | ||
564 | |||
565 | static void efuse_shadow_write_1byte(struct ieee80211_hw *hw, | ||
566 | u16 offset, u8 value) | ||
567 | { | ||
568 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
569 | |||
570 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset] = value; | ||
571 | } | ||
572 | |||
573 | static void efuse_shadow_write_2byte(struct ieee80211_hw *hw, | ||
574 | u16 offset, u16 value) | ||
575 | { | ||
576 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
577 | |||
578 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset] = value & 0x00FF; | ||
579 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 1] = value >> 8; | ||
580 | |||
581 | } | ||
582 | |||
583 | static void efuse_shadow_write_4byte(struct ieee80211_hw *hw, | ||
584 | u16 offset, u32 value) | ||
585 | { | ||
586 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
587 | |||
588 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset] = | ||
589 | (u8) (value & 0x000000FF); | ||
590 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 1] = | ||
591 | (u8) ((value >> 8) & 0x0000FF); | ||
592 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 2] = | ||
593 | (u8) ((value >> 16) & 0x00FF); | ||
594 | rtlefuse->efuse_map[EFUSE_MODIFY_MAP][offset + 3] = | ||
595 | (u8) ((value >> 24) & 0xFF); | ||
596 | |||
597 | } | ||
598 | |||
599 | static int efuse_one_byte_read(struct ieee80211_hw *hw, u16 addr, u8 *data) | ||
600 | { | ||
601 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
602 | u8 tmpidx = 0; | ||
603 | int result; | ||
604 | |||
605 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 1, | ||
606 | (u8) (addr & 0xff)); | ||
607 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 2, | ||
608 | ((u8) ((addr >> 8) & 0x03)) | | ||
609 | (rtl_read_byte(rtlpriv, | ||
610 | rtlpriv->cfg->maps[EFUSE_CTRL] + 2) & | ||
611 | 0xFC)); | ||
612 | |||
613 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3, 0x72); | ||
614 | |||
615 | while (!(0x80 & rtl_read_byte(rtlpriv, | ||
616 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3)) | ||
617 | && (tmpidx < 100)) { | ||
618 | tmpidx++; | ||
619 | } | ||
620 | |||
621 | if (tmpidx < 100) { | ||
622 | *data = rtl_read_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL]); | ||
623 | result = true; | ||
624 | } else { | ||
625 | *data = 0xff; | ||
626 | result = false; | ||
627 | } | ||
628 | return result; | ||
629 | } | ||
630 | |||
631 | static int efuse_one_byte_write(struct ieee80211_hw *hw, u16 addr, u8 data) | ||
632 | { | ||
633 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
634 | u8 tmpidx = 0; | ||
635 | |||
636 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
637 | ("Addr = %x Data=%x\n", addr, data)); | ||
638 | |||
639 | rtl_write_byte(rtlpriv, | ||
640 | rtlpriv->cfg->maps[EFUSE_CTRL] + 1, (u8) (addr & 0xff)); | ||
641 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 2, | ||
642 | (rtl_read_byte(rtlpriv, | ||
643 | rtlpriv->cfg->maps[EFUSE_CTRL] + | ||
644 | 2) & 0xFC) | (u8) ((addr >> 8) & 0x03)); | ||
645 | |||
646 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL], data); | ||
647 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CTRL] + 3, 0xF2); | ||
648 | |||
649 | while ((0x80 & rtl_read_byte(rtlpriv, | ||
650 | rtlpriv->cfg->maps[EFUSE_CTRL] + 3)) | ||
651 | && (tmpidx < 100)) { | ||
652 | tmpidx++; | ||
653 | } | ||
654 | |||
655 | if (tmpidx < 100) | ||
656 | return true; | ||
657 | |||
658 | return false; | ||
659 | } | ||
660 | |||
661 | static void efuse_read_all_map(struct ieee80211_hw *hw, u8 * efuse) | ||
662 | { | ||
663 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
664 | efuse_power_switch(hw, false, true); | ||
665 | read_efuse(hw, 0, rtlpriv->cfg->maps[EFUSE_HWSET_MAX_SIZE], efuse); | ||
666 | efuse_power_switch(hw, false, false); | ||
667 | } | ||
668 | |||
669 | static void efuse_read_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | ||
670 | u8 efuse_data, u8 offset, u8 *tmpdata, | ||
671 | u8 *readstate) | ||
672 | { | ||
673 | bool dataempty = true; | ||
674 | u8 hoffset; | ||
675 | u8 tmpidx; | ||
676 | u8 hworden; | ||
677 | u8 word_cnts; | ||
678 | |||
679 | hoffset = (efuse_data >> 4) & 0x0F; | ||
680 | hworden = efuse_data & 0x0F; | ||
681 | word_cnts = efuse_calculate_word_cnts(hworden); | ||
682 | |||
683 | if (hoffset == offset) { | ||
684 | for (tmpidx = 0; tmpidx < word_cnts * 2; tmpidx++) { | ||
685 | if (efuse_one_byte_read(hw, *efuse_addr + 1 + tmpidx, | ||
686 | &efuse_data)) { | ||
687 | tmpdata[tmpidx] = efuse_data; | ||
688 | if (efuse_data != 0xff) | ||
689 | dataempty = true; | ||
690 | } | ||
691 | } | ||
692 | |||
693 | if (dataempty == true) { | ||
694 | *readstate = PG_STATE_DATA; | ||
695 | } else { | ||
696 | *efuse_addr = *efuse_addr + (word_cnts * 2) + 1; | ||
697 | *readstate = PG_STATE_HEADER; | ||
698 | } | ||
699 | |||
700 | } else { | ||
701 | *efuse_addr = *efuse_addr + (word_cnts * 2) + 1; | ||
702 | *readstate = PG_STATE_HEADER; | ||
703 | } | ||
704 | } | ||
705 | |||
706 | static int efuse_pg_packet_read(struct ieee80211_hw *hw, u8 offset, u8 *data) | ||
707 | { | ||
708 | u8 readstate = PG_STATE_HEADER; | ||
709 | bool continual = true; | ||
710 | u8 efuse_data, word_cnts = 0; | ||
711 | u16 efuse_addr = 0; | ||
712 | u8 tmpdata[8]; | ||
713 | |||
714 | if (data == NULL) | ||
715 | return false; | ||
716 | if (offset > 15) | ||
717 | return false; | ||
718 | |||
719 | memset(data, 0xff, PGPKT_DATA_SIZE * sizeof(u8)); | ||
720 | memset(tmpdata, 0xff, PGPKT_DATA_SIZE * sizeof(u8)); | ||
721 | |||
722 | while (continual && (efuse_addr < EFUSE_MAX_SIZE)) { | ||
723 | if (readstate & PG_STATE_HEADER) { | ||
724 | if (efuse_one_byte_read(hw, efuse_addr, &efuse_data) | ||
725 | && (efuse_data != 0xFF)) | ||
726 | efuse_read_data_case1(hw, &efuse_addr, | ||
727 | efuse_data, | ||
728 | offset, tmpdata, | ||
729 | &readstate); | ||
730 | else | ||
731 | continual = false; | ||
732 | } else if (readstate & PG_STATE_DATA) { | ||
733 | efuse_word_enable_data_read(0, tmpdata, data); | ||
734 | efuse_addr = efuse_addr + (word_cnts * 2) + 1; | ||
735 | readstate = PG_STATE_HEADER; | ||
736 | } | ||
737 | |||
738 | } | ||
739 | |||
740 | if ((data[0] == 0xff) && (data[1] == 0xff) && | ||
741 | (data[2] == 0xff) && (data[3] == 0xff) && | ||
742 | (data[4] == 0xff) && (data[5] == 0xff) && | ||
743 | (data[6] == 0xff) && (data[7] == 0xff)) | ||
744 | return false; | ||
745 | else | ||
746 | return true; | ||
747 | |||
748 | } | ||
749 | |||
750 | static void efuse_write_data_case1(struct ieee80211_hw *hw, u16 *efuse_addr, | ||
751 | u8 efuse_data, u8 offset, int *continual, | ||
752 | u8 *write_state, struct pgpkt_struct *target_pkt, | ||
753 | int *repeat_times, int *result, u8 word_en) | ||
754 | { | ||
755 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
756 | struct pgpkt_struct tmp_pkt; | ||
757 | bool dataempty = true; | ||
758 | u8 originaldata[8 * sizeof(u8)]; | ||
759 | u8 badworden = 0x0F; | ||
760 | u8 match_word_en, tmp_word_en; | ||
761 | u8 tmpindex; | ||
762 | u8 tmp_header = efuse_data; | ||
763 | u8 tmp_word_cnts; | ||
764 | |||
765 | tmp_pkt.offset = (tmp_header >> 4) & 0x0F; | ||
766 | tmp_pkt.word_en = tmp_header & 0x0F; | ||
767 | tmp_word_cnts = efuse_calculate_word_cnts(tmp_pkt.word_en); | ||
768 | |||
769 | if (tmp_pkt.offset != target_pkt->offset) { | ||
770 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) + 1; | ||
771 | *write_state = PG_STATE_HEADER; | ||
772 | } else { | ||
773 | for (tmpindex = 0; tmpindex < (tmp_word_cnts * 2); tmpindex++) { | ||
774 | u16 address = *efuse_addr + 1 + tmpindex; | ||
775 | if (efuse_one_byte_read(hw, address, | ||
776 | &efuse_data) && (efuse_data != 0xFF)) | ||
777 | dataempty = false; | ||
778 | } | ||
779 | |||
780 | if (dataempty == false) { | ||
781 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) + 1; | ||
782 | *write_state = PG_STATE_HEADER; | ||
783 | } else { | ||
784 | match_word_en = 0x0F; | ||
785 | if (!((target_pkt->word_en & BIT(0)) | | ||
786 | (tmp_pkt.word_en & BIT(0)))) | ||
787 | match_word_en &= (~BIT(0)); | ||
788 | |||
789 | if (!((target_pkt->word_en & BIT(1)) | | ||
790 | (tmp_pkt.word_en & BIT(1)))) | ||
791 | match_word_en &= (~BIT(1)); | ||
792 | |||
793 | if (!((target_pkt->word_en & BIT(2)) | | ||
794 | (tmp_pkt.word_en & BIT(2)))) | ||
795 | match_word_en &= (~BIT(2)); | ||
796 | |||
797 | if (!((target_pkt->word_en & BIT(3)) | | ||
798 | (tmp_pkt.word_en & BIT(3)))) | ||
799 | match_word_en &= (~BIT(3)); | ||
800 | |||
801 | if ((match_word_en & 0x0F) != 0x0F) { | ||
802 | badworden = efuse_word_enable_data_write( | ||
803 | hw, *efuse_addr + 1, | ||
804 | tmp_pkt.word_en, | ||
805 | target_pkt->data); | ||
806 | |||
807 | if (0x0F != (badworden & 0x0F)) { | ||
808 | u8 reorg_offset = offset; | ||
809 | u8 reorg_worden = badworden; | ||
810 | efuse_pg_packet_write(hw, reorg_offset, | ||
811 | reorg_worden, | ||
812 | originaldata); | ||
813 | } | ||
814 | |||
815 | tmp_word_en = 0x0F; | ||
816 | if ((target_pkt->word_en & BIT(0)) ^ | ||
817 | (match_word_en & BIT(0))) | ||
818 | tmp_word_en &= (~BIT(0)); | ||
819 | |||
820 | if ((target_pkt->word_en & BIT(1)) ^ | ||
821 | (match_word_en & BIT(1))) | ||
822 | tmp_word_en &= (~BIT(1)); | ||
823 | |||
824 | if ((target_pkt->word_en & BIT(2)) ^ | ||
825 | (match_word_en & BIT(2))) | ||
826 | tmp_word_en &= (~BIT(2)); | ||
827 | |||
828 | if ((target_pkt->word_en & BIT(3)) ^ | ||
829 | (match_word_en & BIT(3))) | ||
830 | tmp_word_en &= (~BIT(3)); | ||
831 | |||
832 | if ((tmp_word_en & 0x0F) != 0x0F) { | ||
833 | *efuse_addr = efuse_get_current_size(hw); | ||
834 | target_pkt->offset = offset; | ||
835 | target_pkt->word_en = tmp_word_en; | ||
836 | } else { | ||
837 | *continual = false; | ||
838 | } | ||
839 | *write_state = PG_STATE_HEADER; | ||
840 | *repeat_times += 1; | ||
841 | if (*repeat_times > EFUSE_REPEAT_THRESHOLD_) { | ||
842 | *continual = false; | ||
843 | *result = false; | ||
844 | } | ||
845 | } else { | ||
846 | *efuse_addr += (2 * tmp_word_cnts) + 1; | ||
847 | target_pkt->offset = offset; | ||
848 | target_pkt->word_en = word_en; | ||
849 | *write_state = PG_STATE_HEADER; | ||
850 | } | ||
851 | } | ||
852 | } | ||
853 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, ("efuse PG_STATE_HEADER-1\n")); | ||
854 | } | ||
855 | |||
856 | static void efuse_write_data_case2(struct ieee80211_hw *hw, u16 *efuse_addr, | ||
857 | int *continual, u8 *write_state, | ||
858 | struct pgpkt_struct target_pkt, | ||
859 | int *repeat_times, int *result) | ||
860 | { | ||
861 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
862 | struct pgpkt_struct tmp_pkt; | ||
863 | u8 pg_header; | ||
864 | u8 tmp_header; | ||
865 | u8 originaldata[8 * sizeof(u8)]; | ||
866 | u8 tmp_word_cnts; | ||
867 | u8 badworden = 0x0F; | ||
868 | |||
869 | pg_header = ((target_pkt.offset << 4) & 0xf0) | target_pkt.word_en; | ||
870 | efuse_one_byte_write(hw, *efuse_addr, pg_header); | ||
871 | efuse_one_byte_read(hw, *efuse_addr, &tmp_header); | ||
872 | |||
873 | if (tmp_header == pg_header) { | ||
874 | *write_state = PG_STATE_DATA; | ||
875 | } else if (tmp_header == 0xFF) { | ||
876 | *write_state = PG_STATE_HEADER; | ||
877 | *repeat_times += 1; | ||
878 | if (*repeat_times > EFUSE_REPEAT_THRESHOLD_) { | ||
879 | *continual = false; | ||
880 | *result = false; | ||
881 | } | ||
882 | } else { | ||
883 | tmp_pkt.offset = (tmp_header >> 4) & 0x0F; | ||
884 | tmp_pkt.word_en = tmp_header & 0x0F; | ||
885 | |||
886 | tmp_word_cnts = efuse_calculate_word_cnts(tmp_pkt.word_en); | ||
887 | |||
888 | memset(originaldata, 0xff, 8 * sizeof(u8)); | ||
889 | |||
890 | if (efuse_pg_packet_read(hw, tmp_pkt.offset, originaldata)) { | ||
891 | badworden = efuse_word_enable_data_write(hw, | ||
892 | *efuse_addr + 1, tmp_pkt.word_en, | ||
893 | originaldata); | ||
894 | |||
895 | if (0x0F != (badworden & 0x0F)) { | ||
896 | u8 reorg_offset = tmp_pkt.offset; | ||
897 | u8 reorg_worden = badworden; | ||
898 | efuse_pg_packet_write(hw, reorg_offset, | ||
899 | reorg_worden, | ||
900 | originaldata); | ||
901 | *efuse_addr = efuse_get_current_size(hw); | ||
902 | } else { | ||
903 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) | ||
904 | + 1; | ||
905 | } | ||
906 | } else { | ||
907 | *efuse_addr = *efuse_addr + (tmp_word_cnts * 2) + 1; | ||
908 | } | ||
909 | |||
910 | *write_state = PG_STATE_HEADER; | ||
911 | *repeat_times += 1; | ||
912 | if (*repeat_times > EFUSE_REPEAT_THRESHOLD_) { | ||
913 | *continual = false; | ||
914 | *result = false; | ||
915 | } | ||
916 | |||
917 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, | ||
918 | ("efuse PG_STATE_HEADER-2\n")); | ||
919 | } | ||
920 | } | ||
921 | |||
922 | static int efuse_pg_packet_write(struct ieee80211_hw *hw, | ||
923 | u8 offset, u8 word_en, u8 *data) | ||
924 | { | ||
925 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
926 | struct pgpkt_struct target_pkt; | ||
927 | u8 write_state = PG_STATE_HEADER; | ||
928 | int continual = true, dataempty = true, result = true; | ||
929 | u16 efuse_addr = 0; | ||
930 | u8 efuse_data; | ||
931 | u8 target_word_cnts = 0; | ||
932 | u8 badworden = 0x0F; | ||
933 | static int repeat_times; | ||
934 | |||
935 | if (efuse_get_current_size(hw) >= | ||
936 | (EFUSE_MAX_SIZE - EFUSE_OOB_PROTECT_BYTES)) { | ||
937 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, | ||
938 | ("efuse_pg_packet_write error\n")); | ||
939 | return false; | ||
940 | } | ||
941 | |||
942 | target_pkt.offset = offset; | ||
943 | target_pkt.word_en = word_en; | ||
944 | |||
945 | memset(target_pkt.data, 0xFF, 8 * sizeof(u8)); | ||
946 | |||
947 | efuse_word_enable_data_read(word_en, data, target_pkt.data); | ||
948 | target_word_cnts = efuse_calculate_word_cnts(target_pkt.word_en); | ||
949 | |||
950 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, ("efuse Power ON\n")); | ||
951 | |||
952 | while (continual && (efuse_addr < | ||
953 | (EFUSE_MAX_SIZE - EFUSE_OOB_PROTECT_BYTES))) { | ||
954 | |||
955 | if (write_state == PG_STATE_HEADER) { | ||
956 | dataempty = true; | ||
957 | badworden = 0x0F; | ||
958 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, | ||
959 | ("efuse PG_STATE_HEADER\n")); | ||
960 | |||
961 | if (efuse_one_byte_read(hw, efuse_addr, &efuse_data) && | ||
962 | (efuse_data != 0xFF)) | ||
963 | efuse_write_data_case1(hw, &efuse_addr, | ||
964 | efuse_data, offset, | ||
965 | &continual, | ||
966 | &write_state, &target_pkt, | ||
967 | &repeat_times, &result, | ||
968 | word_en); | ||
969 | else | ||
970 | efuse_write_data_case2(hw, &efuse_addr, | ||
971 | &continual, | ||
972 | &write_state, | ||
973 | target_pkt, | ||
974 | &repeat_times, | ||
975 | &result); | ||
976 | |||
977 | } else if (write_state == PG_STATE_DATA) { | ||
978 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, | ||
979 | ("efuse PG_STATE_DATA\n")); | ||
980 | badworden = | ||
981 | efuse_word_enable_data_write(hw, efuse_addr + 1, | ||
982 | target_pkt.word_en, | ||
983 | target_pkt.data); | ||
984 | |||
985 | if ((badworden & 0x0F) == 0x0F) { | ||
986 | continual = false; | ||
987 | } else { | ||
988 | efuse_addr += (2 * target_word_cnts) + 1; | ||
989 | |||
990 | target_pkt.offset = offset; | ||
991 | target_pkt.word_en = badworden; | ||
992 | target_word_cnts = | ||
993 | efuse_calculate_word_cnts(target_pkt. | ||
994 | word_en); | ||
995 | write_state = PG_STATE_HEADER; | ||
996 | repeat_times++; | ||
997 | if (repeat_times > EFUSE_REPEAT_THRESHOLD_) { | ||
998 | continual = false; | ||
999 | result = false; | ||
1000 | } | ||
1001 | RTPRINT(rtlpriv, FEEPROM, EFUSE_PG, | ||
1002 | ("efuse PG_STATE_HEADER-3\n")); | ||
1003 | } | ||
1004 | } | ||
1005 | } | ||
1006 | |||
1007 | if (efuse_addr >= (EFUSE_MAX_SIZE - EFUSE_OOB_PROTECT_BYTES)) { | ||
1008 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
1009 | ("efuse_addr(%#x) Out of size!!\n", efuse_addr)); | ||
1010 | } | ||
1011 | |||
1012 | return true; | ||
1013 | } | ||
1014 | |||
1015 | static void efuse_word_enable_data_read(u8 word_en, | ||
1016 | u8 *sourdata, u8 *targetdata) | ||
1017 | { | ||
1018 | if (!(word_en & BIT(0))) { | ||
1019 | targetdata[0] = sourdata[0]; | ||
1020 | targetdata[1] = sourdata[1]; | ||
1021 | } | ||
1022 | |||
1023 | if (!(word_en & BIT(1))) { | ||
1024 | targetdata[2] = sourdata[2]; | ||
1025 | targetdata[3] = sourdata[3]; | ||
1026 | } | ||
1027 | |||
1028 | if (!(word_en & BIT(2))) { | ||
1029 | targetdata[4] = sourdata[4]; | ||
1030 | targetdata[5] = sourdata[5]; | ||
1031 | } | ||
1032 | |||
1033 | if (!(word_en & BIT(3))) { | ||
1034 | targetdata[6] = sourdata[6]; | ||
1035 | targetdata[7] = sourdata[7]; | ||
1036 | } | ||
1037 | } | ||
1038 | |||
1039 | static u8 efuse_word_enable_data_write(struct ieee80211_hw *hw, | ||
1040 | u16 efuse_addr, u8 word_en, u8 *data) | ||
1041 | { | ||
1042 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1043 | u16 tmpaddr; | ||
1044 | u16 start_addr = efuse_addr; | ||
1045 | u8 badworden = 0x0F; | ||
1046 | u8 tmpdata[8]; | ||
1047 | |||
1048 | memset(tmpdata, 0xff, PGPKT_DATA_SIZE); | ||
1049 | RT_TRACE(rtlpriv, COMP_EFUSE, DBG_LOUD, | ||
1050 | ("word_en = %x efuse_addr=%x\n", word_en, efuse_addr)); | ||
1051 | |||
1052 | if (!(word_en & BIT(0))) { | ||
1053 | tmpaddr = start_addr; | ||
1054 | efuse_one_byte_write(hw, start_addr++, data[0]); | ||
1055 | efuse_one_byte_write(hw, start_addr++, data[1]); | ||
1056 | |||
1057 | efuse_one_byte_read(hw, tmpaddr, &tmpdata[0]); | ||
1058 | efuse_one_byte_read(hw, tmpaddr + 1, &tmpdata[1]); | ||
1059 | if ((data[0] != tmpdata[0]) || (data[1] != tmpdata[1])) | ||
1060 | badworden &= (~BIT(0)); | ||
1061 | } | ||
1062 | |||
1063 | if (!(word_en & BIT(1))) { | ||
1064 | tmpaddr = start_addr; | ||
1065 | efuse_one_byte_write(hw, start_addr++, data[2]); | ||
1066 | efuse_one_byte_write(hw, start_addr++, data[3]); | ||
1067 | |||
1068 | efuse_one_byte_read(hw, tmpaddr, &tmpdata[2]); | ||
1069 | efuse_one_byte_read(hw, tmpaddr + 1, &tmpdata[3]); | ||
1070 | if ((data[2] != tmpdata[2]) || (data[3] != tmpdata[3])) | ||
1071 | badworden &= (~BIT(1)); | ||
1072 | } | ||
1073 | |||
1074 | if (!(word_en & BIT(2))) { | ||
1075 | tmpaddr = start_addr; | ||
1076 | efuse_one_byte_write(hw, start_addr++, data[4]); | ||
1077 | efuse_one_byte_write(hw, start_addr++, data[5]); | ||
1078 | |||
1079 | efuse_one_byte_read(hw, tmpaddr, &tmpdata[4]); | ||
1080 | efuse_one_byte_read(hw, tmpaddr + 1, &tmpdata[5]); | ||
1081 | if ((data[4] != tmpdata[4]) || (data[5] != tmpdata[5])) | ||
1082 | badworden &= (~BIT(2)); | ||
1083 | } | ||
1084 | |||
1085 | if (!(word_en & BIT(3))) { | ||
1086 | tmpaddr = start_addr; | ||
1087 | efuse_one_byte_write(hw, start_addr++, data[6]); | ||
1088 | efuse_one_byte_write(hw, start_addr++, data[7]); | ||
1089 | |||
1090 | efuse_one_byte_read(hw, tmpaddr, &tmpdata[6]); | ||
1091 | efuse_one_byte_read(hw, tmpaddr + 1, &tmpdata[7]); | ||
1092 | if ((data[6] != tmpdata[6]) || (data[7] != tmpdata[7])) | ||
1093 | badworden &= (~BIT(3)); | ||
1094 | } | ||
1095 | |||
1096 | return badworden; | ||
1097 | } | ||
1098 | |||
1099 | static void efuse_power_switch(struct ieee80211_hw *hw, u8 write, u8 pwrstate) | ||
1100 | { | ||
1101 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1102 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1103 | u8 tempval; | ||
1104 | u16 tmpV16; | ||
1105 | |||
1106 | if (pwrstate && (rtlhal->hw_type != | ||
1107 | HARDWARE_TYPE_RTL8192SE)) { | ||
1108 | tmpV16 = rtl_read_word(rtlpriv, | ||
1109 | rtlpriv->cfg->maps[SYS_ISO_CTRL]); | ||
1110 | if (!(tmpV16 & rtlpriv->cfg->maps[EFUSE_PWC_EV12V])) { | ||
1111 | tmpV16 |= rtlpriv->cfg->maps[EFUSE_PWC_EV12V]; | ||
1112 | rtl_write_word(rtlpriv, | ||
1113 | rtlpriv->cfg->maps[SYS_ISO_CTRL], | ||
1114 | tmpV16); | ||
1115 | } | ||
1116 | |||
1117 | tmpV16 = rtl_read_word(rtlpriv, | ||
1118 | rtlpriv->cfg->maps[SYS_FUNC_EN]); | ||
1119 | if (!(tmpV16 & rtlpriv->cfg->maps[EFUSE_FEN_ELDR])) { | ||
1120 | tmpV16 |= rtlpriv->cfg->maps[EFUSE_FEN_ELDR]; | ||
1121 | rtl_write_word(rtlpriv, | ||
1122 | rtlpriv->cfg->maps[SYS_FUNC_EN], tmpV16); | ||
1123 | } | ||
1124 | |||
1125 | tmpV16 = rtl_read_word(rtlpriv, rtlpriv->cfg->maps[SYS_CLK]); | ||
1126 | if ((!(tmpV16 & rtlpriv->cfg->maps[EFUSE_LOADER_CLK_EN])) || | ||
1127 | (!(tmpV16 & rtlpriv->cfg->maps[EFUSE_ANA8M]))) { | ||
1128 | tmpV16 |= (rtlpriv->cfg->maps[EFUSE_LOADER_CLK_EN] | | ||
1129 | rtlpriv->cfg->maps[EFUSE_ANA8M]); | ||
1130 | rtl_write_word(rtlpriv, | ||
1131 | rtlpriv->cfg->maps[SYS_CLK], tmpV16); | ||
1132 | } | ||
1133 | } | ||
1134 | |||
1135 | if (pwrstate) { | ||
1136 | if (write) { | ||
1137 | tempval = rtl_read_byte(rtlpriv, | ||
1138 | rtlpriv->cfg->maps[EFUSE_TEST] + | ||
1139 | 3); | ||
1140 | |||
1141 | if (rtlhal->hw_type != HARDWARE_TYPE_RTL8192SE) { | ||
1142 | tempval &= 0x0F; | ||
1143 | tempval |= (VOLTAGE_V25 << 4); | ||
1144 | } | ||
1145 | |||
1146 | rtl_write_byte(rtlpriv, | ||
1147 | rtlpriv->cfg->maps[EFUSE_TEST] + 3, | ||
1148 | (tempval | 0x80)); | ||
1149 | } | ||
1150 | |||
1151 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) { | ||
1152 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CLK], | ||
1153 | 0x03); | ||
1154 | } | ||
1155 | |||
1156 | } else { | ||
1157 | if (write) { | ||
1158 | tempval = rtl_read_byte(rtlpriv, | ||
1159 | rtlpriv->cfg->maps[EFUSE_TEST] + | ||
1160 | 3); | ||
1161 | rtl_write_byte(rtlpriv, | ||
1162 | rtlpriv->cfg->maps[EFUSE_TEST] + 3, | ||
1163 | (tempval & 0x7F)); | ||
1164 | } | ||
1165 | |||
1166 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) { | ||
1167 | rtl_write_byte(rtlpriv, rtlpriv->cfg->maps[EFUSE_CLK], | ||
1168 | 0x02); | ||
1169 | } | ||
1170 | |||
1171 | } | ||
1172 | |||
1173 | } | ||
1174 | |||
1175 | static u16 efuse_get_current_size(struct ieee80211_hw *hw) | ||
1176 | { | ||
1177 | int continual = true; | ||
1178 | u16 efuse_addr = 0; | ||
1179 | u8 hoffset, hworden; | ||
1180 | u8 efuse_data, word_cnts; | ||
1181 | |||
1182 | while (continual && efuse_one_byte_read(hw, efuse_addr, &efuse_data) | ||
1183 | && (efuse_addr < EFUSE_MAX_SIZE)) { | ||
1184 | if (efuse_data != 0xFF) { | ||
1185 | hoffset = (efuse_data >> 4) & 0x0F; | ||
1186 | hworden = efuse_data & 0x0F; | ||
1187 | word_cnts = efuse_calculate_word_cnts(hworden); | ||
1188 | efuse_addr = efuse_addr + (word_cnts * 2) + 1; | ||
1189 | } else { | ||
1190 | continual = false; | ||
1191 | } | ||
1192 | } | ||
1193 | |||
1194 | return efuse_addr; | ||
1195 | } | ||
1196 | |||
1197 | static u8 efuse_calculate_word_cnts(u8 word_en) | ||
1198 | { | ||
1199 | u8 word_cnts = 0; | ||
1200 | if (!(word_en & BIT(0))) | ||
1201 | word_cnts++; | ||
1202 | if (!(word_en & BIT(1))) | ||
1203 | word_cnts++; | ||
1204 | if (!(word_en & BIT(2))) | ||
1205 | word_cnts++; | ||
1206 | if (!(word_en & BIT(3))) | ||
1207 | word_cnts++; | ||
1208 | return word_cnts; | ||
1209 | } | ||
1210 | |||
diff --git a/drivers/net/wireless/rtlwifi/efuse.h b/drivers/net/wireless/rtlwifi/efuse.h new file mode 100644 index 000000000000..164dabaa7615 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/efuse.h | |||
@@ -0,0 +1,124 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_EFUSE_H_ | ||
31 | #define __RTL_EFUSE_H_ | ||
32 | |||
33 | #define EFUSE_IC_ID_OFFSET 506 | ||
34 | |||
35 | #define EFUSE_REAL_CONTENT_LEN 512 | ||
36 | #define EFUSE_MAP_LEN 128 | ||
37 | #define EFUSE_MAX_WORD_UNIT 4 | ||
38 | |||
39 | #define EFUSE_INIT_MAP 0 | ||
40 | #define EFUSE_MODIFY_MAP 1 | ||
41 | |||
42 | #define PG_STATE_HEADER 0x01 | ||
43 | #define PG_STATE_WORD_0 0x02 | ||
44 | #define PG_STATE_WORD_1 0x04 | ||
45 | #define PG_STATE_WORD_2 0x08 | ||
46 | #define PG_STATE_WORD_3 0x10 | ||
47 | #define PG_STATE_DATA 0x20 | ||
48 | |||
49 | #define PG_SWBYTE_H 0x01 | ||
50 | #define PG_SWBYTE_L 0x02 | ||
51 | |||
52 | #define _POWERON_DELAY_ | ||
53 | #define _PRE_EXECUTE_READ_CMD_ | ||
54 | |||
55 | #define EFUSE_REPEAT_THRESHOLD_ 3 | ||
56 | #define EFUSE_ERROE_HANDLE 1 | ||
57 | |||
58 | struct efuse_map { | ||
59 | u8 offset; | ||
60 | u8 word_start; | ||
61 | u8 byte_start; | ||
62 | u8 byte_cnts; | ||
63 | }; | ||
64 | |||
65 | struct pgpkt_struct { | ||
66 | u8 offset; | ||
67 | u8 word_en; | ||
68 | u8 data[8]; | ||
69 | }; | ||
70 | |||
71 | enum efuse_data_item { | ||
72 | EFUSE_CHIP_ID = 0, | ||
73 | EFUSE_LDO_SETTING, | ||
74 | EFUSE_CLK_SETTING, | ||
75 | EFUSE_SDIO_SETTING, | ||
76 | EFUSE_CCCR, | ||
77 | EFUSE_SDIO_MODE, | ||
78 | EFUSE_OCR, | ||
79 | EFUSE_F0CIS, | ||
80 | EFUSE_F1CIS, | ||
81 | EFUSE_MAC_ADDR, | ||
82 | EFUSE_EEPROM_VER, | ||
83 | EFUSE_CHAN_PLAN, | ||
84 | EFUSE_TXPW_TAB | ||
85 | }; | ||
86 | |||
87 | enum { | ||
88 | VOLTAGE_V25 = 0x03, | ||
89 | LDOE25_SHIFT = 28, | ||
90 | }; | ||
91 | |||
92 | struct efuse_priv { | ||
93 | u8 id[2]; | ||
94 | u8 ldo_setting[2]; | ||
95 | u8 clk_setting[2]; | ||
96 | u8 cccr; | ||
97 | u8 sdio_mode; | ||
98 | u8 ocr[3]; | ||
99 | u8 cis0[17]; | ||
100 | u8 cis1[48]; | ||
101 | u8 mac_addr[6]; | ||
102 | u8 eeprom_verno; | ||
103 | u8 channel_plan; | ||
104 | u8 tx_power_b[14]; | ||
105 | u8 tx_power_g[14]; | ||
106 | }; | ||
107 | |||
108 | extern void read_efuse_byte(struct ieee80211_hw *hw, u16 _offset, u8 *pbuf); | ||
109 | extern void efuse_initialize(struct ieee80211_hw *hw); | ||
110 | extern u8 efuse_read_1byte(struct ieee80211_hw *hw, u16 address); | ||
111 | extern void efuse_write_1byte(struct ieee80211_hw *hw, u16 address, u8 value); | ||
112 | extern void read_efuse(struct ieee80211_hw *hw, u16 _offset, | ||
113 | u16 _size_byte, u8 *pbuf); | ||
114 | extern void efuse_shadow_read(struct ieee80211_hw *hw, u8 type, | ||
115 | u16 offset, u32 *value); | ||
116 | extern void efuse_shadow_write(struct ieee80211_hw *hw, u8 type, | ||
117 | u16 offset, u32 value); | ||
118 | extern bool efuse_shadow_update(struct ieee80211_hw *hw); | ||
119 | extern bool efuse_shadow_update_chk(struct ieee80211_hw *hw); | ||
120 | extern void rtl_efuse_shadow_map_update(struct ieee80211_hw *hw); | ||
121 | extern void efuse_force_write_vendor_Id(struct ieee80211_hw *hw); | ||
122 | extern void efuse_re_pg_section(struct ieee80211_hw *hw, u8 section_idx); | ||
123 | |||
124 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/pci.c b/drivers/net/wireless/rtlwifi/pci.c new file mode 100644 index 000000000000..254b64ba4bf6 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/pci.c | |||
@@ -0,0 +1,2071 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "core.h" | ||
31 | #include "wifi.h" | ||
32 | #include "pci.h" | ||
33 | #include "base.h" | ||
34 | #include "ps.h" | ||
35 | #include "efuse.h" | ||
36 | |||
37 | static const u16 pcibridge_vendors[PCI_BRIDGE_VENDOR_MAX] = { | ||
38 | INTEL_VENDOR_ID, | ||
39 | ATI_VENDOR_ID, | ||
40 | AMD_VENDOR_ID, | ||
41 | SIS_VENDOR_ID | ||
42 | }; | ||
43 | |||
44 | static const u8 ac_to_hwq[] = { | ||
45 | VO_QUEUE, | ||
46 | VI_QUEUE, | ||
47 | BE_QUEUE, | ||
48 | BK_QUEUE | ||
49 | }; | ||
50 | |||
51 | static u8 _rtl_mac_to_hwqueue(struct ieee80211_hw *hw, | ||
52 | struct sk_buff *skb) | ||
53 | { | ||
54 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
55 | __le16 fc = rtl_get_fc(skb); | ||
56 | u8 queue_index = skb_get_queue_mapping(skb); | ||
57 | |||
58 | if (unlikely(ieee80211_is_beacon(fc))) | ||
59 | return BEACON_QUEUE; | ||
60 | if (ieee80211_is_mgmt(fc)) | ||
61 | return MGNT_QUEUE; | ||
62 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) | ||
63 | if (ieee80211_is_nullfunc(fc)) | ||
64 | return HIGH_QUEUE; | ||
65 | |||
66 | return ac_to_hwq[queue_index]; | ||
67 | } | ||
68 | |||
69 | /* Update PCI dependent default settings*/ | ||
70 | static void _rtl_pci_update_default_setting(struct ieee80211_hw *hw) | ||
71 | { | ||
72 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
73 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
74 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
75 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
76 | u8 pcibridge_vendor = pcipriv->ndis_adapter.pcibridge_vendor; | ||
77 | u8 init_aspm; | ||
78 | |||
79 | ppsc->reg_rfps_level = 0; | ||
80 | ppsc->support_aspm = 0; | ||
81 | |||
82 | /*Update PCI ASPM setting */ | ||
83 | ppsc->const_amdpci_aspm = rtlpci->const_amdpci_aspm; | ||
84 | switch (rtlpci->const_pci_aspm) { | ||
85 | case 0: | ||
86 | /*No ASPM */ | ||
87 | break; | ||
88 | |||
89 | case 1: | ||
90 | /*ASPM dynamically enabled/disable. */ | ||
91 | ppsc->reg_rfps_level |= RT_RF_LPS_LEVEL_ASPM; | ||
92 | break; | ||
93 | |||
94 | case 2: | ||
95 | /*ASPM with Clock Req dynamically enabled/disable. */ | ||
96 | ppsc->reg_rfps_level |= (RT_RF_LPS_LEVEL_ASPM | | ||
97 | RT_RF_OFF_LEVL_CLK_REQ); | ||
98 | break; | ||
99 | |||
100 | case 3: | ||
101 | /* | ||
102 | * Always enable ASPM and Clock Req | ||
103 | * from initialization to halt. | ||
104 | * */ | ||
105 | ppsc->reg_rfps_level &= ~(RT_RF_LPS_LEVEL_ASPM); | ||
106 | ppsc->reg_rfps_level |= (RT_RF_PS_LEVEL_ALWAYS_ASPM | | ||
107 | RT_RF_OFF_LEVL_CLK_REQ); | ||
108 | break; | ||
109 | |||
110 | case 4: | ||
111 | /* | ||
112 | * Always enable ASPM without Clock Req | ||
113 | * from initialization to halt. | ||
114 | * */ | ||
115 | ppsc->reg_rfps_level &= ~(RT_RF_LPS_LEVEL_ASPM | | ||
116 | RT_RF_OFF_LEVL_CLK_REQ); | ||
117 | ppsc->reg_rfps_level |= RT_RF_PS_LEVEL_ALWAYS_ASPM; | ||
118 | break; | ||
119 | } | ||
120 | |||
121 | ppsc->reg_rfps_level |= RT_RF_OFF_LEVL_HALT_NIC; | ||
122 | |||
123 | /*Update Radio OFF setting */ | ||
124 | switch (rtlpci->const_hwsw_rfoff_d3) { | ||
125 | case 1: | ||
126 | if (ppsc->reg_rfps_level & RT_RF_LPS_LEVEL_ASPM) | ||
127 | ppsc->reg_rfps_level |= RT_RF_OFF_LEVL_ASPM; | ||
128 | break; | ||
129 | |||
130 | case 2: | ||
131 | if (ppsc->reg_rfps_level & RT_RF_LPS_LEVEL_ASPM) | ||
132 | ppsc->reg_rfps_level |= RT_RF_OFF_LEVL_ASPM; | ||
133 | ppsc->reg_rfps_level |= RT_RF_OFF_LEVL_HALT_NIC; | ||
134 | break; | ||
135 | |||
136 | case 3: | ||
137 | ppsc->reg_rfps_level |= RT_RF_OFF_LEVL_PCI_D3; | ||
138 | break; | ||
139 | } | ||
140 | |||
141 | /*Set HW definition to determine if it supports ASPM. */ | ||
142 | switch (rtlpci->const_support_pciaspm) { | ||
143 | case 0:{ | ||
144 | /*Not support ASPM. */ | ||
145 | bool support_aspm = false; | ||
146 | ppsc->support_aspm = support_aspm; | ||
147 | break; | ||
148 | } | ||
149 | case 1:{ | ||
150 | /*Support ASPM. */ | ||
151 | bool support_aspm = true; | ||
152 | bool support_backdoor = true; | ||
153 | ppsc->support_aspm = support_aspm; | ||
154 | |||
155 | /*if (priv->oem_id == RT_CID_TOSHIBA && | ||
156 | !priv->ndis_adapter.amd_l1_patch) | ||
157 | support_backdoor = false; */ | ||
158 | |||
159 | ppsc->support_backdoor = support_backdoor; | ||
160 | |||
161 | break; | ||
162 | } | ||
163 | case 2: | ||
164 | /*ASPM value set by chipset. */ | ||
165 | if (pcibridge_vendor == PCI_BRIDGE_VENDOR_INTEL) { | ||
166 | bool support_aspm = true; | ||
167 | ppsc->support_aspm = support_aspm; | ||
168 | } | ||
169 | break; | ||
170 | default: | ||
171 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
172 | ("switch case not process\n")); | ||
173 | break; | ||
174 | } | ||
175 | |||
176 | /* toshiba aspm issue, toshiba will set aspm selfly | ||
177 | * so we should not set aspm in driver */ | ||
178 | pci_read_config_byte(rtlpci->pdev, 0x80, &init_aspm); | ||
179 | if (rtlpriv->rtlhal.hw_type == HARDWARE_TYPE_RTL8192SE && | ||
180 | init_aspm == 0x43) | ||
181 | ppsc->support_aspm = false; | ||
182 | } | ||
183 | |||
184 | static bool _rtl_pci_platform_switch_device_pci_aspm( | ||
185 | struct ieee80211_hw *hw, | ||
186 | u8 value) | ||
187 | { | ||
188 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
189 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
190 | |||
191 | if (rtlhal->hw_type != HARDWARE_TYPE_RTL8192SE) | ||
192 | value |= 0x40; | ||
193 | |||
194 | pci_write_config_byte(rtlpci->pdev, 0x80, value); | ||
195 | |||
196 | return false; | ||
197 | } | ||
198 | |||
199 | /*When we set 0x01 to enable clk request. Set 0x0 to disable clk req.*/ | ||
200 | static bool _rtl_pci_switch_clk_req(struct ieee80211_hw *hw, u8 value) | ||
201 | { | ||
202 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
203 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
204 | |||
205 | pci_write_config_byte(rtlpci->pdev, 0x81, value); | ||
206 | |||
207 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) | ||
208 | udelay(100); | ||
209 | |||
210 | return true; | ||
211 | } | ||
212 | |||
213 | /*Disable RTL8192SE ASPM & Disable Pci Bridge ASPM*/ | ||
214 | static void rtl_pci_disable_aspm(struct ieee80211_hw *hw) | ||
215 | { | ||
216 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
217 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
218 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
219 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
220 | u8 pcibridge_vendor = pcipriv->ndis_adapter.pcibridge_vendor; | ||
221 | u32 pcicfg_addrport = pcipriv->ndis_adapter.pcicfg_addrport; | ||
222 | u8 num4bytes = pcipriv->ndis_adapter.num4bytes; | ||
223 | /*Retrieve original configuration settings. */ | ||
224 | u8 linkctrl_reg = pcipriv->ndis_adapter.linkctrl_reg; | ||
225 | u16 pcibridge_linkctrlreg = pcipriv->ndis_adapter. | ||
226 | pcibridge_linkctrlreg; | ||
227 | u16 aspmlevel = 0; | ||
228 | u8 tmp_u1b = 0; | ||
229 | |||
230 | if (!ppsc->support_aspm) | ||
231 | return; | ||
232 | |||
233 | if (pcibridge_vendor == PCI_BRIDGE_VENDOR_UNKNOWN) { | ||
234 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
235 | ("PCI(Bridge) UNKNOWN.\n")); | ||
236 | |||
237 | return; | ||
238 | } | ||
239 | |||
240 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_CLK_REQ) { | ||
241 | RT_CLEAR_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_CLK_REQ); | ||
242 | _rtl_pci_switch_clk_req(hw, 0x0); | ||
243 | } | ||
244 | |||
245 | /*for promising device will in L0 state after an I/O. */ | ||
246 | pci_read_config_byte(rtlpci->pdev, 0x80, &tmp_u1b); | ||
247 | |||
248 | /*Set corresponding value. */ | ||
249 | aspmlevel |= BIT(0) | BIT(1); | ||
250 | linkctrl_reg &= ~aspmlevel; | ||
251 | pcibridge_linkctrlreg &= ~(BIT(0) | BIT(1)); | ||
252 | |||
253 | _rtl_pci_platform_switch_device_pci_aspm(hw, linkctrl_reg); | ||
254 | udelay(50); | ||
255 | |||
256 | /*4 Disable Pci Bridge ASPM */ | ||
257 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
258 | pcicfg_addrport + (num4bytes << 2)); | ||
259 | rtl_pci_raw_write_port_uchar(PCI_CONF_DATA, pcibridge_linkctrlreg); | ||
260 | |||
261 | udelay(50); | ||
262 | } | ||
263 | |||
264 | /* | ||
265 | *Enable RTL8192SE ASPM & Enable Pci Bridge ASPM for | ||
266 | *power saving We should follow the sequence to enable | ||
267 | *RTL8192SE first then enable Pci Bridge ASPM | ||
268 | *or the system will show bluescreen. | ||
269 | */ | ||
270 | static void rtl_pci_enable_aspm(struct ieee80211_hw *hw) | ||
271 | { | ||
272 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
273 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
274 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
275 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
276 | u8 pcibridge_busnum = pcipriv->ndis_adapter.pcibridge_busnum; | ||
277 | u8 pcibridge_devnum = pcipriv->ndis_adapter.pcibridge_devnum; | ||
278 | u8 pcibridge_funcnum = pcipriv->ndis_adapter.pcibridge_funcnum; | ||
279 | u8 pcibridge_vendor = pcipriv->ndis_adapter.pcibridge_vendor; | ||
280 | u32 pcicfg_addrport = pcipriv->ndis_adapter.pcicfg_addrport; | ||
281 | u8 num4bytes = pcipriv->ndis_adapter.num4bytes; | ||
282 | u16 aspmlevel; | ||
283 | u8 u_pcibridge_aspmsetting; | ||
284 | u8 u_device_aspmsetting; | ||
285 | |||
286 | if (!ppsc->support_aspm) | ||
287 | return; | ||
288 | |||
289 | if (pcibridge_vendor == PCI_BRIDGE_VENDOR_UNKNOWN) { | ||
290 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
291 | ("PCI(Bridge) UNKNOWN.\n")); | ||
292 | return; | ||
293 | } | ||
294 | |||
295 | /*4 Enable Pci Bridge ASPM */ | ||
296 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
297 | pcicfg_addrport + (num4bytes << 2)); | ||
298 | |||
299 | u_pcibridge_aspmsetting = | ||
300 | pcipriv->ndis_adapter.pcibridge_linkctrlreg | | ||
301 | rtlpci->const_hostpci_aspm_setting; | ||
302 | |||
303 | if (pcibridge_vendor == PCI_BRIDGE_VENDOR_INTEL) | ||
304 | u_pcibridge_aspmsetting &= ~BIT(0); | ||
305 | |||
306 | rtl_pci_raw_write_port_uchar(PCI_CONF_DATA, u_pcibridge_aspmsetting); | ||
307 | |||
308 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
309 | ("PlatformEnableASPM():PciBridge busnumber[%x], " | ||
310 | "DevNumbe[%x], funcnumber[%x], Write reg[%x] = %x\n", | ||
311 | pcibridge_busnum, pcibridge_devnum, pcibridge_funcnum, | ||
312 | (pcipriv->ndis_adapter.pcibridge_pciehdr_offset + 0x10), | ||
313 | u_pcibridge_aspmsetting)); | ||
314 | |||
315 | udelay(50); | ||
316 | |||
317 | /*Get ASPM level (with/without Clock Req) */ | ||
318 | aspmlevel = rtlpci->const_devicepci_aspm_setting; | ||
319 | u_device_aspmsetting = pcipriv->ndis_adapter.linkctrl_reg; | ||
320 | |||
321 | /*_rtl_pci_platform_switch_device_pci_aspm(dev,*/ | ||
322 | /*(priv->ndis_adapter.linkctrl_reg | ASPMLevel)); */ | ||
323 | |||
324 | u_device_aspmsetting |= aspmlevel; | ||
325 | |||
326 | _rtl_pci_platform_switch_device_pci_aspm(hw, u_device_aspmsetting); | ||
327 | |||
328 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_CLK_REQ) { | ||
329 | _rtl_pci_switch_clk_req(hw, (ppsc->reg_rfps_level & | ||
330 | RT_RF_OFF_LEVL_CLK_REQ) ? 1 : 0); | ||
331 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_CLK_REQ); | ||
332 | } | ||
333 | udelay(100); | ||
334 | } | ||
335 | |||
336 | static bool rtl_pci_get_amd_l1_patch(struct ieee80211_hw *hw) | ||
337 | { | ||
338 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
339 | u32 pcicfg_addrport = pcipriv->ndis_adapter.pcicfg_addrport; | ||
340 | |||
341 | bool status = false; | ||
342 | u8 offset_e0; | ||
343 | unsigned offset_e4; | ||
344 | |||
345 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
346 | pcicfg_addrport + 0xE0); | ||
347 | rtl_pci_raw_write_port_uchar(PCI_CONF_DATA, 0xA0); | ||
348 | |||
349 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
350 | pcicfg_addrport + 0xE0); | ||
351 | rtl_pci_raw_read_port_uchar(PCI_CONF_DATA, &offset_e0); | ||
352 | |||
353 | if (offset_e0 == 0xA0) { | ||
354 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
355 | pcicfg_addrport + 0xE4); | ||
356 | rtl_pci_raw_read_port_ulong(PCI_CONF_DATA, &offset_e4); | ||
357 | if (offset_e4 & BIT(23)) | ||
358 | status = true; | ||
359 | } | ||
360 | |||
361 | return status; | ||
362 | } | ||
363 | |||
364 | static void rtl_pci_get_linkcontrol_field(struct ieee80211_hw *hw) | ||
365 | { | ||
366 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
367 | u8 capabilityoffset = pcipriv->ndis_adapter.pcibridge_pciehdr_offset; | ||
368 | u32 pcicfg_addrport = pcipriv->ndis_adapter.pcicfg_addrport; | ||
369 | u8 linkctrl_reg; | ||
370 | u8 num4bbytes; | ||
371 | |||
372 | num4bbytes = (capabilityoffset + 0x10) / 4; | ||
373 | |||
374 | /*Read Link Control Register */ | ||
375 | rtl_pci_raw_write_port_ulong(PCI_CONF_ADDRESS, | ||
376 | pcicfg_addrport + (num4bbytes << 2)); | ||
377 | rtl_pci_raw_read_port_uchar(PCI_CONF_DATA, &linkctrl_reg); | ||
378 | |||
379 | pcipriv->ndis_adapter.pcibridge_linkctrlreg = linkctrl_reg; | ||
380 | } | ||
381 | |||
382 | static void rtl_pci_parse_configuration(struct pci_dev *pdev, | ||
383 | struct ieee80211_hw *hw) | ||
384 | { | ||
385 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
386 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
387 | |||
388 | u8 tmp; | ||
389 | int pos; | ||
390 | u8 linkctrl_reg; | ||
391 | |||
392 | /*Link Control Register */ | ||
393 | pos = pci_find_capability(pdev, PCI_CAP_ID_EXP); | ||
394 | pci_read_config_byte(pdev, pos + PCI_EXP_LNKCTL, &linkctrl_reg); | ||
395 | pcipriv->ndis_adapter.linkctrl_reg = linkctrl_reg; | ||
396 | |||
397 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
398 | ("Link Control Register =%x\n", | ||
399 | pcipriv->ndis_adapter.linkctrl_reg)); | ||
400 | |||
401 | pci_read_config_byte(pdev, 0x98, &tmp); | ||
402 | tmp |= BIT(4); | ||
403 | pci_write_config_byte(pdev, 0x98, tmp); | ||
404 | |||
405 | tmp = 0x17; | ||
406 | pci_write_config_byte(pdev, 0x70f, tmp); | ||
407 | } | ||
408 | |||
409 | static void rtl_pci_init_aspm(struct ieee80211_hw *hw) | ||
410 | { | ||
411 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
412 | |||
413 | _rtl_pci_update_default_setting(hw); | ||
414 | |||
415 | if (ppsc->reg_rfps_level & RT_RF_PS_LEVEL_ALWAYS_ASPM) { | ||
416 | /*Always enable ASPM & Clock Req. */ | ||
417 | rtl_pci_enable_aspm(hw); | ||
418 | RT_SET_PS_LEVEL(ppsc, RT_RF_PS_LEVEL_ALWAYS_ASPM); | ||
419 | } | ||
420 | |||
421 | } | ||
422 | |||
423 | static void _rtl_pci_io_handler_init(struct device *dev, | ||
424 | struct ieee80211_hw *hw) | ||
425 | { | ||
426 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
427 | |||
428 | rtlpriv->io.dev = dev; | ||
429 | |||
430 | rtlpriv->io.write8_async = pci_write8_async; | ||
431 | rtlpriv->io.write16_async = pci_write16_async; | ||
432 | rtlpriv->io.write32_async = pci_write32_async; | ||
433 | |||
434 | rtlpriv->io.read8_sync = pci_read8_sync; | ||
435 | rtlpriv->io.read16_sync = pci_read16_sync; | ||
436 | rtlpriv->io.read32_sync = pci_read32_sync; | ||
437 | |||
438 | } | ||
439 | |||
440 | static void _rtl_pci_io_handler_release(struct ieee80211_hw *hw) | ||
441 | { | ||
442 | } | ||
443 | |||
444 | static bool _rtl_update_earlymode_info(struct ieee80211_hw *hw, | ||
445 | struct sk_buff *skb, struct rtl_tcb_desc *tcb_desc, u8 tid) | ||
446 | { | ||
447 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
448 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
449 | u8 additionlen = FCS_LEN; | ||
450 | struct sk_buff *next_skb; | ||
451 | |||
452 | /* here open is 4, wep/tkip is 8, aes is 12*/ | ||
453 | if (info->control.hw_key) | ||
454 | additionlen += info->control.hw_key->icv_len; | ||
455 | |||
456 | /* The most skb num is 6 */ | ||
457 | tcb_desc->empkt_num = 0; | ||
458 | spin_lock_bh(&rtlpriv->locks.waitq_lock); | ||
459 | skb_queue_walk(&rtlpriv->mac80211.skb_waitq[tid], next_skb) { | ||
460 | struct ieee80211_tx_info *next_info; | ||
461 | |||
462 | next_info = IEEE80211_SKB_CB(next_skb); | ||
463 | if (next_info->flags & IEEE80211_TX_CTL_AMPDU) { | ||
464 | tcb_desc->empkt_len[tcb_desc->empkt_num] = | ||
465 | next_skb->len + additionlen; | ||
466 | tcb_desc->empkt_num++; | ||
467 | } else { | ||
468 | break; | ||
469 | } | ||
470 | |||
471 | if (skb_queue_is_last(&rtlpriv->mac80211.skb_waitq[tid], | ||
472 | next_skb)) | ||
473 | break; | ||
474 | |||
475 | if (tcb_desc->empkt_num >= 5) | ||
476 | break; | ||
477 | } | ||
478 | spin_unlock_bh(&rtlpriv->locks.waitq_lock); | ||
479 | |||
480 | return true; | ||
481 | } | ||
482 | |||
483 | /* just for early mode now */ | ||
484 | static void _rtl_pci_tx_chk_waitq(struct ieee80211_hw *hw) | ||
485 | { | ||
486 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
487 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
488 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
489 | struct sk_buff *skb = NULL; | ||
490 | struct ieee80211_tx_info *info = NULL; | ||
491 | int tid; /* should be int */ | ||
492 | |||
493 | if (!rtlpriv->rtlhal.earlymode_enable) | ||
494 | return; | ||
495 | |||
496 | /* we juse use em for BE/BK/VI/VO */ | ||
497 | for (tid = 7; tid >= 0; tid--) { | ||
498 | u8 hw_queue = ac_to_hwq[rtl_tid_to_ac(hw, tid)]; | ||
499 | struct rtl8192_tx_ring *ring = &rtlpci->tx_ring[hw_queue]; | ||
500 | while (!mac->act_scanning && | ||
501 | rtlpriv->psc.rfpwr_state == ERFON) { | ||
502 | struct rtl_tcb_desc tcb_desc; | ||
503 | memset(&tcb_desc, 0, sizeof(struct rtl_tcb_desc)); | ||
504 | |||
505 | spin_lock_bh(&rtlpriv->locks.waitq_lock); | ||
506 | if (!skb_queue_empty(&mac->skb_waitq[tid]) && | ||
507 | (ring->entries - skb_queue_len(&ring->queue) > 5)) { | ||
508 | skb = skb_dequeue(&mac->skb_waitq[tid]); | ||
509 | } else { | ||
510 | spin_unlock_bh(&rtlpriv->locks.waitq_lock); | ||
511 | break; | ||
512 | } | ||
513 | spin_unlock_bh(&rtlpriv->locks.waitq_lock); | ||
514 | |||
515 | /* Some macaddr can't do early mode. like | ||
516 | * multicast/broadcast/no_qos data */ | ||
517 | info = IEEE80211_SKB_CB(skb); | ||
518 | if (info->flags & IEEE80211_TX_CTL_AMPDU) | ||
519 | _rtl_update_earlymode_info(hw, skb, | ||
520 | &tcb_desc, tid); | ||
521 | |||
522 | rtlpriv->intf_ops->adapter_tx(hw, skb, &tcb_desc); | ||
523 | } | ||
524 | } | ||
525 | } | ||
526 | |||
527 | |||
528 | static void _rtl_pci_tx_isr(struct ieee80211_hw *hw, int prio) | ||
529 | { | ||
530 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
531 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
532 | |||
533 | struct rtl8192_tx_ring *ring = &rtlpci->tx_ring[prio]; | ||
534 | |||
535 | while (skb_queue_len(&ring->queue)) { | ||
536 | struct rtl_tx_desc *entry = &ring->desc[ring->idx]; | ||
537 | struct sk_buff *skb; | ||
538 | struct ieee80211_tx_info *info; | ||
539 | __le16 fc; | ||
540 | u8 tid; | ||
541 | |||
542 | u8 own = (u8) rtlpriv->cfg->ops->get_desc((u8 *) entry, true, | ||
543 | HW_DESC_OWN); | ||
544 | |||
545 | /* | ||
546 | *beacon packet will only use the first | ||
547 | *descriptor defautly,and the own may not | ||
548 | *be cleared by the hardware | ||
549 | */ | ||
550 | if (own) | ||
551 | return; | ||
552 | ring->idx = (ring->idx + 1) % ring->entries; | ||
553 | |||
554 | skb = __skb_dequeue(&ring->queue); | ||
555 | pci_unmap_single(rtlpci->pdev, | ||
556 | rtlpriv->cfg->ops-> | ||
557 | get_desc((u8 *) entry, true, | ||
558 | HW_DESC_TXBUFF_ADDR), | ||
559 | skb->len, PCI_DMA_TODEVICE); | ||
560 | |||
561 | /* remove early mode header */ | ||
562 | if (rtlpriv->rtlhal.earlymode_enable) | ||
563 | skb_pull(skb, EM_HDR_LEN); | ||
564 | |||
565 | RT_TRACE(rtlpriv, (COMP_INTR | COMP_SEND), DBG_TRACE, | ||
566 | ("new ring->idx:%d, " | ||
567 | "free: skb_queue_len:%d, free: seq:%x\n", | ||
568 | ring->idx, | ||
569 | skb_queue_len(&ring->queue), | ||
570 | *(u16 *) (skb->data + 22))); | ||
571 | |||
572 | if (prio == TXCMD_QUEUE) { | ||
573 | dev_kfree_skb(skb); | ||
574 | goto tx_status_ok; | ||
575 | |||
576 | } | ||
577 | |||
578 | /* for sw LPS, just after NULL skb send out, we can | ||
579 | * sure AP kown we are sleeped, our we should not let | ||
580 | * rf to sleep*/ | ||
581 | fc = rtl_get_fc(skb); | ||
582 | if (ieee80211_is_nullfunc(fc)) { | ||
583 | if (ieee80211_has_pm(fc)) { | ||
584 | rtlpriv->mac80211.offchan_deley = true; | ||
585 | rtlpriv->psc.state_inap = 1; | ||
586 | } else { | ||
587 | rtlpriv->psc.state_inap = 0; | ||
588 | } | ||
589 | } | ||
590 | |||
591 | /* update tid tx pkt num */ | ||
592 | tid = rtl_get_tid(skb); | ||
593 | if (tid <= 7) | ||
594 | rtlpriv->link_info.tidtx_inperiod[tid]++; | ||
595 | |||
596 | info = IEEE80211_SKB_CB(skb); | ||
597 | ieee80211_tx_info_clear_status(info); | ||
598 | |||
599 | info->flags |= IEEE80211_TX_STAT_ACK; | ||
600 | /*info->status.rates[0].count = 1; */ | ||
601 | |||
602 | ieee80211_tx_status_irqsafe(hw, skb); | ||
603 | |||
604 | if ((ring->entries - skb_queue_len(&ring->queue)) | ||
605 | == 2) { | ||
606 | |||
607 | RT_TRACE(rtlpriv, COMP_ERR, DBG_LOUD, | ||
608 | ("more desc left, wake" | ||
609 | "skb_queue@%d,ring->idx = %d," | ||
610 | "skb_queue_len = 0x%d\n", | ||
611 | prio, ring->idx, | ||
612 | skb_queue_len(&ring->queue))); | ||
613 | |||
614 | ieee80211_wake_queue(hw, | ||
615 | skb_get_queue_mapping | ||
616 | (skb)); | ||
617 | } | ||
618 | tx_status_ok: | ||
619 | skb = NULL; | ||
620 | } | ||
621 | |||
622 | if (((rtlpriv->link_info.num_rx_inperiod + | ||
623 | rtlpriv->link_info.num_tx_inperiod) > 8) || | ||
624 | (rtlpriv->link_info.num_rx_inperiod > 2)) { | ||
625 | rtl_lps_leave(hw); | ||
626 | } | ||
627 | } | ||
628 | |||
629 | static void _rtl_pci_rx_interrupt(struct ieee80211_hw *hw) | ||
630 | { | ||
631 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
632 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
633 | int rx_queue_idx = RTL_PCI_RX_MPDU_QUEUE; | ||
634 | |||
635 | struct ieee80211_rx_status rx_status = { 0 }; | ||
636 | unsigned int count = rtlpci->rxringcount; | ||
637 | u8 own; | ||
638 | u8 tmp_one; | ||
639 | u32 bufferaddress; | ||
640 | bool unicast = false; | ||
641 | |||
642 | struct rtl_stats stats = { | ||
643 | .signal = 0, | ||
644 | .noise = -98, | ||
645 | .rate = 0, | ||
646 | }; | ||
647 | |||
648 | /*RX NORMAL PKT */ | ||
649 | while (count--) { | ||
650 | /*rx descriptor */ | ||
651 | struct rtl_rx_desc *pdesc = &rtlpci->rx_ring[rx_queue_idx].desc[ | ||
652 | rtlpci->rx_ring[rx_queue_idx].idx]; | ||
653 | /*rx pkt */ | ||
654 | struct sk_buff *skb = rtlpci->rx_ring[rx_queue_idx].rx_buf[ | ||
655 | rtlpci->rx_ring[rx_queue_idx].idx]; | ||
656 | |||
657 | own = (u8) rtlpriv->cfg->ops->get_desc((u8 *) pdesc, | ||
658 | false, HW_DESC_OWN); | ||
659 | |||
660 | if (own) { | ||
661 | /*wait data to be filled by hardware */ | ||
662 | return; | ||
663 | } else { | ||
664 | struct ieee80211_hdr *hdr; | ||
665 | __le16 fc; | ||
666 | struct sk_buff *new_skb = NULL; | ||
667 | |||
668 | rtlpriv->cfg->ops->query_rx_desc(hw, &stats, | ||
669 | &rx_status, | ||
670 | (u8 *) pdesc, skb); | ||
671 | |||
672 | new_skb = dev_alloc_skb(rtlpci->rxbuffersize); | ||
673 | if (unlikely(!new_skb)) { | ||
674 | RT_TRACE(rtlpriv, (COMP_INTR | COMP_RECV), | ||
675 | DBG_DMESG, | ||
676 | ("can't alloc skb for rx\n")); | ||
677 | goto done; | ||
678 | } | ||
679 | |||
680 | pci_unmap_single(rtlpci->pdev, | ||
681 | *((dma_addr_t *) skb->cb), | ||
682 | rtlpci->rxbuffersize, | ||
683 | PCI_DMA_FROMDEVICE); | ||
684 | |||
685 | skb_put(skb, rtlpriv->cfg->ops->get_desc((u8 *) pdesc, | ||
686 | false, | ||
687 | HW_DESC_RXPKT_LEN)); | ||
688 | skb_reserve(skb, | ||
689 | stats.rx_drvinfo_size + stats.rx_bufshift); | ||
690 | |||
691 | /* | ||
692 | *NOTICE This can not be use for mac80211, | ||
693 | *this is done in mac80211 code, | ||
694 | *if you done here sec DHCP will fail | ||
695 | *skb_trim(skb, skb->len - 4); | ||
696 | */ | ||
697 | |||
698 | hdr = rtl_get_hdr(skb); | ||
699 | fc = rtl_get_fc(skb); | ||
700 | |||
701 | if (!stats.crc && !stats.hwerror) { | ||
702 | memcpy(IEEE80211_SKB_RXCB(skb), &rx_status, | ||
703 | sizeof(rx_status)); | ||
704 | |||
705 | if (is_broadcast_ether_addr(hdr->addr1)) { | ||
706 | ;/*TODO*/ | ||
707 | } else if (is_multicast_ether_addr(hdr->addr1)) { | ||
708 | ;/*TODO*/ | ||
709 | } else { | ||
710 | unicast = true; | ||
711 | rtlpriv->stats.rxbytesunicast += | ||
712 | skb->len; | ||
713 | } | ||
714 | |||
715 | rtl_is_special_data(hw, skb, false); | ||
716 | |||
717 | if (ieee80211_is_data(fc)) { | ||
718 | rtlpriv->cfg->ops->led_control(hw, | ||
719 | LED_CTL_RX); | ||
720 | |||
721 | if (unicast) | ||
722 | rtlpriv->link_info. | ||
723 | num_rx_inperiod++; | ||
724 | } | ||
725 | |||
726 | /* for sw lps */ | ||
727 | rtl_swlps_beacon(hw, (void *)skb->data, | ||
728 | skb->len); | ||
729 | rtl_recognize_peer(hw, (void *)skb->data, | ||
730 | skb->len); | ||
731 | if ((rtlpriv->mac80211.opmode == | ||
732 | NL80211_IFTYPE_AP) && | ||
733 | (rtlpriv->rtlhal.current_bandtype == | ||
734 | BAND_ON_2_4G) && | ||
735 | (ieee80211_is_beacon(fc) || | ||
736 | ieee80211_is_probe_resp(fc))) { | ||
737 | dev_kfree_skb_any(skb); | ||
738 | } else { | ||
739 | if (unlikely(!rtl_action_proc(hw, skb, | ||
740 | false))) { | ||
741 | dev_kfree_skb_any(skb); | ||
742 | } else { | ||
743 | struct sk_buff *uskb = NULL; | ||
744 | u8 *pdata; | ||
745 | uskb = dev_alloc_skb(skb->len | ||
746 | + 128); | ||
747 | memcpy(IEEE80211_SKB_RXCB(uskb), | ||
748 | &rx_status, | ||
749 | sizeof(rx_status)); | ||
750 | pdata = (u8 *)skb_put(uskb, | ||
751 | skb->len); | ||
752 | memcpy(pdata, skb->data, | ||
753 | skb->len); | ||
754 | dev_kfree_skb_any(skb); | ||
755 | |||
756 | ieee80211_rx_irqsafe(hw, uskb); | ||
757 | } | ||
758 | } | ||
759 | } else { | ||
760 | dev_kfree_skb_any(skb); | ||
761 | } | ||
762 | |||
763 | if (((rtlpriv->link_info.num_rx_inperiod + | ||
764 | rtlpriv->link_info.num_tx_inperiod) > 8) || | ||
765 | (rtlpriv->link_info.num_rx_inperiod > 2)) { | ||
766 | rtl_lps_leave(hw); | ||
767 | } | ||
768 | |||
769 | skb = new_skb; | ||
770 | |||
771 | rtlpci->rx_ring[rx_queue_idx].rx_buf[rtlpci-> | ||
772 | rx_ring | ||
773 | [rx_queue_idx]. | ||
774 | idx] = skb; | ||
775 | *((dma_addr_t *) skb->cb) = | ||
776 | pci_map_single(rtlpci->pdev, skb_tail_pointer(skb), | ||
777 | rtlpci->rxbuffersize, | ||
778 | PCI_DMA_FROMDEVICE); | ||
779 | |||
780 | } | ||
781 | done: | ||
782 | bufferaddress = (*((dma_addr_t *)skb->cb)); | ||
783 | tmp_one = 1; | ||
784 | rtlpriv->cfg->ops->set_desc((u8 *) pdesc, false, | ||
785 | HW_DESC_RXBUFF_ADDR, | ||
786 | (u8 *)&bufferaddress); | ||
787 | rtlpriv->cfg->ops->set_desc((u8 *)pdesc, false, HW_DESC_RXOWN, | ||
788 | (u8 *)&tmp_one); | ||
789 | rtlpriv->cfg->ops->set_desc((u8 *)pdesc, false, | ||
790 | HW_DESC_RXPKT_LEN, | ||
791 | (u8 *)&rtlpci->rxbuffersize); | ||
792 | |||
793 | if (rtlpci->rx_ring[rx_queue_idx].idx == | ||
794 | rtlpci->rxringcount - 1) | ||
795 | rtlpriv->cfg->ops->set_desc((u8 *)pdesc, false, | ||
796 | HW_DESC_RXERO, | ||
797 | (u8 *)&tmp_one); | ||
798 | |||
799 | rtlpci->rx_ring[rx_queue_idx].idx = | ||
800 | (rtlpci->rx_ring[rx_queue_idx].idx + 1) % | ||
801 | rtlpci->rxringcount; | ||
802 | } | ||
803 | |||
804 | } | ||
805 | |||
806 | static irqreturn_t _rtl_pci_interrupt(int irq, void *dev_id) | ||
807 | { | ||
808 | struct ieee80211_hw *hw = dev_id; | ||
809 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
810 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
811 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
812 | unsigned long flags; | ||
813 | u32 inta = 0; | ||
814 | u32 intb = 0; | ||
815 | |||
816 | if (rtlpci->irq_enabled == 0) | ||
817 | return IRQ_HANDLED; | ||
818 | |||
819 | spin_lock_irqsave(&rtlpriv->locks.irq_th_lock, flags); | ||
820 | |||
821 | /*read ISR: 4/8bytes */ | ||
822 | rtlpriv->cfg->ops->interrupt_recognized(hw, &inta, &intb); | ||
823 | |||
824 | /*Shared IRQ or HW disappared */ | ||
825 | if (!inta || inta == 0xffff) | ||
826 | goto done; | ||
827 | |||
828 | /*<1> beacon related */ | ||
829 | if (inta & rtlpriv->cfg->maps[RTL_IMR_TBDOK]) { | ||
830 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
831 | ("beacon ok interrupt!\n")); | ||
832 | } | ||
833 | |||
834 | if (unlikely(inta & rtlpriv->cfg->maps[RTL_IMR_TBDER])) { | ||
835 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
836 | ("beacon err interrupt!\n")); | ||
837 | } | ||
838 | |||
839 | if (inta & rtlpriv->cfg->maps[RTL_IMR_BDOK]) { | ||
840 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
841 | ("beacon interrupt!\n")); | ||
842 | } | ||
843 | |||
844 | if (inta & rtlpriv->cfg->maps[RTL_IMR_BcnInt]) { | ||
845 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
846 | ("prepare beacon for interrupt!\n")); | ||
847 | tasklet_schedule(&rtlpriv->works.irq_prepare_bcn_tasklet); | ||
848 | } | ||
849 | |||
850 | /*<3> Tx related */ | ||
851 | if (unlikely(inta & rtlpriv->cfg->maps[RTL_IMR_TXFOVW])) | ||
852 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, ("IMR_TXFOVW!\n")); | ||
853 | |||
854 | if (inta & rtlpriv->cfg->maps[RTL_IMR_MGNTDOK]) { | ||
855 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
856 | ("Manage ok interrupt!\n")); | ||
857 | _rtl_pci_tx_isr(hw, MGNT_QUEUE); | ||
858 | } | ||
859 | |||
860 | if (inta & rtlpriv->cfg->maps[RTL_IMR_HIGHDOK]) { | ||
861 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
862 | ("HIGH_QUEUE ok interrupt!\n")); | ||
863 | _rtl_pci_tx_isr(hw, HIGH_QUEUE); | ||
864 | } | ||
865 | |||
866 | if (inta & rtlpriv->cfg->maps[RTL_IMR_BKDOK]) { | ||
867 | rtlpriv->link_info.num_tx_inperiod++; | ||
868 | |||
869 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
870 | ("BK Tx OK interrupt!\n")); | ||
871 | _rtl_pci_tx_isr(hw, BK_QUEUE); | ||
872 | } | ||
873 | |||
874 | if (inta & rtlpriv->cfg->maps[RTL_IMR_BEDOK]) { | ||
875 | rtlpriv->link_info.num_tx_inperiod++; | ||
876 | |||
877 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
878 | ("BE TX OK interrupt!\n")); | ||
879 | _rtl_pci_tx_isr(hw, BE_QUEUE); | ||
880 | } | ||
881 | |||
882 | if (inta & rtlpriv->cfg->maps[RTL_IMR_VIDOK]) { | ||
883 | rtlpriv->link_info.num_tx_inperiod++; | ||
884 | |||
885 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
886 | ("VI TX OK interrupt!\n")); | ||
887 | _rtl_pci_tx_isr(hw, VI_QUEUE); | ||
888 | } | ||
889 | |||
890 | if (inta & rtlpriv->cfg->maps[RTL_IMR_VODOK]) { | ||
891 | rtlpriv->link_info.num_tx_inperiod++; | ||
892 | |||
893 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
894 | ("Vo TX OK interrupt!\n")); | ||
895 | _rtl_pci_tx_isr(hw, VO_QUEUE); | ||
896 | } | ||
897 | |||
898 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) { | ||
899 | if (inta & rtlpriv->cfg->maps[RTL_IMR_COMDOK]) { | ||
900 | rtlpriv->link_info.num_tx_inperiod++; | ||
901 | |||
902 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, | ||
903 | ("CMD TX OK interrupt!\n")); | ||
904 | _rtl_pci_tx_isr(hw, TXCMD_QUEUE); | ||
905 | } | ||
906 | } | ||
907 | |||
908 | /*<2> Rx related */ | ||
909 | if (inta & rtlpriv->cfg->maps[RTL_IMR_ROK]) { | ||
910 | RT_TRACE(rtlpriv, COMP_INTR, DBG_TRACE, ("Rx ok interrupt!\n")); | ||
911 | _rtl_pci_rx_interrupt(hw); | ||
912 | } | ||
913 | |||
914 | if (unlikely(inta & rtlpriv->cfg->maps[RTL_IMR_RDU])) { | ||
915 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
916 | ("rx descriptor unavailable!\n")); | ||
917 | _rtl_pci_rx_interrupt(hw); | ||
918 | } | ||
919 | |||
920 | if (unlikely(inta & rtlpriv->cfg->maps[RTL_IMR_RXFOVW])) { | ||
921 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, ("rx overflow !\n")); | ||
922 | _rtl_pci_rx_interrupt(hw); | ||
923 | } | ||
924 | |||
925 | if (rtlpriv->rtlhal.earlymode_enable) | ||
926 | tasklet_schedule(&rtlpriv->works.irq_tasklet); | ||
927 | |||
928 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
929 | return IRQ_HANDLED; | ||
930 | |||
931 | done: | ||
932 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
933 | return IRQ_HANDLED; | ||
934 | } | ||
935 | |||
936 | static void _rtl_pci_irq_tasklet(struct ieee80211_hw *hw) | ||
937 | { | ||
938 | _rtl_pci_tx_chk_waitq(hw); | ||
939 | } | ||
940 | |||
941 | static void _rtl_pci_prepare_bcn_tasklet(struct ieee80211_hw *hw) | ||
942 | { | ||
943 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
944 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
945 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
946 | struct rtl8192_tx_ring *ring = NULL; | ||
947 | struct ieee80211_hdr *hdr = NULL; | ||
948 | struct ieee80211_tx_info *info = NULL; | ||
949 | struct sk_buff *pskb = NULL; | ||
950 | struct rtl_tx_desc *pdesc = NULL; | ||
951 | struct rtl_tcb_desc tcb_desc; | ||
952 | u8 temp_one = 1; | ||
953 | |||
954 | memset(&tcb_desc, 0, sizeof(struct rtl_tcb_desc)); | ||
955 | ring = &rtlpci->tx_ring[BEACON_QUEUE]; | ||
956 | pskb = __skb_dequeue(&ring->queue); | ||
957 | if (pskb) | ||
958 | kfree_skb(pskb); | ||
959 | |||
960 | /*NB: the beacon data buffer must be 32-bit aligned. */ | ||
961 | pskb = ieee80211_beacon_get(hw, mac->vif); | ||
962 | if (pskb == NULL) | ||
963 | return; | ||
964 | hdr = rtl_get_hdr(pskb); | ||
965 | info = IEEE80211_SKB_CB(pskb); | ||
966 | pdesc = &ring->desc[0]; | ||
967 | rtlpriv->cfg->ops->fill_tx_desc(hw, hdr, (u8 *) pdesc, | ||
968 | info, pskb, BEACON_QUEUE, &tcb_desc); | ||
969 | |||
970 | __skb_queue_tail(&ring->queue, pskb); | ||
971 | |||
972 | rtlpriv->cfg->ops->set_desc((u8 *) pdesc, true, HW_DESC_OWN, | ||
973 | (u8 *)&temp_one); | ||
974 | |||
975 | return; | ||
976 | } | ||
977 | |||
978 | static void _rtl_pci_init_trx_var(struct ieee80211_hw *hw) | ||
979 | { | ||
980 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
981 | u8 i; | ||
982 | |||
983 | for (i = 0; i < RTL_PCI_MAX_TX_QUEUE_COUNT; i++) | ||
984 | rtlpci->txringcount[i] = RT_TXDESC_NUM; | ||
985 | |||
986 | /* | ||
987 | *we just alloc 2 desc for beacon queue, | ||
988 | *because we just need first desc in hw beacon. | ||
989 | */ | ||
990 | rtlpci->txringcount[BEACON_QUEUE] = 2; | ||
991 | |||
992 | /* | ||
993 | *BE queue need more descriptor for performance | ||
994 | *consideration or, No more tx desc will happen, | ||
995 | *and may cause mac80211 mem leakage. | ||
996 | */ | ||
997 | rtlpci->txringcount[BE_QUEUE] = RT_TXDESC_NUM_BE_QUEUE; | ||
998 | |||
999 | rtlpci->rxbuffersize = 9100; /*2048/1024; */ | ||
1000 | rtlpci->rxringcount = RTL_PCI_MAX_RX_COUNT; /*64; */ | ||
1001 | } | ||
1002 | |||
1003 | static void _rtl_pci_init_struct(struct ieee80211_hw *hw, | ||
1004 | struct pci_dev *pdev) | ||
1005 | { | ||
1006 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1007 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1008 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1009 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1010 | |||
1011 | rtlpci->up_first_time = true; | ||
1012 | rtlpci->being_init_adapter = false; | ||
1013 | |||
1014 | rtlhal->hw = hw; | ||
1015 | rtlpci->pdev = pdev; | ||
1016 | |||
1017 | /*Tx/Rx related var */ | ||
1018 | _rtl_pci_init_trx_var(hw); | ||
1019 | |||
1020 | /*IBSS*/ mac->beacon_interval = 100; | ||
1021 | |||
1022 | /*AMPDU*/ | ||
1023 | mac->min_space_cfg = 0; | ||
1024 | mac->max_mss_density = 0; | ||
1025 | /*set sane AMPDU defaults */ | ||
1026 | mac->current_ampdu_density = 7; | ||
1027 | mac->current_ampdu_factor = 3; | ||
1028 | |||
1029 | /*QOS*/ | ||
1030 | rtlpci->acm_method = eAcmWay2_SW; | ||
1031 | |||
1032 | /*task */ | ||
1033 | tasklet_init(&rtlpriv->works.irq_tasklet, | ||
1034 | (void (*)(unsigned long))_rtl_pci_irq_tasklet, | ||
1035 | (unsigned long)hw); | ||
1036 | tasklet_init(&rtlpriv->works.irq_prepare_bcn_tasklet, | ||
1037 | (void (*)(unsigned long))_rtl_pci_prepare_bcn_tasklet, | ||
1038 | (unsigned long)hw); | ||
1039 | } | ||
1040 | |||
1041 | static int _rtl_pci_init_tx_ring(struct ieee80211_hw *hw, | ||
1042 | unsigned int prio, unsigned int entries) | ||
1043 | { | ||
1044 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1045 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1046 | struct rtl_tx_desc *ring; | ||
1047 | dma_addr_t dma; | ||
1048 | u32 nextdescaddress; | ||
1049 | int i; | ||
1050 | |||
1051 | ring = pci_alloc_consistent(rtlpci->pdev, | ||
1052 | sizeof(*ring) * entries, &dma); | ||
1053 | |||
1054 | if (!ring || (unsigned long)ring & 0xFF) { | ||
1055 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1056 | ("Cannot allocate TX ring (prio = %d)\n", prio)); | ||
1057 | return -ENOMEM; | ||
1058 | } | ||
1059 | |||
1060 | memset(ring, 0, sizeof(*ring) * entries); | ||
1061 | rtlpci->tx_ring[prio].desc = ring; | ||
1062 | rtlpci->tx_ring[prio].dma = dma; | ||
1063 | rtlpci->tx_ring[prio].idx = 0; | ||
1064 | rtlpci->tx_ring[prio].entries = entries; | ||
1065 | skb_queue_head_init(&rtlpci->tx_ring[prio].queue); | ||
1066 | |||
1067 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1068 | ("queue:%d, ring_addr:%p\n", prio, ring)); | ||
1069 | |||
1070 | for (i = 0; i < entries; i++) { | ||
1071 | nextdescaddress = (u32) dma + | ||
1072 | ((i + 1) % entries) * | ||
1073 | sizeof(*ring); | ||
1074 | |||
1075 | rtlpriv->cfg->ops->set_desc((u8 *)&(ring[i]), | ||
1076 | true, HW_DESC_TX_NEXTDESC_ADDR, | ||
1077 | (u8 *)&nextdescaddress); | ||
1078 | } | ||
1079 | |||
1080 | return 0; | ||
1081 | } | ||
1082 | |||
1083 | static int _rtl_pci_init_rx_ring(struct ieee80211_hw *hw) | ||
1084 | { | ||
1085 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1086 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1087 | struct rtl_rx_desc *entry = NULL; | ||
1088 | int i, rx_queue_idx; | ||
1089 | u8 tmp_one = 1; | ||
1090 | |||
1091 | /* | ||
1092 | *rx_queue_idx 0:RX_MPDU_QUEUE | ||
1093 | *rx_queue_idx 1:RX_CMD_QUEUE | ||
1094 | */ | ||
1095 | for (rx_queue_idx = 0; rx_queue_idx < RTL_PCI_MAX_RX_QUEUE; | ||
1096 | rx_queue_idx++) { | ||
1097 | rtlpci->rx_ring[rx_queue_idx].desc = | ||
1098 | pci_alloc_consistent(rtlpci->pdev, | ||
1099 | sizeof(*rtlpci->rx_ring[rx_queue_idx]. | ||
1100 | desc) * rtlpci->rxringcount, | ||
1101 | &rtlpci->rx_ring[rx_queue_idx].dma); | ||
1102 | |||
1103 | if (!rtlpci->rx_ring[rx_queue_idx].desc || | ||
1104 | (unsigned long)rtlpci->rx_ring[rx_queue_idx].desc & 0xFF) { | ||
1105 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1106 | ("Cannot allocate RX ring\n")); | ||
1107 | return -ENOMEM; | ||
1108 | } | ||
1109 | |||
1110 | memset(rtlpci->rx_ring[rx_queue_idx].desc, 0, | ||
1111 | sizeof(*rtlpci->rx_ring[rx_queue_idx].desc) * | ||
1112 | rtlpci->rxringcount); | ||
1113 | |||
1114 | rtlpci->rx_ring[rx_queue_idx].idx = 0; | ||
1115 | |||
1116 | /* If amsdu_8k is disabled, set buffersize to 4096. This | ||
1117 | * change will reduce memory fragmentation. | ||
1118 | */ | ||
1119 | if (rtlpci->rxbuffersize > 4096 && | ||
1120 | rtlpriv->rtlhal.disable_amsdu_8k) | ||
1121 | rtlpci->rxbuffersize = 4096; | ||
1122 | |||
1123 | for (i = 0; i < rtlpci->rxringcount; i++) { | ||
1124 | struct sk_buff *skb = | ||
1125 | dev_alloc_skb(rtlpci->rxbuffersize); | ||
1126 | u32 bufferaddress; | ||
1127 | if (!skb) | ||
1128 | return 0; | ||
1129 | entry = &rtlpci->rx_ring[rx_queue_idx].desc[i]; | ||
1130 | |||
1131 | /*skb->dev = dev; */ | ||
1132 | |||
1133 | rtlpci->rx_ring[rx_queue_idx].rx_buf[i] = skb; | ||
1134 | |||
1135 | /* | ||
1136 | *just set skb->cb to mapping addr | ||
1137 | *for pci_unmap_single use | ||
1138 | */ | ||
1139 | *((dma_addr_t *) skb->cb) = | ||
1140 | pci_map_single(rtlpci->pdev, skb_tail_pointer(skb), | ||
1141 | rtlpci->rxbuffersize, | ||
1142 | PCI_DMA_FROMDEVICE); | ||
1143 | |||
1144 | bufferaddress = (*((dma_addr_t *)skb->cb)); | ||
1145 | rtlpriv->cfg->ops->set_desc((u8 *)entry, false, | ||
1146 | HW_DESC_RXBUFF_ADDR, | ||
1147 | (u8 *)&bufferaddress); | ||
1148 | rtlpriv->cfg->ops->set_desc((u8 *)entry, false, | ||
1149 | HW_DESC_RXPKT_LEN, | ||
1150 | (u8 *)&rtlpci-> | ||
1151 | rxbuffersize); | ||
1152 | rtlpriv->cfg->ops->set_desc((u8 *) entry, false, | ||
1153 | HW_DESC_RXOWN, | ||
1154 | (u8 *)&tmp_one); | ||
1155 | } | ||
1156 | |||
1157 | rtlpriv->cfg->ops->set_desc((u8 *) entry, false, | ||
1158 | HW_DESC_RXERO, (u8 *)&tmp_one); | ||
1159 | } | ||
1160 | return 0; | ||
1161 | } | ||
1162 | |||
1163 | static void _rtl_pci_free_tx_ring(struct ieee80211_hw *hw, | ||
1164 | unsigned int prio) | ||
1165 | { | ||
1166 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1167 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1168 | struct rtl8192_tx_ring *ring = &rtlpci->tx_ring[prio]; | ||
1169 | |||
1170 | while (skb_queue_len(&ring->queue)) { | ||
1171 | struct rtl_tx_desc *entry = &ring->desc[ring->idx]; | ||
1172 | struct sk_buff *skb = __skb_dequeue(&ring->queue); | ||
1173 | |||
1174 | pci_unmap_single(rtlpci->pdev, | ||
1175 | rtlpriv->cfg-> | ||
1176 | ops->get_desc((u8 *) entry, true, | ||
1177 | HW_DESC_TXBUFF_ADDR), | ||
1178 | skb->len, PCI_DMA_TODEVICE); | ||
1179 | kfree_skb(skb); | ||
1180 | ring->idx = (ring->idx + 1) % ring->entries; | ||
1181 | } | ||
1182 | |||
1183 | pci_free_consistent(rtlpci->pdev, | ||
1184 | sizeof(*ring->desc) * ring->entries, | ||
1185 | ring->desc, ring->dma); | ||
1186 | ring->desc = NULL; | ||
1187 | } | ||
1188 | |||
1189 | static void _rtl_pci_free_rx_ring(struct rtl_pci *rtlpci) | ||
1190 | { | ||
1191 | int i, rx_queue_idx; | ||
1192 | |||
1193 | /*rx_queue_idx 0:RX_MPDU_QUEUE */ | ||
1194 | /*rx_queue_idx 1:RX_CMD_QUEUE */ | ||
1195 | for (rx_queue_idx = 0; rx_queue_idx < RTL_PCI_MAX_RX_QUEUE; | ||
1196 | rx_queue_idx++) { | ||
1197 | for (i = 0; i < rtlpci->rxringcount; i++) { | ||
1198 | struct sk_buff *skb = | ||
1199 | rtlpci->rx_ring[rx_queue_idx].rx_buf[i]; | ||
1200 | if (!skb) | ||
1201 | continue; | ||
1202 | |||
1203 | pci_unmap_single(rtlpci->pdev, | ||
1204 | *((dma_addr_t *) skb->cb), | ||
1205 | rtlpci->rxbuffersize, | ||
1206 | PCI_DMA_FROMDEVICE); | ||
1207 | kfree_skb(skb); | ||
1208 | } | ||
1209 | |||
1210 | pci_free_consistent(rtlpci->pdev, | ||
1211 | sizeof(*rtlpci->rx_ring[rx_queue_idx]. | ||
1212 | desc) * rtlpci->rxringcount, | ||
1213 | rtlpci->rx_ring[rx_queue_idx].desc, | ||
1214 | rtlpci->rx_ring[rx_queue_idx].dma); | ||
1215 | rtlpci->rx_ring[rx_queue_idx].desc = NULL; | ||
1216 | } | ||
1217 | } | ||
1218 | |||
1219 | static int _rtl_pci_init_trx_ring(struct ieee80211_hw *hw) | ||
1220 | { | ||
1221 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1222 | int ret; | ||
1223 | int i; | ||
1224 | |||
1225 | ret = _rtl_pci_init_rx_ring(hw); | ||
1226 | if (ret) | ||
1227 | return ret; | ||
1228 | |||
1229 | for (i = 0; i < RTL_PCI_MAX_TX_QUEUE_COUNT; i++) { | ||
1230 | ret = _rtl_pci_init_tx_ring(hw, i, | ||
1231 | rtlpci->txringcount[i]); | ||
1232 | if (ret) | ||
1233 | goto err_free_rings; | ||
1234 | } | ||
1235 | |||
1236 | return 0; | ||
1237 | |||
1238 | err_free_rings: | ||
1239 | _rtl_pci_free_rx_ring(rtlpci); | ||
1240 | |||
1241 | for (i = 0; i < RTL_PCI_MAX_TX_QUEUE_COUNT; i++) | ||
1242 | if (rtlpci->tx_ring[i].desc) | ||
1243 | _rtl_pci_free_tx_ring(hw, i); | ||
1244 | |||
1245 | return 1; | ||
1246 | } | ||
1247 | |||
1248 | static int _rtl_pci_deinit_trx_ring(struct ieee80211_hw *hw) | ||
1249 | { | ||
1250 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1251 | u32 i; | ||
1252 | |||
1253 | /*free rx rings */ | ||
1254 | _rtl_pci_free_rx_ring(rtlpci); | ||
1255 | |||
1256 | /*free tx rings */ | ||
1257 | for (i = 0; i < RTL_PCI_MAX_TX_QUEUE_COUNT; i++) | ||
1258 | _rtl_pci_free_tx_ring(hw, i); | ||
1259 | |||
1260 | return 0; | ||
1261 | } | ||
1262 | |||
1263 | int rtl_pci_reset_trx_ring(struct ieee80211_hw *hw) | ||
1264 | { | ||
1265 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1266 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1267 | int i, rx_queue_idx; | ||
1268 | unsigned long flags; | ||
1269 | u8 tmp_one = 1; | ||
1270 | |||
1271 | /*rx_queue_idx 0:RX_MPDU_QUEUE */ | ||
1272 | /*rx_queue_idx 1:RX_CMD_QUEUE */ | ||
1273 | for (rx_queue_idx = 0; rx_queue_idx < RTL_PCI_MAX_RX_QUEUE; | ||
1274 | rx_queue_idx++) { | ||
1275 | /* | ||
1276 | *force the rx_ring[RX_MPDU_QUEUE/ | ||
1277 | *RX_CMD_QUEUE].idx to the first one | ||
1278 | */ | ||
1279 | if (rtlpci->rx_ring[rx_queue_idx].desc) { | ||
1280 | struct rtl_rx_desc *entry = NULL; | ||
1281 | |||
1282 | for (i = 0; i < rtlpci->rxringcount; i++) { | ||
1283 | entry = &rtlpci->rx_ring[rx_queue_idx].desc[i]; | ||
1284 | rtlpriv->cfg->ops->set_desc((u8 *) entry, | ||
1285 | false, | ||
1286 | HW_DESC_RXOWN, | ||
1287 | (u8 *)&tmp_one); | ||
1288 | } | ||
1289 | rtlpci->rx_ring[rx_queue_idx].idx = 0; | ||
1290 | } | ||
1291 | } | ||
1292 | |||
1293 | /* | ||
1294 | *after reset, release previous pending packet, | ||
1295 | *and force the tx idx to the first one | ||
1296 | */ | ||
1297 | spin_lock_irqsave(&rtlpriv->locks.irq_th_lock, flags); | ||
1298 | for (i = 0; i < RTL_PCI_MAX_TX_QUEUE_COUNT; i++) { | ||
1299 | if (rtlpci->tx_ring[i].desc) { | ||
1300 | struct rtl8192_tx_ring *ring = &rtlpci->tx_ring[i]; | ||
1301 | |||
1302 | while (skb_queue_len(&ring->queue)) { | ||
1303 | struct rtl_tx_desc *entry = | ||
1304 | &ring->desc[ring->idx]; | ||
1305 | struct sk_buff *skb = | ||
1306 | __skb_dequeue(&ring->queue); | ||
1307 | |||
1308 | pci_unmap_single(rtlpci->pdev, | ||
1309 | rtlpriv->cfg->ops-> | ||
1310 | get_desc((u8 *) | ||
1311 | entry, | ||
1312 | true, | ||
1313 | HW_DESC_TXBUFF_ADDR), | ||
1314 | skb->len, PCI_DMA_TODEVICE); | ||
1315 | kfree_skb(skb); | ||
1316 | ring->idx = (ring->idx + 1) % ring->entries; | ||
1317 | } | ||
1318 | ring->idx = 0; | ||
1319 | } | ||
1320 | } | ||
1321 | |||
1322 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
1323 | |||
1324 | return 0; | ||
1325 | } | ||
1326 | |||
1327 | static bool rtl_pci_tx_chk_waitq_insert(struct ieee80211_hw *hw, | ||
1328 | struct sk_buff *skb) | ||
1329 | { | ||
1330 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1331 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
1332 | struct ieee80211_sta *sta = info->control.sta; | ||
1333 | struct rtl_sta_info *sta_entry = NULL; | ||
1334 | u8 tid = rtl_get_tid(skb); | ||
1335 | |||
1336 | if (!sta) | ||
1337 | return false; | ||
1338 | sta_entry = (struct rtl_sta_info *)sta->drv_priv; | ||
1339 | |||
1340 | if (!rtlpriv->rtlhal.earlymode_enable) | ||
1341 | return false; | ||
1342 | if (sta_entry->tids[tid].agg.agg_state != RTL_AGG_OPERATIONAL) | ||
1343 | return false; | ||
1344 | if (_rtl_mac_to_hwqueue(hw, skb) > VO_QUEUE) | ||
1345 | return false; | ||
1346 | if (tid > 7) | ||
1347 | return false; | ||
1348 | |||
1349 | /* maybe every tid should be checked */ | ||
1350 | if (!rtlpriv->link_info.higher_busytxtraffic[tid]) | ||
1351 | return false; | ||
1352 | |||
1353 | spin_lock_bh(&rtlpriv->locks.waitq_lock); | ||
1354 | skb_queue_tail(&rtlpriv->mac80211.skb_waitq[tid], skb); | ||
1355 | spin_unlock_bh(&rtlpriv->locks.waitq_lock); | ||
1356 | |||
1357 | return true; | ||
1358 | } | ||
1359 | |||
1360 | static int rtl_pci_tx(struct ieee80211_hw *hw, struct sk_buff *skb, | ||
1361 | struct rtl_tcb_desc *ptcb_desc) | ||
1362 | { | ||
1363 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1364 | struct rtl_sta_info *sta_entry = NULL; | ||
1365 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
1366 | struct ieee80211_sta *sta = info->control.sta; | ||
1367 | struct rtl8192_tx_ring *ring; | ||
1368 | struct rtl_tx_desc *pdesc; | ||
1369 | u8 idx; | ||
1370 | u8 hw_queue = _rtl_mac_to_hwqueue(hw, skb); | ||
1371 | unsigned long flags; | ||
1372 | struct ieee80211_hdr *hdr = rtl_get_hdr(skb); | ||
1373 | __le16 fc = rtl_get_fc(skb); | ||
1374 | u8 *pda_addr = hdr->addr1; | ||
1375 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1376 | /*ssn */ | ||
1377 | u8 tid = 0; | ||
1378 | u16 seq_number = 0; | ||
1379 | u8 own; | ||
1380 | u8 temp_one = 1; | ||
1381 | |||
1382 | if (ieee80211_is_auth(fc)) { | ||
1383 | RT_TRACE(rtlpriv, COMP_SEND, DBG_DMESG, ("MAC80211_LINKING\n")); | ||
1384 | rtl_ips_nic_on(hw); | ||
1385 | } | ||
1386 | |||
1387 | if (rtlpriv->psc.sw_ps_enabled) { | ||
1388 | if (ieee80211_is_data(fc) && !ieee80211_is_nullfunc(fc) && | ||
1389 | !ieee80211_has_pm(fc)) | ||
1390 | hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PM); | ||
1391 | } | ||
1392 | |||
1393 | rtl_action_proc(hw, skb, true); | ||
1394 | |||
1395 | if (is_multicast_ether_addr(pda_addr)) | ||
1396 | rtlpriv->stats.txbytesmulticast += skb->len; | ||
1397 | else if (is_broadcast_ether_addr(pda_addr)) | ||
1398 | rtlpriv->stats.txbytesbroadcast += skb->len; | ||
1399 | else | ||
1400 | rtlpriv->stats.txbytesunicast += skb->len; | ||
1401 | |||
1402 | spin_lock_irqsave(&rtlpriv->locks.irq_th_lock, flags); | ||
1403 | ring = &rtlpci->tx_ring[hw_queue]; | ||
1404 | if (hw_queue != BEACON_QUEUE) | ||
1405 | idx = (ring->idx + skb_queue_len(&ring->queue)) % | ||
1406 | ring->entries; | ||
1407 | else | ||
1408 | idx = 0; | ||
1409 | |||
1410 | pdesc = &ring->desc[idx]; | ||
1411 | own = (u8) rtlpriv->cfg->ops->get_desc((u8 *) pdesc, | ||
1412 | true, HW_DESC_OWN); | ||
1413 | |||
1414 | if ((own == 1) && (hw_queue != BEACON_QUEUE)) { | ||
1415 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1416 | ("No more TX desc@%d, ring->idx = %d," | ||
1417 | "idx = %d, skb_queue_len = 0x%d\n", | ||
1418 | hw_queue, ring->idx, idx, | ||
1419 | skb_queue_len(&ring->queue))); | ||
1420 | |||
1421 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
1422 | return skb->len; | ||
1423 | } | ||
1424 | |||
1425 | if (ieee80211_is_data_qos(fc)) { | ||
1426 | tid = rtl_get_tid(skb); | ||
1427 | if (sta) { | ||
1428 | sta_entry = (struct rtl_sta_info *)sta->drv_priv; | ||
1429 | seq_number = (le16_to_cpu(hdr->seq_ctrl) & | ||
1430 | IEEE80211_SCTL_SEQ) >> 4; | ||
1431 | seq_number += 1; | ||
1432 | |||
1433 | if (!ieee80211_has_morefrags(hdr->frame_control)) | ||
1434 | sta_entry->tids[tid].seq_number = seq_number; | ||
1435 | } | ||
1436 | } | ||
1437 | |||
1438 | if (ieee80211_is_data(fc)) | ||
1439 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_TX); | ||
1440 | |||
1441 | rtlpriv->cfg->ops->fill_tx_desc(hw, hdr, (u8 *)pdesc, | ||
1442 | info, skb, hw_queue, ptcb_desc); | ||
1443 | |||
1444 | __skb_queue_tail(&ring->queue, skb); | ||
1445 | |||
1446 | rtlpriv->cfg->ops->set_desc((u8 *)pdesc, true, | ||
1447 | HW_DESC_OWN, (u8 *)&temp_one); | ||
1448 | |||
1449 | |||
1450 | if ((ring->entries - skb_queue_len(&ring->queue)) < 2 && | ||
1451 | hw_queue != BEACON_QUEUE) { | ||
1452 | |||
1453 | RT_TRACE(rtlpriv, COMP_ERR, DBG_LOUD, | ||
1454 | ("less desc left, stop skb_queue@%d, " | ||
1455 | "ring->idx = %d," | ||
1456 | "idx = %d, skb_queue_len = 0x%d\n", | ||
1457 | hw_queue, ring->idx, idx, | ||
1458 | skb_queue_len(&ring->queue))); | ||
1459 | |||
1460 | ieee80211_stop_queue(hw, skb_get_queue_mapping(skb)); | ||
1461 | } | ||
1462 | |||
1463 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
1464 | |||
1465 | rtlpriv->cfg->ops->tx_polling(hw, hw_queue); | ||
1466 | |||
1467 | return 0; | ||
1468 | } | ||
1469 | |||
1470 | static void rtl_pci_flush(struct ieee80211_hw *hw, bool drop) | ||
1471 | { | ||
1472 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1473 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
1474 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1475 | u16 i = 0; | ||
1476 | int queue_id; | ||
1477 | struct rtl8192_tx_ring *ring; | ||
1478 | |||
1479 | for (queue_id = RTL_PCI_MAX_TX_QUEUE_COUNT - 1; queue_id >= 0;) { | ||
1480 | u32 queue_len; | ||
1481 | ring = &pcipriv->dev.tx_ring[queue_id]; | ||
1482 | queue_len = skb_queue_len(&ring->queue); | ||
1483 | if (queue_len == 0 || queue_id == BEACON_QUEUE || | ||
1484 | queue_id == TXCMD_QUEUE) { | ||
1485 | queue_id--; | ||
1486 | continue; | ||
1487 | } else { | ||
1488 | msleep(20); | ||
1489 | i++; | ||
1490 | } | ||
1491 | |||
1492 | /* we just wait 1s for all queues */ | ||
1493 | if (rtlpriv->psc.rfpwr_state == ERFOFF || | ||
1494 | is_hal_stop(rtlhal) || i >= 200) | ||
1495 | return; | ||
1496 | } | ||
1497 | } | ||
1498 | |||
1499 | static void rtl_pci_deinit(struct ieee80211_hw *hw) | ||
1500 | { | ||
1501 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1502 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1503 | |||
1504 | _rtl_pci_deinit_trx_ring(hw); | ||
1505 | |||
1506 | synchronize_irq(rtlpci->pdev->irq); | ||
1507 | tasklet_kill(&rtlpriv->works.irq_tasklet); | ||
1508 | |||
1509 | flush_workqueue(rtlpriv->works.rtl_wq); | ||
1510 | destroy_workqueue(rtlpriv->works.rtl_wq); | ||
1511 | |||
1512 | } | ||
1513 | |||
1514 | static int rtl_pci_init(struct ieee80211_hw *hw, struct pci_dev *pdev) | ||
1515 | { | ||
1516 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1517 | int err; | ||
1518 | |||
1519 | _rtl_pci_init_struct(hw, pdev); | ||
1520 | |||
1521 | err = _rtl_pci_init_trx_ring(hw); | ||
1522 | if (err) { | ||
1523 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1524 | ("tx ring initialization failed")); | ||
1525 | return err; | ||
1526 | } | ||
1527 | |||
1528 | return 1; | ||
1529 | } | ||
1530 | |||
1531 | static int rtl_pci_start(struct ieee80211_hw *hw) | ||
1532 | { | ||
1533 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1534 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1535 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1536 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1537 | |||
1538 | int err; | ||
1539 | |||
1540 | rtl_pci_reset_trx_ring(hw); | ||
1541 | |||
1542 | rtlpci->driver_is_goingto_unload = false; | ||
1543 | err = rtlpriv->cfg->ops->hw_init(hw); | ||
1544 | if (err) { | ||
1545 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1546 | ("Failed to config hardware!\n")); | ||
1547 | return err; | ||
1548 | } | ||
1549 | |||
1550 | rtlpriv->cfg->ops->enable_interrupt(hw); | ||
1551 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("enable_interrupt OK\n")); | ||
1552 | |||
1553 | rtl_init_rx_config(hw); | ||
1554 | |||
1555 | /*should after adapter start and interrupt enable. */ | ||
1556 | set_hal_start(rtlhal); | ||
1557 | |||
1558 | RT_CLEAR_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
1559 | |||
1560 | rtlpci->up_first_time = false; | ||
1561 | |||
1562 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("OK\n")); | ||
1563 | return 0; | ||
1564 | } | ||
1565 | |||
1566 | static void rtl_pci_stop(struct ieee80211_hw *hw) | ||
1567 | { | ||
1568 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1569 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1570 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1571 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1572 | unsigned long flags; | ||
1573 | u8 RFInProgressTimeOut = 0; | ||
1574 | |||
1575 | /* | ||
1576 | *should before disable interrrupt&adapter | ||
1577 | *and will do it immediately. | ||
1578 | */ | ||
1579 | set_hal_stop(rtlhal); | ||
1580 | |||
1581 | rtlpriv->cfg->ops->disable_interrupt(hw); | ||
1582 | |||
1583 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flags); | ||
1584 | while (ppsc->rfchange_inprogress) { | ||
1585 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flags); | ||
1586 | if (RFInProgressTimeOut > 100) { | ||
1587 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flags); | ||
1588 | break; | ||
1589 | } | ||
1590 | mdelay(1); | ||
1591 | RFInProgressTimeOut++; | ||
1592 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flags); | ||
1593 | } | ||
1594 | ppsc->rfchange_inprogress = true; | ||
1595 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flags); | ||
1596 | |||
1597 | rtlpci->driver_is_goingto_unload = true; | ||
1598 | rtlpriv->cfg->ops->hw_disable(hw); | ||
1599 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_POWER_OFF); | ||
1600 | |||
1601 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flags); | ||
1602 | ppsc->rfchange_inprogress = false; | ||
1603 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flags); | ||
1604 | |||
1605 | rtl_pci_enable_aspm(hw); | ||
1606 | } | ||
1607 | |||
1608 | static bool _rtl_pci_find_adapter(struct pci_dev *pdev, | ||
1609 | struct ieee80211_hw *hw) | ||
1610 | { | ||
1611 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1612 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
1613 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1614 | struct pci_dev *bridge_pdev = pdev->bus->self; | ||
1615 | u16 venderid; | ||
1616 | u16 deviceid; | ||
1617 | u8 revisionid; | ||
1618 | u16 irqline; | ||
1619 | u8 tmp; | ||
1620 | |||
1621 | pcipriv->ndis_adapter.pcibridge_vendor = PCI_BRIDGE_VENDOR_UNKNOWN; | ||
1622 | venderid = pdev->vendor; | ||
1623 | deviceid = pdev->device; | ||
1624 | pci_read_config_byte(pdev, 0x8, &revisionid); | ||
1625 | pci_read_config_word(pdev, 0x3C, &irqline); | ||
1626 | |||
1627 | /* PCI ID 0x10ec:0x8192 occurs for both RTL8192E, which uses | ||
1628 | * r8192e_pci, and RTL8192SE, which uses this driver. If the | ||
1629 | * revision ID is RTL_PCI_REVISION_ID_8192PCIE (0x01), then | ||
1630 | * the correct driver is r8192e_pci, thus this routine should | ||
1631 | * return false. | ||
1632 | */ | ||
1633 | if (deviceid == RTL_PCI_8192SE_DID && | ||
1634 | revisionid == RTL_PCI_REVISION_ID_8192PCIE) | ||
1635 | return false; | ||
1636 | |||
1637 | if (deviceid == RTL_PCI_8192_DID || | ||
1638 | deviceid == RTL_PCI_0044_DID || | ||
1639 | deviceid == RTL_PCI_0047_DID || | ||
1640 | deviceid == RTL_PCI_8192SE_DID || | ||
1641 | deviceid == RTL_PCI_8174_DID || | ||
1642 | deviceid == RTL_PCI_8173_DID || | ||
1643 | deviceid == RTL_PCI_8172_DID || | ||
1644 | deviceid == RTL_PCI_8171_DID) { | ||
1645 | switch (revisionid) { | ||
1646 | case RTL_PCI_REVISION_ID_8192PCIE: | ||
1647 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1648 | ("8192 PCI-E is found - " | ||
1649 | "vid/did=%x/%x\n", venderid, deviceid)); | ||
1650 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192E; | ||
1651 | break; | ||
1652 | case RTL_PCI_REVISION_ID_8192SE: | ||
1653 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1654 | ("8192SE is found - " | ||
1655 | "vid/did=%x/%x\n", venderid, deviceid)); | ||
1656 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192SE; | ||
1657 | break; | ||
1658 | default: | ||
1659 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1660 | ("Err: Unknown device - " | ||
1661 | "vid/did=%x/%x\n", venderid, deviceid)); | ||
1662 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192SE; | ||
1663 | break; | ||
1664 | |||
1665 | } | ||
1666 | } else if (deviceid == RTL_PCI_8192CET_DID || | ||
1667 | deviceid == RTL_PCI_8192CE_DID || | ||
1668 | deviceid == RTL_PCI_8191CE_DID || | ||
1669 | deviceid == RTL_PCI_8188CE_DID) { | ||
1670 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192CE; | ||
1671 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1672 | ("8192C PCI-E is found - " | ||
1673 | "vid/did=%x/%x\n", venderid, deviceid)); | ||
1674 | } else if (deviceid == RTL_PCI_8192DE_DID || | ||
1675 | deviceid == RTL_PCI_8192DE_DID2) { | ||
1676 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192DE; | ||
1677 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1678 | ("8192D PCI-E is found - " | ||
1679 | "vid/did=%x/%x\n", venderid, deviceid)); | ||
1680 | } else { | ||
1681 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1682 | ("Err: Unknown device -" | ||
1683 | " vid/did=%x/%x\n", venderid, deviceid)); | ||
1684 | |||
1685 | rtlhal->hw_type = RTL_DEFAULT_HARDWARE_TYPE; | ||
1686 | } | ||
1687 | |||
1688 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192DE) { | ||
1689 | if (revisionid == 0 || revisionid == 1) { | ||
1690 | if (revisionid == 0) { | ||
1691 | RT_TRACE(rtlpriv, COMP_INIT, | ||
1692 | DBG_LOUD, ("Find 92DE MAC0.\n")); | ||
1693 | rtlhal->interfaceindex = 0; | ||
1694 | } else if (revisionid == 1) { | ||
1695 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1696 | ("Find 92DE MAC1.\n")); | ||
1697 | rtlhal->interfaceindex = 1; | ||
1698 | } | ||
1699 | } else { | ||
1700 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1701 | ("Unknown device - " | ||
1702 | "VendorID/DeviceID=%x/%x, Revision=%x\n", | ||
1703 | venderid, deviceid, revisionid)); | ||
1704 | rtlhal->interfaceindex = 0; | ||
1705 | } | ||
1706 | } | ||
1707 | /*find bus info */ | ||
1708 | pcipriv->ndis_adapter.busnumber = pdev->bus->number; | ||
1709 | pcipriv->ndis_adapter.devnumber = PCI_SLOT(pdev->devfn); | ||
1710 | pcipriv->ndis_adapter.funcnumber = PCI_FUNC(pdev->devfn); | ||
1711 | |||
1712 | /*find bridge info */ | ||
1713 | pcipriv->ndis_adapter.pcibridge_vendorid = bridge_pdev->vendor; | ||
1714 | for (tmp = 0; tmp < PCI_BRIDGE_VENDOR_MAX; tmp++) { | ||
1715 | if (bridge_pdev->vendor == pcibridge_vendors[tmp]) { | ||
1716 | pcipriv->ndis_adapter.pcibridge_vendor = tmp; | ||
1717 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1718 | ("Pci Bridge Vendor is found index: %d\n", | ||
1719 | tmp)); | ||
1720 | break; | ||
1721 | } | ||
1722 | } | ||
1723 | |||
1724 | if (pcipriv->ndis_adapter.pcibridge_vendor != | ||
1725 | PCI_BRIDGE_VENDOR_UNKNOWN) { | ||
1726 | pcipriv->ndis_adapter.pcibridge_busnum = | ||
1727 | bridge_pdev->bus->number; | ||
1728 | pcipriv->ndis_adapter.pcibridge_devnum = | ||
1729 | PCI_SLOT(bridge_pdev->devfn); | ||
1730 | pcipriv->ndis_adapter.pcibridge_funcnum = | ||
1731 | PCI_FUNC(bridge_pdev->devfn); | ||
1732 | pcipriv->ndis_adapter.pcicfg_addrport = | ||
1733 | (pcipriv->ndis_adapter.pcibridge_busnum << 16) | | ||
1734 | (pcipriv->ndis_adapter.pcibridge_devnum << 11) | | ||
1735 | (pcipriv->ndis_adapter.pcibridge_funcnum << 8) | (1 << 31); | ||
1736 | pcipriv->ndis_adapter.pcibridge_pciehdr_offset = | ||
1737 | pci_pcie_cap(bridge_pdev); | ||
1738 | pcipriv->ndis_adapter.num4bytes = | ||
1739 | (pcipriv->ndis_adapter.pcibridge_pciehdr_offset + 0x10) / 4; | ||
1740 | |||
1741 | rtl_pci_get_linkcontrol_field(hw); | ||
1742 | |||
1743 | if (pcipriv->ndis_adapter.pcibridge_vendor == | ||
1744 | PCI_BRIDGE_VENDOR_AMD) { | ||
1745 | pcipriv->ndis_adapter.amd_l1_patch = | ||
1746 | rtl_pci_get_amd_l1_patch(hw); | ||
1747 | } | ||
1748 | } | ||
1749 | |||
1750 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1751 | ("pcidev busnumber:devnumber:funcnumber:" | ||
1752 | "vendor:link_ctl %d:%d:%d:%x:%x\n", | ||
1753 | pcipriv->ndis_adapter.busnumber, | ||
1754 | pcipriv->ndis_adapter.devnumber, | ||
1755 | pcipriv->ndis_adapter.funcnumber, | ||
1756 | pdev->vendor, pcipriv->ndis_adapter.linkctrl_reg)); | ||
1757 | |||
1758 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1759 | ("pci_bridge busnumber:devnumber:funcnumber:vendor:" | ||
1760 | "pcie_cap:link_ctl_reg:amd %d:%d:%d:%x:%x:%x:%x\n", | ||
1761 | pcipriv->ndis_adapter.pcibridge_busnum, | ||
1762 | pcipriv->ndis_adapter.pcibridge_devnum, | ||
1763 | pcipriv->ndis_adapter.pcibridge_funcnum, | ||
1764 | pcibridge_vendors[pcipriv->ndis_adapter.pcibridge_vendor], | ||
1765 | pcipriv->ndis_adapter.pcibridge_pciehdr_offset, | ||
1766 | pcipriv->ndis_adapter.pcibridge_linkctrlreg, | ||
1767 | pcipriv->ndis_adapter.amd_l1_patch)); | ||
1768 | |||
1769 | rtl_pci_parse_configuration(pdev, hw); | ||
1770 | |||
1771 | return true; | ||
1772 | } | ||
1773 | |||
1774 | int __devinit rtl_pci_probe(struct pci_dev *pdev, | ||
1775 | const struct pci_device_id *id) | ||
1776 | { | ||
1777 | struct ieee80211_hw *hw = NULL; | ||
1778 | |||
1779 | struct rtl_priv *rtlpriv = NULL; | ||
1780 | struct rtl_pci_priv *pcipriv = NULL; | ||
1781 | struct rtl_pci *rtlpci; | ||
1782 | unsigned long pmem_start, pmem_len, pmem_flags; | ||
1783 | int err; | ||
1784 | |||
1785 | err = pci_enable_device(pdev); | ||
1786 | if (err) { | ||
1787 | RT_ASSERT(false, | ||
1788 | ("%s : Cannot enable new PCI device\n", | ||
1789 | pci_name(pdev))); | ||
1790 | return err; | ||
1791 | } | ||
1792 | |||
1793 | if (!pci_set_dma_mask(pdev, DMA_BIT_MASK(32))) { | ||
1794 | if (pci_set_consistent_dma_mask(pdev, DMA_BIT_MASK(32))) { | ||
1795 | RT_ASSERT(false, ("Unable to obtain 32bit DMA " | ||
1796 | "for consistent allocations\n")); | ||
1797 | pci_disable_device(pdev); | ||
1798 | return -ENOMEM; | ||
1799 | } | ||
1800 | } | ||
1801 | |||
1802 | pci_set_master(pdev); | ||
1803 | |||
1804 | hw = ieee80211_alloc_hw(sizeof(struct rtl_pci_priv) + | ||
1805 | sizeof(struct rtl_priv), &rtl_ops); | ||
1806 | if (!hw) { | ||
1807 | RT_ASSERT(false, | ||
1808 | ("%s : ieee80211 alloc failed\n", pci_name(pdev))); | ||
1809 | err = -ENOMEM; | ||
1810 | goto fail1; | ||
1811 | } | ||
1812 | |||
1813 | SET_IEEE80211_DEV(hw, &pdev->dev); | ||
1814 | pci_set_drvdata(pdev, hw); | ||
1815 | |||
1816 | rtlpriv = hw->priv; | ||
1817 | pcipriv = (void *)rtlpriv->priv; | ||
1818 | pcipriv->dev.pdev = pdev; | ||
1819 | |||
1820 | /* init cfg & intf_ops */ | ||
1821 | rtlpriv->rtlhal.interface = INTF_PCI; | ||
1822 | rtlpriv->cfg = (struct rtl_hal_cfg *)(id->driver_data); | ||
1823 | rtlpriv->intf_ops = &rtl_pci_ops; | ||
1824 | |||
1825 | /* | ||
1826 | *init dbgp flags before all | ||
1827 | *other functions, because we will | ||
1828 | *use it in other funtions like | ||
1829 | *RT_TRACE/RT_PRINT/RTL_PRINT_DATA | ||
1830 | *you can not use these macro | ||
1831 | *before this | ||
1832 | */ | ||
1833 | rtl_dbgp_flag_init(hw); | ||
1834 | |||
1835 | /* MEM map */ | ||
1836 | err = pci_request_regions(pdev, KBUILD_MODNAME); | ||
1837 | if (err) { | ||
1838 | RT_ASSERT(false, ("Can't obtain PCI resources\n")); | ||
1839 | return err; | ||
1840 | } | ||
1841 | |||
1842 | pmem_start = pci_resource_start(pdev, rtlpriv->cfg->bar_id); | ||
1843 | pmem_len = pci_resource_len(pdev, rtlpriv->cfg->bar_id); | ||
1844 | pmem_flags = pci_resource_flags(pdev, rtlpriv->cfg->bar_id); | ||
1845 | |||
1846 | /*shared mem start */ | ||
1847 | rtlpriv->io.pci_mem_start = | ||
1848 | (unsigned long)pci_iomap(pdev, | ||
1849 | rtlpriv->cfg->bar_id, pmem_len); | ||
1850 | if (rtlpriv->io.pci_mem_start == 0) { | ||
1851 | RT_ASSERT(false, ("Can't map PCI mem\n")); | ||
1852 | goto fail2; | ||
1853 | } | ||
1854 | |||
1855 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1856 | ("mem mapped space: start: 0x%08lx len:%08lx " | ||
1857 | "flags:%08lx, after map:0x%08lx\n", | ||
1858 | pmem_start, pmem_len, pmem_flags, | ||
1859 | rtlpriv->io.pci_mem_start)); | ||
1860 | |||
1861 | /* Disable Clk Request */ | ||
1862 | pci_write_config_byte(pdev, 0x81, 0); | ||
1863 | /* leave D3 mode */ | ||
1864 | pci_write_config_byte(pdev, 0x44, 0); | ||
1865 | pci_write_config_byte(pdev, 0x04, 0x06); | ||
1866 | pci_write_config_byte(pdev, 0x04, 0x07); | ||
1867 | |||
1868 | /* find adapter */ | ||
1869 | if (!_rtl_pci_find_adapter(pdev, hw)) | ||
1870 | goto fail3; | ||
1871 | |||
1872 | /* Init IO handler */ | ||
1873 | _rtl_pci_io_handler_init(&pdev->dev, hw); | ||
1874 | |||
1875 | /*like read eeprom and so on */ | ||
1876 | rtlpriv->cfg->ops->read_eeprom_info(hw); | ||
1877 | |||
1878 | if (rtlpriv->cfg->ops->init_sw_vars(hw)) { | ||
1879 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1880 | ("Can't init_sw_vars.\n")); | ||
1881 | goto fail3; | ||
1882 | } | ||
1883 | |||
1884 | rtlpriv->cfg->ops->init_sw_leds(hw); | ||
1885 | |||
1886 | /*aspm */ | ||
1887 | rtl_pci_init_aspm(hw); | ||
1888 | |||
1889 | /* Init mac80211 sw */ | ||
1890 | err = rtl_init_core(hw); | ||
1891 | if (err) { | ||
1892 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1893 | ("Can't allocate sw for mac80211.\n")); | ||
1894 | goto fail3; | ||
1895 | } | ||
1896 | |||
1897 | /* Init PCI sw */ | ||
1898 | err = !rtl_pci_init(hw, pdev); | ||
1899 | if (err) { | ||
1900 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1901 | ("Failed to init PCI.\n")); | ||
1902 | goto fail3; | ||
1903 | } | ||
1904 | |||
1905 | err = ieee80211_register_hw(hw); | ||
1906 | if (err) { | ||
1907 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1908 | ("Can't register mac80211 hw.\n")); | ||
1909 | goto fail3; | ||
1910 | } else { | ||
1911 | rtlpriv->mac80211.mac80211_registered = 1; | ||
1912 | } | ||
1913 | |||
1914 | err = sysfs_create_group(&pdev->dev.kobj, &rtl_attribute_group); | ||
1915 | if (err) { | ||
1916 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1917 | ("failed to create sysfs device attributes\n")); | ||
1918 | goto fail3; | ||
1919 | } | ||
1920 | |||
1921 | /*init rfkill */ | ||
1922 | rtl_init_rfkill(hw); | ||
1923 | |||
1924 | rtlpci = rtl_pcidev(pcipriv); | ||
1925 | err = request_irq(rtlpci->pdev->irq, &_rtl_pci_interrupt, | ||
1926 | IRQF_SHARED, KBUILD_MODNAME, hw); | ||
1927 | if (err) { | ||
1928 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1929 | ("%s: failed to register IRQ handler\n", | ||
1930 | wiphy_name(hw->wiphy))); | ||
1931 | goto fail3; | ||
1932 | } else { | ||
1933 | rtlpci->irq_alloc = 1; | ||
1934 | } | ||
1935 | |||
1936 | set_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status); | ||
1937 | return 0; | ||
1938 | |||
1939 | fail3: | ||
1940 | pci_set_drvdata(pdev, NULL); | ||
1941 | rtl_deinit_core(hw); | ||
1942 | _rtl_pci_io_handler_release(hw); | ||
1943 | ieee80211_free_hw(hw); | ||
1944 | |||
1945 | if (rtlpriv->io.pci_mem_start != 0) | ||
1946 | pci_iounmap(pdev, (void __iomem *)rtlpriv->io.pci_mem_start); | ||
1947 | |||
1948 | fail2: | ||
1949 | pci_release_regions(pdev); | ||
1950 | |||
1951 | fail1: | ||
1952 | |||
1953 | pci_disable_device(pdev); | ||
1954 | |||
1955 | return -ENODEV; | ||
1956 | |||
1957 | } | ||
1958 | EXPORT_SYMBOL(rtl_pci_probe); | ||
1959 | |||
1960 | void rtl_pci_disconnect(struct pci_dev *pdev) | ||
1961 | { | ||
1962 | struct ieee80211_hw *hw = pci_get_drvdata(pdev); | ||
1963 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
1964 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1965 | struct rtl_pci *rtlpci = rtl_pcidev(pcipriv); | ||
1966 | struct rtl_mac *rtlmac = rtl_mac(rtlpriv); | ||
1967 | |||
1968 | clear_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status); | ||
1969 | |||
1970 | sysfs_remove_group(&pdev->dev.kobj, &rtl_attribute_group); | ||
1971 | |||
1972 | /*ieee80211_unregister_hw will call ops_stop */ | ||
1973 | if (rtlmac->mac80211_registered == 1) { | ||
1974 | ieee80211_unregister_hw(hw); | ||
1975 | rtlmac->mac80211_registered = 0; | ||
1976 | } else { | ||
1977 | rtl_deinit_deferred_work(hw); | ||
1978 | rtlpriv->intf_ops->adapter_stop(hw); | ||
1979 | } | ||
1980 | |||
1981 | /*deinit rfkill */ | ||
1982 | rtl_deinit_rfkill(hw); | ||
1983 | |||
1984 | rtl_pci_deinit(hw); | ||
1985 | rtl_deinit_core(hw); | ||
1986 | _rtl_pci_io_handler_release(hw); | ||
1987 | rtlpriv->cfg->ops->deinit_sw_vars(hw); | ||
1988 | |||
1989 | if (rtlpci->irq_alloc) { | ||
1990 | free_irq(rtlpci->pdev->irq, hw); | ||
1991 | rtlpci->irq_alloc = 0; | ||
1992 | } | ||
1993 | |||
1994 | if (rtlpriv->io.pci_mem_start != 0) { | ||
1995 | pci_iounmap(pdev, (void __iomem *)rtlpriv->io.pci_mem_start); | ||
1996 | pci_release_regions(pdev); | ||
1997 | } | ||
1998 | |||
1999 | pci_disable_device(pdev); | ||
2000 | |||
2001 | rtl_pci_disable_aspm(hw); | ||
2002 | |||
2003 | pci_set_drvdata(pdev, NULL); | ||
2004 | |||
2005 | ieee80211_free_hw(hw); | ||
2006 | } | ||
2007 | EXPORT_SYMBOL(rtl_pci_disconnect); | ||
2008 | |||
2009 | /*************************************** | ||
2010 | kernel pci power state define: | ||
2011 | PCI_D0 ((pci_power_t __force) 0) | ||
2012 | PCI_D1 ((pci_power_t __force) 1) | ||
2013 | PCI_D2 ((pci_power_t __force) 2) | ||
2014 | PCI_D3hot ((pci_power_t __force) 3) | ||
2015 | PCI_D3cold ((pci_power_t __force) 4) | ||
2016 | PCI_UNKNOWN ((pci_power_t __force) 5) | ||
2017 | |||
2018 | This function is called when system | ||
2019 | goes into suspend state mac80211 will | ||
2020 | call rtl_mac_stop() from the mac80211 | ||
2021 | suspend function first, So there is | ||
2022 | no need to call hw_disable here. | ||
2023 | ****************************************/ | ||
2024 | int rtl_pci_suspend(struct pci_dev *pdev, pm_message_t state) | ||
2025 | { | ||
2026 | struct ieee80211_hw *hw = pci_get_drvdata(pdev); | ||
2027 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2028 | |||
2029 | rtlpriv->cfg->ops->hw_suspend(hw); | ||
2030 | rtl_deinit_rfkill(hw); | ||
2031 | |||
2032 | pci_save_state(pdev); | ||
2033 | pci_disable_device(pdev); | ||
2034 | pci_set_power_state(pdev, PCI_D3hot); | ||
2035 | return 0; | ||
2036 | } | ||
2037 | EXPORT_SYMBOL(rtl_pci_suspend); | ||
2038 | |||
2039 | int rtl_pci_resume(struct pci_dev *pdev) | ||
2040 | { | ||
2041 | int ret; | ||
2042 | struct ieee80211_hw *hw = pci_get_drvdata(pdev); | ||
2043 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2044 | |||
2045 | pci_set_power_state(pdev, PCI_D0); | ||
2046 | ret = pci_enable_device(pdev); | ||
2047 | if (ret) { | ||
2048 | RT_ASSERT(false, ("ERR: <======\n")); | ||
2049 | return ret; | ||
2050 | } | ||
2051 | |||
2052 | pci_restore_state(pdev); | ||
2053 | |||
2054 | rtlpriv->cfg->ops->hw_resume(hw); | ||
2055 | rtl_init_rfkill(hw); | ||
2056 | return 0; | ||
2057 | } | ||
2058 | EXPORT_SYMBOL(rtl_pci_resume); | ||
2059 | |||
2060 | struct rtl_intf_ops rtl_pci_ops = { | ||
2061 | .read_efuse_byte = read_efuse_byte, | ||
2062 | .adapter_start = rtl_pci_start, | ||
2063 | .adapter_stop = rtl_pci_stop, | ||
2064 | .adapter_tx = rtl_pci_tx, | ||
2065 | .flush = rtl_pci_flush, | ||
2066 | .reset_trx_ring = rtl_pci_reset_trx_ring, | ||
2067 | .waitq_insert = rtl_pci_tx_chk_waitq_insert, | ||
2068 | |||
2069 | .disable_aspm = rtl_pci_disable_aspm, | ||
2070 | .enable_aspm = rtl_pci_enable_aspm, | ||
2071 | }; | ||
diff --git a/drivers/net/wireless/rtlwifi/pci.h b/drivers/net/wireless/rtlwifi/pci.h new file mode 100644 index 000000000000..671b1f5aa0cf --- /dev/null +++ b/drivers/net/wireless/rtlwifi/pci.h | |||
@@ -0,0 +1,313 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_PCI_H__ | ||
31 | #define __RTL_PCI_H__ | ||
32 | |||
33 | #include <linux/pci.h> | ||
34 | /* | ||
35 | 1: MSDU packet queue, | ||
36 | 2: Rx Command Queue | ||
37 | */ | ||
38 | #define RTL_PCI_RX_MPDU_QUEUE 0 | ||
39 | #define RTL_PCI_RX_CMD_QUEUE 1 | ||
40 | #define RTL_PCI_MAX_RX_QUEUE 2 | ||
41 | |||
42 | #define RTL_PCI_MAX_RX_COUNT 64 | ||
43 | #define RTL_PCI_MAX_TX_QUEUE_COUNT 9 | ||
44 | |||
45 | #define RT_TXDESC_NUM 128 | ||
46 | #define RT_TXDESC_NUM_BE_QUEUE 256 | ||
47 | |||
48 | #define BK_QUEUE 0 | ||
49 | #define BE_QUEUE 1 | ||
50 | #define VI_QUEUE 2 | ||
51 | #define VO_QUEUE 3 | ||
52 | #define BEACON_QUEUE 4 | ||
53 | #define TXCMD_QUEUE 5 | ||
54 | #define MGNT_QUEUE 6 | ||
55 | #define HIGH_QUEUE 7 | ||
56 | #define HCCA_QUEUE 8 | ||
57 | |||
58 | #define RTL_PCI_DEVICE(vend, dev, cfg) \ | ||
59 | .vendor = (vend), \ | ||
60 | .device = (dev), \ | ||
61 | .subvendor = PCI_ANY_ID, \ | ||
62 | .subdevice = PCI_ANY_ID,\ | ||
63 | .driver_data = (kernel_ulong_t)&(cfg) | ||
64 | |||
65 | #define INTEL_VENDOR_ID 0x8086 | ||
66 | #define SIS_VENDOR_ID 0x1039 | ||
67 | #define ATI_VENDOR_ID 0x1002 | ||
68 | #define ATI_DEVICE_ID 0x7914 | ||
69 | #define AMD_VENDOR_ID 0x1022 | ||
70 | |||
71 | #define PCI_MAX_BRIDGE_NUMBER 255 | ||
72 | #define PCI_MAX_DEVICES 32 | ||
73 | #define PCI_MAX_FUNCTION 8 | ||
74 | |||
75 | #define PCI_CONF_ADDRESS 0x0CF8 /*PCI Configuration Space Address */ | ||
76 | #define PCI_CONF_DATA 0x0CFC /*PCI Configuration Space Data */ | ||
77 | |||
78 | #define PCI_CLASS_BRIDGE_DEV 0x06 | ||
79 | #define PCI_SUBCLASS_BR_PCI_TO_PCI 0x04 | ||
80 | #define PCI_CAPABILITY_ID_PCI_EXPRESS 0x10 | ||
81 | #define PCI_CAP_ID_EXP 0x10 | ||
82 | |||
83 | #define U1DONTCARE 0xFF | ||
84 | #define U2DONTCARE 0xFFFF | ||
85 | #define U4DONTCARE 0xFFFFFFFF | ||
86 | |||
87 | #define RTL_PCI_8192_DID 0x8192 /*8192 PCI-E */ | ||
88 | #define RTL_PCI_8192SE_DID 0x8192 /*8192 SE */ | ||
89 | #define RTL_PCI_8174_DID 0x8174 /*8192 SE */ | ||
90 | #define RTL_PCI_8173_DID 0x8173 /*8191 SE Crab */ | ||
91 | #define RTL_PCI_8172_DID 0x8172 /*8191 SE RE */ | ||
92 | #define RTL_PCI_8171_DID 0x8171 /*8191 SE Unicron */ | ||
93 | #define RTL_PCI_0045_DID 0x0045 /*8190 PCI for Ceraga */ | ||
94 | #define RTL_PCI_0046_DID 0x0046 /*8190 Cardbus for Ceraga */ | ||
95 | #define RTL_PCI_0044_DID 0x0044 /*8192e PCIE for Ceraga */ | ||
96 | #define RTL_PCI_0047_DID 0x0047 /*8192e Express Card for Ceraga */ | ||
97 | #define RTL_PCI_700F_DID 0x700F | ||
98 | #define RTL_PCI_701F_DID 0x701F | ||
99 | #define RTL_PCI_DLINK_DID 0x3304 | ||
100 | #define RTL_PCI_8192CET_DID 0x8191 /*8192ce */ | ||
101 | #define RTL_PCI_8192CE_DID 0x8178 /*8192ce */ | ||
102 | #define RTL_PCI_8191CE_DID 0x8177 /*8192ce */ | ||
103 | #define RTL_PCI_8188CE_DID 0x8176 /*8192ce */ | ||
104 | #define RTL_PCI_8192CU_DID 0x8191 /*8192ce */ | ||
105 | #define RTL_PCI_8192DE_DID 0x8193 /*8192de */ | ||
106 | #define RTL_PCI_8192DE_DID2 0x002B /*92DE*/ | ||
107 | |||
108 | /*8192 support 16 pages of IO registers*/ | ||
109 | #define RTL_MEM_MAPPED_IO_RANGE_8190PCI 0x1000 | ||
110 | #define RTL_MEM_MAPPED_IO_RANGE_8192PCIE 0x4000 | ||
111 | #define RTL_MEM_MAPPED_IO_RANGE_8192SE 0x4000 | ||
112 | #define RTL_MEM_MAPPED_IO_RANGE_8192CE 0x4000 | ||
113 | #define RTL_MEM_MAPPED_IO_RANGE_8192DE 0x4000 | ||
114 | |||
115 | #define RTL_PCI_REVISION_ID_8190PCI 0x00 | ||
116 | #define RTL_PCI_REVISION_ID_8192PCIE 0x01 | ||
117 | #define RTL_PCI_REVISION_ID_8192SE 0x10 | ||
118 | #define RTL_PCI_REVISION_ID_8192CE 0x1 | ||
119 | #define RTL_PCI_REVISION_ID_8192DE 0x0 | ||
120 | |||
121 | #define RTL_DEFAULT_HARDWARE_TYPE HARDWARE_TYPE_RTL8192CE | ||
122 | |||
123 | enum pci_bridge_vendor { | ||
124 | PCI_BRIDGE_VENDOR_INTEL = 0x0, /*0b'0000,0001 */ | ||
125 | PCI_BRIDGE_VENDOR_ATI, /*0b'0000,0010*/ | ||
126 | PCI_BRIDGE_VENDOR_AMD, /*0b'0000,0100*/ | ||
127 | PCI_BRIDGE_VENDOR_SIS, /*0b'0000,1000*/ | ||
128 | PCI_BRIDGE_VENDOR_UNKNOWN, /*0b'0100,0000*/ | ||
129 | PCI_BRIDGE_VENDOR_MAX, | ||
130 | }; | ||
131 | |||
132 | struct rtl_pci_capabilities_header { | ||
133 | u8 capability_id; | ||
134 | u8 next; | ||
135 | }; | ||
136 | |||
137 | struct rtl_rx_desc { | ||
138 | u32 dword[8]; | ||
139 | } __packed; | ||
140 | |||
141 | struct rtl_tx_desc { | ||
142 | u32 dword[16]; | ||
143 | } __packed; | ||
144 | |||
145 | struct rtl_tx_cmd_desc { | ||
146 | u32 dword[16]; | ||
147 | } __packed; | ||
148 | |||
149 | struct rtl8192_tx_ring { | ||
150 | struct rtl_tx_desc *desc; | ||
151 | dma_addr_t dma; | ||
152 | unsigned int idx; | ||
153 | unsigned int entries; | ||
154 | struct sk_buff_head queue; | ||
155 | }; | ||
156 | |||
157 | struct rtl8192_rx_ring { | ||
158 | struct rtl_rx_desc *desc; | ||
159 | dma_addr_t dma; | ||
160 | unsigned int idx; | ||
161 | struct sk_buff *rx_buf[RTL_PCI_MAX_RX_COUNT]; | ||
162 | }; | ||
163 | |||
164 | struct rtl_pci { | ||
165 | struct pci_dev *pdev; | ||
166 | |||
167 | bool driver_is_goingto_unload; | ||
168 | bool up_first_time; | ||
169 | bool first_init; | ||
170 | bool being_init_adapter; | ||
171 | bool init_ready; | ||
172 | bool irq_enabled; | ||
173 | |||
174 | /*Tx */ | ||
175 | struct rtl8192_tx_ring tx_ring[RTL_PCI_MAX_TX_QUEUE_COUNT]; | ||
176 | int txringcount[RTL_PCI_MAX_TX_QUEUE_COUNT]; | ||
177 | u32 transmit_config; | ||
178 | |||
179 | /*Rx */ | ||
180 | struct rtl8192_rx_ring rx_ring[RTL_PCI_MAX_RX_QUEUE]; | ||
181 | int rxringcount; | ||
182 | u16 rxbuffersize; | ||
183 | u32 receive_config; | ||
184 | |||
185 | /*irq */ | ||
186 | u8 irq_alloc; | ||
187 | u32 irq_mask[2]; | ||
188 | |||
189 | /*Bcn control register setting */ | ||
190 | u32 reg_bcn_ctrl_val; | ||
191 | |||
192 | /*ASPM*/ u8 const_pci_aspm; | ||
193 | u8 const_amdpci_aspm; | ||
194 | u8 const_hwsw_rfoff_d3; | ||
195 | u8 const_support_pciaspm; | ||
196 | /*pci-e bridge */ | ||
197 | u8 const_hostpci_aspm_setting; | ||
198 | /*pci-e device */ | ||
199 | u8 const_devicepci_aspm_setting; | ||
200 | /*If it supports ASPM, Offset[560h] = 0x40, | ||
201 | otherwise Offset[560h] = 0x00. */ | ||
202 | bool support_aspm; | ||
203 | bool support_backdoor; | ||
204 | |||
205 | /*QOS & EDCA */ | ||
206 | enum acm_method acm_method; | ||
207 | |||
208 | u16 shortretry_limit; | ||
209 | u16 longretry_limit; | ||
210 | }; | ||
211 | |||
212 | struct mp_adapter { | ||
213 | u8 linkctrl_reg; | ||
214 | |||
215 | u8 busnumber; | ||
216 | u8 devnumber; | ||
217 | u8 funcnumber; | ||
218 | |||
219 | u8 pcibridge_busnum; | ||
220 | u8 pcibridge_devnum; | ||
221 | u8 pcibridge_funcnum; | ||
222 | |||
223 | u8 pcibridge_vendor; | ||
224 | u16 pcibridge_vendorid; | ||
225 | u16 pcibridge_deviceid; | ||
226 | |||
227 | u32 pcicfg_addrport; | ||
228 | u8 num4bytes; | ||
229 | |||
230 | u8 pcibridge_pciehdr_offset; | ||
231 | u8 pcibridge_linkctrlreg; | ||
232 | |||
233 | bool amd_l1_patch; | ||
234 | }; | ||
235 | |||
236 | struct rtl_pci_priv { | ||
237 | struct rtl_pci dev; | ||
238 | struct mp_adapter ndis_adapter; | ||
239 | struct rtl_led_ctl ledctl; | ||
240 | struct bt_coexist_info bt_coexist; | ||
241 | }; | ||
242 | |||
243 | #define rtl_pcipriv(hw) (((struct rtl_pci_priv *)(rtl_priv(hw))->priv)) | ||
244 | #define rtl_pcidev(pcipriv) (&((pcipriv)->dev)) | ||
245 | |||
246 | int rtl_pci_reset_trx_ring(struct ieee80211_hw *hw); | ||
247 | |||
248 | extern struct rtl_intf_ops rtl_pci_ops; | ||
249 | |||
250 | int __devinit rtl_pci_probe(struct pci_dev *pdev, | ||
251 | const struct pci_device_id *id); | ||
252 | void rtl_pci_disconnect(struct pci_dev *pdev); | ||
253 | int rtl_pci_suspend(struct pci_dev *pdev, pm_message_t state); | ||
254 | int rtl_pci_resume(struct pci_dev *pdev); | ||
255 | |||
256 | static inline u8 pci_read8_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
257 | { | ||
258 | return readb((u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
259 | } | ||
260 | |||
261 | static inline u16 pci_read16_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
262 | { | ||
263 | return readw((u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
264 | } | ||
265 | |||
266 | static inline u32 pci_read32_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
267 | { | ||
268 | return readl((u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
269 | } | ||
270 | |||
271 | static inline void pci_write8_async(struct rtl_priv *rtlpriv, u32 addr, u8 val) | ||
272 | { | ||
273 | writeb(val, (u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
274 | } | ||
275 | |||
276 | static inline void pci_write16_async(struct rtl_priv *rtlpriv, | ||
277 | u32 addr, u16 val) | ||
278 | { | ||
279 | writew(val, (u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
280 | } | ||
281 | |||
282 | static inline void pci_write32_async(struct rtl_priv *rtlpriv, | ||
283 | u32 addr, u32 val) | ||
284 | { | ||
285 | writel(val, (u8 __iomem *) rtlpriv->io.pci_mem_start + addr); | ||
286 | } | ||
287 | |||
288 | static inline void rtl_pci_raw_write_port_ulong(u32 port, u32 val) | ||
289 | { | ||
290 | outl(val, port); | ||
291 | } | ||
292 | |||
293 | static inline void rtl_pci_raw_write_port_uchar(u32 port, u8 val) | ||
294 | { | ||
295 | outb(val, port); | ||
296 | } | ||
297 | |||
298 | static inline void rtl_pci_raw_read_port_uchar(u32 port, u8 *pval) | ||
299 | { | ||
300 | *pval = inb(port); | ||
301 | } | ||
302 | |||
303 | static inline void rtl_pci_raw_read_port_ushort(u32 port, u16 *pval) | ||
304 | { | ||
305 | *pval = inw(port); | ||
306 | } | ||
307 | |||
308 | static inline void rtl_pci_raw_read_port_ulong(u32 port, u32 *pval) | ||
309 | { | ||
310 | *pval = inl(port); | ||
311 | } | ||
312 | |||
313 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/ps.c b/drivers/net/wireless/rtlwifi/ps.c new file mode 100644 index 000000000000..39b0297ce925 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/ps.c | |||
@@ -0,0 +1,708 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "base.h" | ||
32 | #include "ps.h" | ||
33 | |||
34 | bool rtl_ps_enable_nic(struct ieee80211_hw *hw) | ||
35 | { | ||
36 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
37 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
38 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
39 | |||
40 | /*<1> reset trx ring */ | ||
41 | if (rtlhal->interface == INTF_PCI) | ||
42 | rtlpriv->intf_ops->reset_trx_ring(hw); | ||
43 | |||
44 | if (is_hal_stop(rtlhal)) | ||
45 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
46 | ("Driver is already down!\n")); | ||
47 | |||
48 | /*<2> Enable Adapter */ | ||
49 | rtlpriv->cfg->ops->hw_init(hw); | ||
50 | RT_CLEAR_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
51 | |||
52 | /*<3> Enable Interrupt */ | ||
53 | rtlpriv->cfg->ops->enable_interrupt(hw); | ||
54 | |||
55 | /*<enable timer> */ | ||
56 | rtl_watch_dog_timer_callback((unsigned long)hw); | ||
57 | |||
58 | return true; | ||
59 | } | ||
60 | EXPORT_SYMBOL(rtl_ps_enable_nic); | ||
61 | |||
62 | bool rtl_ps_disable_nic(struct ieee80211_hw *hw) | ||
63 | { | ||
64 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
65 | |||
66 | /*<1> Stop all timer */ | ||
67 | rtl_deinit_deferred_work(hw); | ||
68 | |||
69 | /*<2> Disable Interrupt */ | ||
70 | rtlpriv->cfg->ops->disable_interrupt(hw); | ||
71 | |||
72 | /*<3> Disable Adapter */ | ||
73 | rtlpriv->cfg->ops->hw_disable(hw); | ||
74 | |||
75 | return true; | ||
76 | } | ||
77 | EXPORT_SYMBOL(rtl_ps_disable_nic); | ||
78 | |||
79 | bool rtl_ps_set_rf_state(struct ieee80211_hw *hw, | ||
80 | enum rf_pwrstate state_toset, | ||
81 | u32 changesource, bool protect_or_not) | ||
82 | { | ||
83 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
84 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
85 | enum rf_pwrstate rtstate; | ||
86 | bool actionallowed = false; | ||
87 | u16 rfwait_cnt = 0; | ||
88 | unsigned long flag; | ||
89 | |||
90 | /*protect_or_not = true; */ | ||
91 | |||
92 | if (protect_or_not) | ||
93 | goto no_protect; | ||
94 | |||
95 | /* | ||
96 | *Only one thread can change | ||
97 | *the RF state at one time, and others | ||
98 | *should wait to be executed. | ||
99 | */ | ||
100 | while (true) { | ||
101 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
102 | if (ppsc->rfchange_inprogress) { | ||
103 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, | ||
104 | flag); | ||
105 | |||
106 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
107 | ("RF Change in progress!" | ||
108 | "Wait to set..state_toset(%d).\n", | ||
109 | state_toset)); | ||
110 | |||
111 | /* Set RF after the previous action is done. */ | ||
112 | while (ppsc->rfchange_inprogress) { | ||
113 | rfwait_cnt++; | ||
114 | mdelay(1); | ||
115 | |||
116 | /* | ||
117 | *Wait too long, return false to avoid | ||
118 | *to be stuck here. | ||
119 | */ | ||
120 | if (rfwait_cnt > 100) | ||
121 | return false; | ||
122 | } | ||
123 | } else { | ||
124 | ppsc->rfchange_inprogress = true; | ||
125 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, | ||
126 | flag); | ||
127 | break; | ||
128 | } | ||
129 | } | ||
130 | |||
131 | no_protect: | ||
132 | rtstate = ppsc->rfpwr_state; | ||
133 | |||
134 | switch (state_toset) { | ||
135 | case ERFON: | ||
136 | ppsc->rfoff_reason &= (~changesource); | ||
137 | |||
138 | if ((changesource == RF_CHANGE_BY_HW) && | ||
139 | (ppsc->hwradiooff == true)) { | ||
140 | ppsc->hwradiooff = false; | ||
141 | } | ||
142 | |||
143 | if (!ppsc->rfoff_reason) { | ||
144 | ppsc->rfoff_reason = 0; | ||
145 | actionallowed = true; | ||
146 | } | ||
147 | |||
148 | break; | ||
149 | |||
150 | case ERFOFF: | ||
151 | |||
152 | if ((changesource == RF_CHANGE_BY_HW) | ||
153 | && (ppsc->hwradiooff == false)) { | ||
154 | ppsc->hwradiooff = true; | ||
155 | } | ||
156 | |||
157 | ppsc->rfoff_reason |= changesource; | ||
158 | actionallowed = true; | ||
159 | break; | ||
160 | |||
161 | case ERFSLEEP: | ||
162 | ppsc->rfoff_reason |= changesource; | ||
163 | actionallowed = true; | ||
164 | break; | ||
165 | |||
166 | default: | ||
167 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
168 | ("switch case not process\n")); | ||
169 | break; | ||
170 | } | ||
171 | |||
172 | if (actionallowed) | ||
173 | rtlpriv->cfg->ops->set_rf_power_state(hw, state_toset); | ||
174 | |||
175 | if (!protect_or_not) { | ||
176 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
177 | ppsc->rfchange_inprogress = false; | ||
178 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
179 | } | ||
180 | |||
181 | return actionallowed; | ||
182 | } | ||
183 | EXPORT_SYMBOL(rtl_ps_set_rf_state); | ||
184 | |||
185 | static void _rtl_ps_inactive_ps(struct ieee80211_hw *hw) | ||
186 | { | ||
187 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
188 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
189 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
190 | |||
191 | ppsc->swrf_processing = true; | ||
192 | |||
193 | if (ppsc->inactive_pwrstate == ERFON && | ||
194 | rtlhal->interface == INTF_PCI) { | ||
195 | if ((ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM) && | ||
196 | RT_IN_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM) && | ||
197 | rtlhal->interface == INTF_PCI) { | ||
198 | rtlpriv->intf_ops->disable_aspm(hw); | ||
199 | RT_CLEAR_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM); | ||
200 | } | ||
201 | } | ||
202 | |||
203 | rtl_ps_set_rf_state(hw, ppsc->inactive_pwrstate, | ||
204 | RF_CHANGE_BY_IPS, false); | ||
205 | |||
206 | if (ppsc->inactive_pwrstate == ERFOFF && | ||
207 | rtlhal->interface == INTF_PCI) { | ||
208 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM && | ||
209 | !RT_IN_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM)) { | ||
210 | rtlpriv->intf_ops->enable_aspm(hw); | ||
211 | RT_SET_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM); | ||
212 | } | ||
213 | } | ||
214 | |||
215 | ppsc->swrf_processing = false; | ||
216 | } | ||
217 | |||
218 | void rtl_ips_nic_off_wq_callback(void *data) | ||
219 | { | ||
220 | struct rtl_works *rtlworks = | ||
221 | container_of_dwork_rtl(data, struct rtl_works, ips_nic_off_wq); | ||
222 | struct ieee80211_hw *hw = rtlworks->hw; | ||
223 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
224 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
225 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
226 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
227 | enum rf_pwrstate rtstate; | ||
228 | |||
229 | if (mac->opmode != NL80211_IFTYPE_STATION) { | ||
230 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
231 | ("not station return\n")); | ||
232 | return; | ||
233 | } | ||
234 | |||
235 | if (mac->link_state > MAC80211_NOLINK) | ||
236 | return; | ||
237 | |||
238 | if (is_hal_stop(rtlhal)) | ||
239 | return; | ||
240 | |||
241 | if (rtlpriv->sec.being_setkey) | ||
242 | return; | ||
243 | |||
244 | if (ppsc->inactiveps) { | ||
245 | rtstate = ppsc->rfpwr_state; | ||
246 | |||
247 | /* | ||
248 | *Do not enter IPS in the following conditions: | ||
249 | *(1) RF is already OFF or Sleep | ||
250 | *(2) swrf_processing (indicates the IPS is still under going) | ||
251 | *(3) Connectted (only disconnected can trigger IPS) | ||
252 | *(4) IBSS (send Beacon) | ||
253 | *(5) AP mode (send Beacon) | ||
254 | *(6) monitor mode (rcv packet) | ||
255 | */ | ||
256 | |||
257 | if (rtstate == ERFON && | ||
258 | !ppsc->swrf_processing && | ||
259 | (mac->link_state == MAC80211_NOLINK) && | ||
260 | !mac->act_scanning) { | ||
261 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
262 | ("IPSEnter(): Turn off RF.\n")); | ||
263 | |||
264 | ppsc->inactive_pwrstate = ERFOFF; | ||
265 | ppsc->in_powersavemode = true; | ||
266 | |||
267 | /*rtl_pci_reset_trx_ring(hw); */ | ||
268 | _rtl_ps_inactive_ps(hw); | ||
269 | } | ||
270 | } | ||
271 | } | ||
272 | |||
273 | void rtl_ips_nic_off(struct ieee80211_hw *hw) | ||
274 | { | ||
275 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
276 | |||
277 | /* | ||
278 | *because when link with ap, mac80211 will ask us | ||
279 | *to disable nic quickly after scan before linking, | ||
280 | *this will cause link failed, so we delay 100ms here | ||
281 | */ | ||
282 | queue_delayed_work(rtlpriv->works.rtl_wq, | ||
283 | &rtlpriv->works.ips_nic_off_wq, MSECS(100)); | ||
284 | } | ||
285 | |||
286 | void rtl_ips_nic_on(struct ieee80211_hw *hw) | ||
287 | { | ||
288 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
289 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
290 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
291 | enum rf_pwrstate rtstate; | ||
292 | unsigned long flags; | ||
293 | |||
294 | if (mac->opmode != NL80211_IFTYPE_STATION) | ||
295 | return; | ||
296 | |||
297 | spin_lock_irqsave(&rtlpriv->locks.ips_lock, flags); | ||
298 | |||
299 | if (ppsc->inactiveps) { | ||
300 | rtstate = ppsc->rfpwr_state; | ||
301 | |||
302 | if (rtstate != ERFON && | ||
303 | !ppsc->swrf_processing && | ||
304 | ppsc->rfoff_reason <= RF_CHANGE_BY_IPS) { | ||
305 | |||
306 | ppsc->inactive_pwrstate = ERFON; | ||
307 | ppsc->in_powersavemode = false; | ||
308 | |||
309 | _rtl_ps_inactive_ps(hw); | ||
310 | } | ||
311 | } | ||
312 | |||
313 | spin_unlock_irqrestore(&rtlpriv->locks.ips_lock, flags); | ||
314 | } | ||
315 | |||
316 | /*for FW LPS*/ | ||
317 | |||
318 | /* | ||
319 | *Determine if we can set Fw into PS mode | ||
320 | *in current condition.Return TRUE if it | ||
321 | *can enter PS mode. | ||
322 | */ | ||
323 | static bool rtl_get_fwlps_doze(struct ieee80211_hw *hw) | ||
324 | { | ||
325 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
326 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
327 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
328 | u32 ps_timediff; | ||
329 | |||
330 | ps_timediff = jiffies_to_msecs(jiffies - | ||
331 | ppsc->last_delaylps_stamp_jiffies); | ||
332 | |||
333 | if (ps_timediff < 2000) { | ||
334 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
335 | ("Delay enter Fw LPS for DHCP, ARP," | ||
336 | " or EAPOL exchanging state.\n")); | ||
337 | return false; | ||
338 | } | ||
339 | |||
340 | if (mac->link_state != MAC80211_LINKED) | ||
341 | return false; | ||
342 | |||
343 | if (mac->opmode == NL80211_IFTYPE_ADHOC) | ||
344 | return false; | ||
345 | |||
346 | return true; | ||
347 | } | ||
348 | |||
349 | /* Change current and default preamble mode.*/ | ||
350 | static void rtl_lps_set_psmode(struct ieee80211_hw *hw, u8 rt_psmode) | ||
351 | { | ||
352 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
353 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
354 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
355 | u8 rpwm_val, fw_pwrmode; | ||
356 | |||
357 | if (mac->opmode == NL80211_IFTYPE_ADHOC) | ||
358 | return; | ||
359 | |||
360 | if (mac->link_state != MAC80211_LINKED) | ||
361 | return; | ||
362 | |||
363 | if (ppsc->dot11_psmode == rt_psmode) | ||
364 | return; | ||
365 | |||
366 | /* Update power save mode configured. */ | ||
367 | ppsc->dot11_psmode = rt_psmode; | ||
368 | |||
369 | /* | ||
370 | *<FW control LPS> | ||
371 | *1. Enter PS mode | ||
372 | * Set RPWM to Fw to turn RF off and send H2C fw_pwrmode | ||
373 | * cmd to set Fw into PS mode. | ||
374 | *2. Leave PS mode | ||
375 | * Send H2C fw_pwrmode cmd to Fw to set Fw into Active | ||
376 | * mode and set RPWM to turn RF on. | ||
377 | */ | ||
378 | |||
379 | if ((ppsc->fwctrl_lps) && ppsc->report_linked) { | ||
380 | bool fw_current_inps; | ||
381 | if (ppsc->dot11_psmode == EACTIVE) { | ||
382 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
383 | ("FW LPS leave ps_mode:%x\n", | ||
384 | FW_PS_ACTIVE_MODE)); | ||
385 | |||
386 | rpwm_val = 0x0C; /* RF on */ | ||
387 | fw_pwrmode = FW_PS_ACTIVE_MODE; | ||
388 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SET_RPWM, | ||
389 | (u8 *) (&rpwm_val)); | ||
390 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
391 | HW_VAR_H2C_FW_PWRMODE, | ||
392 | (u8 *) (&fw_pwrmode)); | ||
393 | fw_current_inps = false; | ||
394 | |||
395 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
396 | HW_VAR_FW_PSMODE_STATUS, | ||
397 | (u8 *) (&fw_current_inps)); | ||
398 | |||
399 | } else { | ||
400 | if (rtl_get_fwlps_doze(hw)) { | ||
401 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
402 | ("FW LPS enter ps_mode:%x\n", | ||
403 | ppsc->fwctrl_psmode)); | ||
404 | |||
405 | rpwm_val = 0x02; /* RF off */ | ||
406 | fw_current_inps = true; | ||
407 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
408 | HW_VAR_FW_PSMODE_STATUS, | ||
409 | (u8 *) (&fw_current_inps)); | ||
410 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
411 | HW_VAR_H2C_FW_PWRMODE, | ||
412 | (u8 *) (&ppsc->fwctrl_psmode)); | ||
413 | |||
414 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
415 | HW_VAR_SET_RPWM, | ||
416 | (u8 *) (&rpwm_val)); | ||
417 | } else { | ||
418 | /* Reset the power save related parameters. */ | ||
419 | ppsc->dot11_psmode = EACTIVE; | ||
420 | } | ||
421 | } | ||
422 | } | ||
423 | } | ||
424 | |||
425 | /*Enter the leisure power save mode.*/ | ||
426 | void rtl_lps_enter(struct ieee80211_hw *hw) | ||
427 | { | ||
428 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
429 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
430 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
431 | unsigned long flag; | ||
432 | |||
433 | if (!ppsc->fwctrl_lps) | ||
434 | return; | ||
435 | |||
436 | if (rtlpriv->sec.being_setkey) | ||
437 | return; | ||
438 | |||
439 | if (rtlpriv->link_info.busytraffic) | ||
440 | return; | ||
441 | |||
442 | /*sleep after linked 10s, to let DHCP and 4-way handshake ok enough!! */ | ||
443 | if (mac->cnt_after_linked < 5) | ||
444 | return; | ||
445 | |||
446 | if (mac->opmode == NL80211_IFTYPE_ADHOC) | ||
447 | return; | ||
448 | |||
449 | if (mac->link_state != MAC80211_LINKED) | ||
450 | return; | ||
451 | |||
452 | spin_lock_irqsave(&rtlpriv->locks.lps_lock, flag); | ||
453 | |||
454 | /* Idle for a while if we connect to AP a while ago. */ | ||
455 | if (mac->cnt_after_linked >= 2) { | ||
456 | if (ppsc->dot11_psmode == EACTIVE) { | ||
457 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
458 | ("Enter 802.11 power save mode...\n")); | ||
459 | |||
460 | rtl_lps_set_psmode(hw, EAUTOPS); | ||
461 | } | ||
462 | } | ||
463 | |||
464 | spin_unlock_irqrestore(&rtlpriv->locks.lps_lock, flag); | ||
465 | } | ||
466 | |||
467 | /*Leave the leisure power save mode.*/ | ||
468 | void rtl_lps_leave(struct ieee80211_hw *hw) | ||
469 | { | ||
470 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
471 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
472 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
473 | unsigned long flag; | ||
474 | |||
475 | spin_lock_irqsave(&rtlpriv->locks.lps_lock, flag); | ||
476 | |||
477 | if (ppsc->fwctrl_lps) { | ||
478 | if (ppsc->dot11_psmode != EACTIVE) { | ||
479 | |||
480 | /*FIX ME */ | ||
481 | rtlpriv->cfg->ops->enable_interrupt(hw); | ||
482 | |||
483 | if (ppsc->reg_rfps_level & RT_RF_LPS_LEVEL_ASPM && | ||
484 | RT_IN_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM) && | ||
485 | rtlhal->interface == INTF_PCI) { | ||
486 | rtlpriv->intf_ops->disable_aspm(hw); | ||
487 | RT_CLEAR_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM); | ||
488 | } | ||
489 | |||
490 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
491 | ("Busy Traffic,Leave 802.11 power save..\n")); | ||
492 | |||
493 | rtl_lps_set_psmode(hw, EACTIVE); | ||
494 | } | ||
495 | } | ||
496 | spin_unlock_irqrestore(&rtlpriv->locks.lps_lock, flag); | ||
497 | } | ||
498 | |||
499 | /* For sw LPS*/ | ||
500 | void rtl_swlps_beacon(struct ieee80211_hw *hw, void *data, unsigned int len) | ||
501 | { | ||
502 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
503 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
504 | struct ieee80211_hdr *hdr = (void *) data; | ||
505 | struct ieee80211_tim_ie *tim_ie; | ||
506 | u8 *tim; | ||
507 | u8 tim_len; | ||
508 | bool u_buffed; | ||
509 | bool m_buffed; | ||
510 | |||
511 | if (mac->opmode != NL80211_IFTYPE_STATION) | ||
512 | return; | ||
513 | |||
514 | if (!rtlpriv->psc.swctrl_lps) | ||
515 | return; | ||
516 | |||
517 | if (rtlpriv->mac80211.link_state != MAC80211_LINKED) | ||
518 | return; | ||
519 | |||
520 | if (!rtlpriv->psc.sw_ps_enabled) | ||
521 | return; | ||
522 | |||
523 | if (rtlpriv->psc.fwctrl_lps) | ||
524 | return; | ||
525 | |||
526 | if (likely(!(hw->conf.flags & IEEE80211_CONF_PS))) | ||
527 | return; | ||
528 | |||
529 | /* check if this really is a beacon */ | ||
530 | if (!ieee80211_is_beacon(hdr->frame_control)) | ||
531 | return; | ||
532 | |||
533 | /* min. beacon length + FCS_LEN */ | ||
534 | if (len <= 40 + FCS_LEN) | ||
535 | return; | ||
536 | |||
537 | /* and only beacons from the associated BSSID, please */ | ||
538 | if (compare_ether_addr(hdr->addr3, rtlpriv->mac80211.bssid)) | ||
539 | return; | ||
540 | |||
541 | rtlpriv->psc.last_beacon = jiffies; | ||
542 | |||
543 | tim = rtl_find_ie(data, len - FCS_LEN, WLAN_EID_TIM); | ||
544 | if (!tim) | ||
545 | return; | ||
546 | |||
547 | if (tim[1] < sizeof(*tim_ie)) | ||
548 | return; | ||
549 | |||
550 | tim_len = tim[1]; | ||
551 | tim_ie = (struct ieee80211_tim_ie *) &tim[2]; | ||
552 | |||
553 | if (!WARN_ON_ONCE(!hw->conf.ps_dtim_period)) | ||
554 | rtlpriv->psc.dtim_counter = tim_ie->dtim_count; | ||
555 | |||
556 | /* Check whenever the PHY can be turned off again. */ | ||
557 | |||
558 | /* 1. What about buffered unicast traffic for our AID? */ | ||
559 | u_buffed = ieee80211_check_tim(tim_ie, tim_len, | ||
560 | rtlpriv->mac80211.assoc_id); | ||
561 | |||
562 | /* 2. Maybe the AP wants to send multicast/broadcast data? */ | ||
563 | m_buffed = tim_ie->bitmap_ctrl & 0x01; | ||
564 | rtlpriv->psc.multi_buffered = m_buffed; | ||
565 | |||
566 | /* unicast will process by mac80211 through | ||
567 | * set ~IEEE80211_CONF_PS, So we just check | ||
568 | * multicast frames here */ | ||
569 | if (!m_buffed) { | ||
570 | /* back to low-power land. and delay is | ||
571 | * prevent null power save frame tx fail */ | ||
572 | queue_delayed_work(rtlpriv->works.rtl_wq, | ||
573 | &rtlpriv->works.ps_work, MSECS(5)); | ||
574 | } else { | ||
575 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, ("u_bufferd: %x, " | ||
576 | "m_buffered: %x\n", u_buffed, m_buffed)); | ||
577 | } | ||
578 | } | ||
579 | |||
580 | void rtl_swlps_rf_awake(struct ieee80211_hw *hw) | ||
581 | { | ||
582 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
583 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
584 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
585 | unsigned long flag; | ||
586 | |||
587 | if (!rtlpriv->psc.swctrl_lps) | ||
588 | return; | ||
589 | if (mac->link_state != MAC80211_LINKED) | ||
590 | return; | ||
591 | |||
592 | if (ppsc->reg_rfps_level & RT_RF_LPS_LEVEL_ASPM && | ||
593 | RT_IN_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM)) { | ||
594 | rtlpriv->intf_ops->disable_aspm(hw); | ||
595 | RT_CLEAR_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM); | ||
596 | } | ||
597 | |||
598 | spin_lock_irqsave(&rtlpriv->locks.lps_lock, flag); | ||
599 | rtl_ps_set_rf_state(hw, ERFON, RF_CHANGE_BY_PS, false); | ||
600 | spin_unlock_irqrestore(&rtlpriv->locks.lps_lock, flag); | ||
601 | } | ||
602 | |||
603 | void rtl_swlps_rfon_wq_callback(void *data) | ||
604 | { | ||
605 | struct rtl_works *rtlworks = | ||
606 | container_of_dwork_rtl(data, struct rtl_works, ps_rfon_wq); | ||
607 | struct ieee80211_hw *hw = rtlworks->hw; | ||
608 | |||
609 | rtl_swlps_rf_awake(hw); | ||
610 | } | ||
611 | |||
612 | void rtl_swlps_rf_sleep(struct ieee80211_hw *hw) | ||
613 | { | ||
614 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
615 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
616 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
617 | unsigned long flag; | ||
618 | u8 sleep_intv; | ||
619 | |||
620 | if (!rtlpriv->psc.sw_ps_enabled) | ||
621 | return; | ||
622 | |||
623 | if ((rtlpriv->sec.being_setkey) || | ||
624 | (mac->opmode == NL80211_IFTYPE_ADHOC)) | ||
625 | return; | ||
626 | |||
627 | /*sleep after linked 10s, to let DHCP and 4-way handshake ok enough!! */ | ||
628 | if ((mac->link_state != MAC80211_LINKED) || (mac->cnt_after_linked < 5)) | ||
629 | return; | ||
630 | |||
631 | if (rtlpriv->link_info.busytraffic) | ||
632 | return; | ||
633 | |||
634 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
635 | if (rtlpriv->psc.rfchange_inprogress) { | ||
636 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
637 | return; | ||
638 | } | ||
639 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
640 | |||
641 | spin_lock_irqsave(&rtlpriv->locks.lps_lock, flag); | ||
642 | rtl_ps_set_rf_state(hw, ERFSLEEP, RF_CHANGE_BY_PS, false); | ||
643 | spin_unlock_irqrestore(&rtlpriv->locks.lps_lock, flag); | ||
644 | |||
645 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM && | ||
646 | !RT_IN_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM)) { | ||
647 | rtlpriv->intf_ops->enable_aspm(hw); | ||
648 | RT_SET_PS_LEVEL(ppsc, RT_PS_LEVEL_ASPM); | ||
649 | } | ||
650 | |||
651 | /* here is power save alg, when this beacon is DTIM | ||
652 | * we will set sleep time to dtim_period * n; | ||
653 | * when this beacon is not DTIM, we will set sleep | ||
654 | * time to sleep_intv = rtlpriv->psc.dtim_counter or | ||
655 | * MAX_SW_LPS_SLEEP_INTV(default set to 5) */ | ||
656 | |||
657 | if (rtlpriv->psc.dtim_counter == 0) { | ||
658 | if (hw->conf.ps_dtim_period == 1) | ||
659 | sleep_intv = hw->conf.ps_dtim_period * 2; | ||
660 | else | ||
661 | sleep_intv = hw->conf.ps_dtim_period; | ||
662 | } else { | ||
663 | sleep_intv = rtlpriv->psc.dtim_counter; | ||
664 | } | ||
665 | |||
666 | if (sleep_intv > MAX_SW_LPS_SLEEP_INTV) | ||
667 | sleep_intv = MAX_SW_LPS_SLEEP_INTV; | ||
668 | |||
669 | /* this print should always be dtim_conter = 0 & | ||
670 | * sleep = dtim_period, that meaons, we should | ||
671 | * awake before every dtim */ | ||
672 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
673 | ("dtim_counter:%x will sleep :%d" | ||
674 | " beacon_intv\n", rtlpriv->psc.dtim_counter, sleep_intv)); | ||
675 | |||
676 | /* we tested that 40ms is enough for sw & hw sw delay */ | ||
677 | queue_delayed_work(rtlpriv->works.rtl_wq, &rtlpriv->works.ps_rfon_wq, | ||
678 | MSECS(sleep_intv * mac->vif->bss_conf.beacon_int - 40)); | ||
679 | } | ||
680 | |||
681 | |||
682 | void rtl_swlps_wq_callback(void *data) | ||
683 | { | ||
684 | struct rtl_works *rtlworks = container_of_dwork_rtl(data, | ||
685 | struct rtl_works, | ||
686 | ps_work); | ||
687 | struct ieee80211_hw *hw = rtlworks->hw; | ||
688 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
689 | bool ps = false; | ||
690 | |||
691 | ps = (hw->conf.flags & IEEE80211_CONF_PS); | ||
692 | |||
693 | /* we can sleep after ps null send ok */ | ||
694 | if (rtlpriv->psc.state_inap) { | ||
695 | rtl_swlps_rf_sleep(hw); | ||
696 | |||
697 | if (rtlpriv->psc.state && !ps) { | ||
698 | rtlpriv->psc.sleep_ms = jiffies_to_msecs(jiffies - | ||
699 | rtlpriv->psc.last_action); | ||
700 | } | ||
701 | |||
702 | if (ps) | ||
703 | rtlpriv->psc.last_slept = jiffies; | ||
704 | |||
705 | rtlpriv->psc.last_action = jiffies; | ||
706 | rtlpriv->psc.state = ps; | ||
707 | } | ||
708 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/ps.h b/drivers/net/wireless/rtlwifi/ps.h new file mode 100644 index 000000000000..e3bf89840370 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/ps.h | |||
@@ -0,0 +1,52 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __REALTEK_RTL_PCI_PS_H__ | ||
31 | #define __REALTEK_RTL_PCI_PS_H__ | ||
32 | |||
33 | #define MAX_SW_LPS_SLEEP_INTV 5 | ||
34 | |||
35 | bool rtl_ps_set_rf_state(struct ieee80211_hw *hw, | ||
36 | enum rf_pwrstate state_toset, u32 changesource, | ||
37 | bool protect_or_not); | ||
38 | bool rtl_ps_enable_nic(struct ieee80211_hw *hw); | ||
39 | bool rtl_ps_disable_nic(struct ieee80211_hw *hw); | ||
40 | void rtl_ips_nic_off(struct ieee80211_hw *hw); | ||
41 | void rtl_ips_nic_on(struct ieee80211_hw *hw); | ||
42 | void rtl_ips_nic_off_wq_callback(void *data); | ||
43 | void rtl_lps_enter(struct ieee80211_hw *hw); | ||
44 | void rtl_lps_leave(struct ieee80211_hw *hw); | ||
45 | |||
46 | void rtl_swlps_beacon(struct ieee80211_hw *hw, void *data, unsigned int len); | ||
47 | void rtl_swlps_wq_callback(void *data); | ||
48 | void rtl_swlps_rfon_wq_callback(void *data); | ||
49 | void rtl_swlps_rf_awake(struct ieee80211_hw *hw); | ||
50 | void rtl_swlps_rf_sleep(struct ieee80211_hw *hw); | ||
51 | |||
52 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rc.c b/drivers/net/wireless/rtlwifi/rc.c new file mode 100644 index 000000000000..30da68a77786 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rc.c | |||
@@ -0,0 +1,291 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "base.h" | ||
32 | #include "rc.h" | ||
33 | |||
34 | /* | ||
35 | *Finds the highest rate index we can use | ||
36 | *if skb is special data like DHCP/EAPOL, we set should | ||
37 | *it to lowest rate CCK_1M, otherwise we set rate to | ||
38 | *CCK11M or OFDM_54M based on wireless mode. | ||
39 | */ | ||
40 | static u8 _rtl_rc_get_highest_rix(struct rtl_priv *rtlpriv, | ||
41 | struct ieee80211_sta *sta, | ||
42 | struct sk_buff *skb, bool not_data) | ||
43 | { | ||
44 | struct rtl_mac *rtlmac = rtl_mac(rtlpriv); | ||
45 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
46 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
47 | struct rtl_sta_info *sta_entry = NULL; | ||
48 | u8 wireless_mode = 0; | ||
49 | |||
50 | /* | ||
51 | *this rate is no use for true rate, firmware | ||
52 | *will control rate at all it just used for | ||
53 | *1.show in iwconfig in B/G mode | ||
54 | *2.in rtl_get_tcb_desc when we check rate is | ||
55 | * 1M we will not use FW rate but user rate. | ||
56 | */ | ||
57 | if (rtlmac->opmode == NL80211_IFTYPE_AP || | ||
58 | rtlmac->opmode == NL80211_IFTYPE_ADHOC) { | ||
59 | if (sta) { | ||
60 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
61 | wireless_mode = sta_entry->wireless_mode; | ||
62 | } else { | ||
63 | return 0; | ||
64 | } | ||
65 | } else { | ||
66 | wireless_mode = rtlmac->mode; | ||
67 | } | ||
68 | |||
69 | if (rtl_is_special_data(rtlpriv->mac80211.hw, skb, true) || | ||
70 | not_data) { | ||
71 | return 0; | ||
72 | } else { | ||
73 | if (rtlhal->current_bandtype == BAND_ON_2_4G) { | ||
74 | if (wireless_mode == WIRELESS_MODE_B) { | ||
75 | return B_MODE_MAX_RIX; | ||
76 | } else if (wireless_mode == WIRELESS_MODE_G) { | ||
77 | return G_MODE_MAX_RIX; | ||
78 | } else { | ||
79 | if (get_rf_type(rtlphy) != RF_2T2R) | ||
80 | return N_MODE_MCS7_RIX; | ||
81 | else | ||
82 | return N_MODE_MCS15_RIX; | ||
83 | } | ||
84 | } else { | ||
85 | if (wireless_mode == WIRELESS_MODE_A) { | ||
86 | return A_MODE_MAX_RIX; | ||
87 | } else { | ||
88 | if (get_rf_type(rtlphy) != RF_2T2R) | ||
89 | return N_MODE_MCS7_RIX; | ||
90 | else | ||
91 | return N_MODE_MCS15_RIX; | ||
92 | } | ||
93 | } | ||
94 | } | ||
95 | } | ||
96 | |||
97 | static void _rtl_rc_rate_set_series(struct rtl_priv *rtlpriv, | ||
98 | struct ieee80211_sta *sta, | ||
99 | struct ieee80211_tx_rate *rate, | ||
100 | struct ieee80211_tx_rate_control *txrc, | ||
101 | u8 tries, char rix, int rtsctsenable, | ||
102 | bool not_data) | ||
103 | { | ||
104 | struct rtl_mac *mac = rtl_mac(rtlpriv); | ||
105 | u8 sgi_20 = 0, sgi_40 = 0; | ||
106 | |||
107 | if (sta) { | ||
108 | sgi_20 = sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20; | ||
109 | sgi_40 = sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40; | ||
110 | } | ||
111 | rate->count = tries; | ||
112 | rate->idx = rix >= 0x00 ? rix : 0x00; | ||
113 | |||
114 | if (!not_data) { | ||
115 | if (txrc->short_preamble) | ||
116 | rate->flags |= IEEE80211_TX_RC_USE_SHORT_PREAMBLE; | ||
117 | if (mac->opmode == NL80211_IFTYPE_AP || | ||
118 | mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
119 | if (sta && (sta->ht_cap.cap & | ||
120 | IEEE80211_HT_CAP_SUP_WIDTH_20_40)) | ||
121 | rate->flags |= IEEE80211_TX_RC_40_MHZ_WIDTH; | ||
122 | } else { | ||
123 | if (mac->bw_40) | ||
124 | rate->flags |= IEEE80211_TX_RC_40_MHZ_WIDTH; | ||
125 | } | ||
126 | if (sgi_20 || sgi_40) | ||
127 | rate->flags |= IEEE80211_TX_RC_SHORT_GI; | ||
128 | if (sta && sta->ht_cap.ht_supported) | ||
129 | rate->flags |= IEEE80211_TX_RC_MCS; | ||
130 | } | ||
131 | } | ||
132 | |||
133 | static void rtl_get_rate(void *ppriv, struct ieee80211_sta *sta, | ||
134 | void *priv_sta, struct ieee80211_tx_rate_control *txrc) | ||
135 | { | ||
136 | struct rtl_priv *rtlpriv = ppriv; | ||
137 | struct sk_buff *skb = txrc->skb; | ||
138 | struct ieee80211_tx_info *tx_info = IEEE80211_SKB_CB(skb); | ||
139 | struct ieee80211_tx_rate *rates = tx_info->control.rates; | ||
140 | __le16 fc = rtl_get_fc(skb); | ||
141 | u8 try_per_rate, i, rix; | ||
142 | bool not_data = !ieee80211_is_data(fc); | ||
143 | |||
144 | if (rate_control_send_low(sta, priv_sta, txrc)) | ||
145 | return; | ||
146 | |||
147 | rix = _rtl_rc_get_highest_rix(rtlpriv, sta, skb, not_data); | ||
148 | try_per_rate = 1; | ||
149 | _rtl_rc_rate_set_series(rtlpriv, sta, &rates[0], txrc, | ||
150 | try_per_rate, rix, 1, not_data); | ||
151 | |||
152 | if (!not_data) { | ||
153 | for (i = 1; i < 4; i++) | ||
154 | _rtl_rc_rate_set_series(rtlpriv, sta, &rates[i], | ||
155 | txrc, i, (rix - i), 1, | ||
156 | not_data); | ||
157 | } | ||
158 | } | ||
159 | |||
160 | static bool _rtl_tx_aggr_check(struct rtl_priv *rtlpriv, | ||
161 | struct rtl_sta_info *sta_entry, u16 tid) | ||
162 | { | ||
163 | struct rtl_mac *mac = rtl_mac(rtlpriv); | ||
164 | |||
165 | if (mac->act_scanning) | ||
166 | return false; | ||
167 | |||
168 | if (mac->opmode == NL80211_IFTYPE_STATION && | ||
169 | mac->cnt_after_linked < 3) | ||
170 | return false; | ||
171 | |||
172 | if (sta_entry->tids[tid].agg.agg_state == RTL_AGG_STOP) | ||
173 | return true; | ||
174 | |||
175 | return false; | ||
176 | } | ||
177 | |||
178 | /*mac80211 Rate Control callbacks*/ | ||
179 | static void rtl_tx_status(void *ppriv, | ||
180 | struct ieee80211_supported_band *sband, | ||
181 | struct ieee80211_sta *sta, void *priv_sta, | ||
182 | struct sk_buff *skb) | ||
183 | { | ||
184 | struct rtl_priv *rtlpriv = ppriv; | ||
185 | struct rtl_mac *mac = rtl_mac(rtlpriv); | ||
186 | struct ieee80211_hdr *hdr = rtl_get_hdr(skb); | ||
187 | __le16 fc = rtl_get_fc(skb); | ||
188 | struct rtl_sta_info *sta_entry; | ||
189 | |||
190 | if (!priv_sta || !ieee80211_is_data(fc)) | ||
191 | return; | ||
192 | |||
193 | if (rtl_is_special_data(mac->hw, skb, true)) | ||
194 | return; | ||
195 | |||
196 | if (is_multicast_ether_addr(ieee80211_get_DA(hdr)) | ||
197 | || is_broadcast_ether_addr(ieee80211_get_DA(hdr))) | ||
198 | return; | ||
199 | |||
200 | if (sta) { | ||
201 | /* Check if aggregation has to be enabled for this tid */ | ||
202 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
203 | if ((sta->ht_cap.ht_supported == true) && | ||
204 | !(skb->protocol == cpu_to_be16(ETH_P_PAE))) { | ||
205 | if (ieee80211_is_data_qos(fc)) { | ||
206 | u8 tid = rtl_get_tid(skb); | ||
207 | if (_rtl_tx_aggr_check(rtlpriv, sta_entry, | ||
208 | tid)) { | ||
209 | sta_entry->tids[tid].agg.agg_state = | ||
210 | RTL_AGG_PROGRESS; | ||
211 | ieee80211_start_tx_ba_session(sta, | ||
212 | tid, 5000); | ||
213 | } | ||
214 | } | ||
215 | } | ||
216 | } | ||
217 | } | ||
218 | |||
219 | static void rtl_rate_init(void *ppriv, | ||
220 | struct ieee80211_supported_band *sband, | ||
221 | struct ieee80211_sta *sta, void *priv_sta) | ||
222 | { | ||
223 | } | ||
224 | |||
225 | static void rtl_rate_update(void *ppriv, | ||
226 | struct ieee80211_supported_band *sband, | ||
227 | struct ieee80211_sta *sta, void *priv_sta, | ||
228 | u32 changed, | ||
229 | enum nl80211_channel_type oper_chan_type) | ||
230 | { | ||
231 | } | ||
232 | |||
233 | static void *rtl_rate_alloc(struct ieee80211_hw *hw, | ||
234 | struct dentry *debugfsdir) | ||
235 | { | ||
236 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
237 | return rtlpriv; | ||
238 | } | ||
239 | |||
240 | static void rtl_rate_free(void *rtlpriv) | ||
241 | { | ||
242 | return; | ||
243 | } | ||
244 | |||
245 | static void *rtl_rate_alloc_sta(void *ppriv, | ||
246 | struct ieee80211_sta *sta, gfp_t gfp) | ||
247 | { | ||
248 | struct rtl_priv *rtlpriv = ppriv; | ||
249 | struct rtl_rate_priv *rate_priv; | ||
250 | |||
251 | rate_priv = kzalloc(sizeof(struct rtl_rate_priv), gfp); | ||
252 | if (!rate_priv) { | ||
253 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
254 | ("Unable to allocate private rc structure\n")); | ||
255 | return NULL; | ||
256 | } | ||
257 | |||
258 | rtlpriv->rate_priv = rate_priv; | ||
259 | |||
260 | return rate_priv; | ||
261 | } | ||
262 | |||
263 | static void rtl_rate_free_sta(void *rtlpriv, | ||
264 | struct ieee80211_sta *sta, void *priv_sta) | ||
265 | { | ||
266 | struct rtl_rate_priv *rate_priv = priv_sta; | ||
267 | kfree(rate_priv); | ||
268 | } | ||
269 | |||
270 | static struct rate_control_ops rtl_rate_ops = { | ||
271 | .module = NULL, | ||
272 | .name = "rtl_rc", | ||
273 | .alloc = rtl_rate_alloc, | ||
274 | .free = rtl_rate_free, | ||
275 | .alloc_sta = rtl_rate_alloc_sta, | ||
276 | .free_sta = rtl_rate_free_sta, | ||
277 | .rate_init = rtl_rate_init, | ||
278 | .rate_update = rtl_rate_update, | ||
279 | .tx_status = rtl_tx_status, | ||
280 | .get_rate = rtl_get_rate, | ||
281 | }; | ||
282 | |||
283 | int rtl_rate_control_register(void) | ||
284 | { | ||
285 | return ieee80211_rate_control_register(&rtl_rate_ops); | ||
286 | } | ||
287 | |||
288 | void rtl_rate_control_unregister(void) | ||
289 | { | ||
290 | ieee80211_rate_control_unregister(&rtl_rate_ops); | ||
291 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rc.h b/drivers/net/wireless/rtlwifi/rc.h new file mode 100644 index 000000000000..4afa2c20adcf --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rc.h | |||
@@ -0,0 +1,47 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_RC_H__ | ||
31 | #define __RTL_RC_H__ | ||
32 | |||
33 | #define B_MODE_MAX_RIX 3 | ||
34 | #define G_MODE_MAX_RIX 11 | ||
35 | #define A_MODE_MAX_RIX 7 | ||
36 | |||
37 | /* in mac80211 mcs0-mcs15 is idx0-idx15*/ | ||
38 | #define N_MODE_MCS7_RIX 7 | ||
39 | #define N_MODE_MCS15_RIX 15 | ||
40 | |||
41 | struct rtl_rate_priv { | ||
42 | u8 ht_cap; | ||
43 | }; | ||
44 | |||
45 | int rtl_rate_control_register(void); | ||
46 | void rtl_rate_control_unregister(void); | ||
47 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/regd.c b/drivers/net/wireless/rtlwifi/regd.c new file mode 100644 index 000000000000..8f6718f163e5 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/regd.c | |||
@@ -0,0 +1,457 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "wifi.h" | ||
31 | #include "regd.h" | ||
32 | |||
33 | static struct country_code_to_enum_rd allCountries[] = { | ||
34 | {COUNTRY_CODE_FCC, "US"}, | ||
35 | {COUNTRY_CODE_IC, "US"}, | ||
36 | {COUNTRY_CODE_ETSI, "EC"}, | ||
37 | {COUNTRY_CODE_SPAIN, "EC"}, | ||
38 | {COUNTRY_CODE_FRANCE, "EC"}, | ||
39 | {COUNTRY_CODE_MKK, "JP"}, | ||
40 | {COUNTRY_CODE_MKK1, "JP"}, | ||
41 | {COUNTRY_CODE_ISRAEL, "EC"}, | ||
42 | {COUNTRY_CODE_TELEC, "JP"}, | ||
43 | {COUNTRY_CODE_MIC, "JP"}, | ||
44 | {COUNTRY_CODE_GLOBAL_DOMAIN, "JP"}, | ||
45 | {COUNTRY_CODE_WORLD_WIDE_13, "EC"}, | ||
46 | {COUNTRY_CODE_TELEC_NETGEAR, "EC"}, | ||
47 | }; | ||
48 | |||
49 | /* | ||
50 | *Only these channels all allow active | ||
51 | *scan on all world regulatory domains | ||
52 | */ | ||
53 | #define RTL819x_2GHZ_CH01_11 \ | ||
54 | REG_RULE(2412-10, 2462+10, 40, 0, 20, 0) | ||
55 | |||
56 | /* | ||
57 | *We enable active scan on these a case | ||
58 | *by case basis by regulatory domain | ||
59 | */ | ||
60 | #define RTL819x_2GHZ_CH12_13 \ | ||
61 | REG_RULE(2467-10, 2472+10, 40, 0, 20,\ | ||
62 | NL80211_RRF_PASSIVE_SCAN) | ||
63 | |||
64 | #define RTL819x_2GHZ_CH14 \ | ||
65 | REG_RULE(2484-10, 2484+10, 40, 0, 20, \ | ||
66 | NL80211_RRF_PASSIVE_SCAN | \ | ||
67 | NL80211_RRF_NO_OFDM) | ||
68 | |||
69 | /* 5G chan 36 - chan 64*/ | ||
70 | #define RTL819x_5GHZ_5150_5350 \ | ||
71 | REG_RULE(5150-10, 5350+10, 40, 0, 30, \ | ||
72 | NL80211_RRF_PASSIVE_SCAN | \ | ||
73 | NL80211_RRF_NO_IBSS) | ||
74 | |||
75 | /* 5G chan 100 - chan 165*/ | ||
76 | #define RTL819x_5GHZ_5470_5850 \ | ||
77 | REG_RULE(5470-10, 5850+10, 40, 0, 30, \ | ||
78 | NL80211_RRF_PASSIVE_SCAN | \ | ||
79 | NL80211_RRF_NO_IBSS) | ||
80 | |||
81 | /* 5G chan 149 - chan 165*/ | ||
82 | #define RTL819x_5GHZ_5725_5850 \ | ||
83 | REG_RULE(5725-10, 5850+10, 40, 0, 30, \ | ||
84 | NL80211_RRF_PASSIVE_SCAN | \ | ||
85 | NL80211_RRF_NO_IBSS) | ||
86 | |||
87 | #define RTL819x_5GHZ_ALL \ | ||
88 | (RTL819x_5GHZ_5150_5350, RTL819x_5GHZ_5470_5850) | ||
89 | |||
90 | static const struct ieee80211_regdomain rtl_regdom_11 = { | ||
91 | .n_reg_rules = 1, | ||
92 | .alpha2 = "99", | ||
93 | .reg_rules = { | ||
94 | RTL819x_2GHZ_CH01_11, | ||
95 | } | ||
96 | }; | ||
97 | |||
98 | static const struct ieee80211_regdomain rtl_regdom_12_13 = { | ||
99 | .n_reg_rules = 2, | ||
100 | .alpha2 = "99", | ||
101 | .reg_rules = { | ||
102 | RTL819x_2GHZ_CH01_11, | ||
103 | RTL819x_2GHZ_CH12_13, | ||
104 | } | ||
105 | }; | ||
106 | |||
107 | static const struct ieee80211_regdomain rtl_regdom_no_midband = { | ||
108 | .n_reg_rules = 3, | ||
109 | .alpha2 = "99", | ||
110 | .reg_rules = { | ||
111 | RTL819x_2GHZ_CH01_11, | ||
112 | RTL819x_5GHZ_5150_5350, | ||
113 | RTL819x_5GHZ_5725_5850, | ||
114 | } | ||
115 | }; | ||
116 | |||
117 | static const struct ieee80211_regdomain rtl_regdom_60_64 = { | ||
118 | .n_reg_rules = 3, | ||
119 | .alpha2 = "99", | ||
120 | .reg_rules = { | ||
121 | RTL819x_2GHZ_CH01_11, | ||
122 | RTL819x_2GHZ_CH12_13, | ||
123 | RTL819x_5GHZ_5725_5850, | ||
124 | } | ||
125 | }; | ||
126 | |||
127 | static const struct ieee80211_regdomain rtl_regdom_14_60_64 = { | ||
128 | .n_reg_rules = 4, | ||
129 | .alpha2 = "99", | ||
130 | .reg_rules = { | ||
131 | RTL819x_2GHZ_CH01_11, | ||
132 | RTL819x_2GHZ_CH12_13, | ||
133 | RTL819x_2GHZ_CH14, | ||
134 | RTL819x_5GHZ_5725_5850, | ||
135 | } | ||
136 | }; | ||
137 | |||
138 | static const struct ieee80211_regdomain rtl_regdom_14 = { | ||
139 | .n_reg_rules = 3, | ||
140 | .alpha2 = "99", | ||
141 | .reg_rules = { | ||
142 | RTL819x_2GHZ_CH01_11, | ||
143 | RTL819x_2GHZ_CH12_13, | ||
144 | RTL819x_2GHZ_CH14, | ||
145 | } | ||
146 | }; | ||
147 | |||
148 | static bool _rtl_is_radar_freq(u16 center_freq) | ||
149 | { | ||
150 | return (center_freq >= 5260 && center_freq <= 5700); | ||
151 | } | ||
152 | |||
153 | static void _rtl_reg_apply_beaconing_flags(struct wiphy *wiphy, | ||
154 | enum nl80211_reg_initiator initiator) | ||
155 | { | ||
156 | enum ieee80211_band band; | ||
157 | struct ieee80211_supported_band *sband; | ||
158 | const struct ieee80211_reg_rule *reg_rule; | ||
159 | struct ieee80211_channel *ch; | ||
160 | unsigned int i; | ||
161 | u32 bandwidth = 0; | ||
162 | int r; | ||
163 | |||
164 | for (band = 0; band < IEEE80211_NUM_BANDS; band++) { | ||
165 | |||
166 | if (!wiphy->bands[band]) | ||
167 | continue; | ||
168 | |||
169 | sband = wiphy->bands[band]; | ||
170 | |||
171 | for (i = 0; i < sband->n_channels; i++) { | ||
172 | ch = &sband->channels[i]; | ||
173 | if (_rtl_is_radar_freq(ch->center_freq) || | ||
174 | (ch->flags & IEEE80211_CHAN_RADAR)) | ||
175 | continue; | ||
176 | if (initiator == NL80211_REGDOM_SET_BY_COUNTRY_IE) { | ||
177 | r = freq_reg_info(wiphy, ch->center_freq, | ||
178 | bandwidth, ®_rule); | ||
179 | if (r) | ||
180 | continue; | ||
181 | |||
182 | /* | ||
183 | *If 11d had a rule for this channel ensure | ||
184 | *we enable adhoc/beaconing if it allows us to | ||
185 | *use it. Note that we would have disabled it | ||
186 | *by applying our static world regdomain by | ||
187 | *default during init, prior to calling our | ||
188 | *regulatory_hint(). | ||
189 | */ | ||
190 | |||
191 | if (!(reg_rule->flags & NL80211_RRF_NO_IBSS)) | ||
192 | ch->flags &= ~IEEE80211_CHAN_NO_IBSS; | ||
193 | if (!(reg_rule-> | ||
194 | flags & NL80211_RRF_PASSIVE_SCAN)) | ||
195 | ch->flags &= | ||
196 | ~IEEE80211_CHAN_PASSIVE_SCAN; | ||
197 | } else { | ||
198 | if (ch->beacon_found) | ||
199 | ch->flags &= ~(IEEE80211_CHAN_NO_IBSS | | ||
200 | IEEE80211_CHAN_PASSIVE_SCAN); | ||
201 | } | ||
202 | } | ||
203 | } | ||
204 | } | ||
205 | |||
206 | /* Allows active scan scan on Ch 12 and 13 */ | ||
207 | static void _rtl_reg_apply_active_scan_flags(struct wiphy *wiphy, | ||
208 | enum nl80211_reg_initiator | ||
209 | initiator) | ||
210 | { | ||
211 | struct ieee80211_supported_band *sband; | ||
212 | struct ieee80211_channel *ch; | ||
213 | const struct ieee80211_reg_rule *reg_rule; | ||
214 | u32 bandwidth = 0; | ||
215 | int r; | ||
216 | |||
217 | if (!wiphy->bands[IEEE80211_BAND_2GHZ]) | ||
218 | return; | ||
219 | sband = wiphy->bands[IEEE80211_BAND_2GHZ]; | ||
220 | |||
221 | /* | ||
222 | *If no country IE has been received always enable active scan | ||
223 | *on these channels. This is only done for specific regulatory SKUs | ||
224 | */ | ||
225 | if (initiator != NL80211_REGDOM_SET_BY_COUNTRY_IE) { | ||
226 | ch = &sband->channels[11]; /* CH 12 */ | ||
227 | if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN) | ||
228 | ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN; | ||
229 | ch = &sband->channels[12]; /* CH 13 */ | ||
230 | if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN) | ||
231 | ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN; | ||
232 | return; | ||
233 | } | ||
234 | |||
235 | /* | ||
236 | *If a country IE has been received check its rule for this | ||
237 | *channel first before enabling active scan. The passive scan | ||
238 | *would have been enforced by the initial processing of our | ||
239 | *custom regulatory domain. | ||
240 | */ | ||
241 | |||
242 | ch = &sband->channels[11]; /* CH 12 */ | ||
243 | r = freq_reg_info(wiphy, ch->center_freq, bandwidth, ®_rule); | ||
244 | if (!r) { | ||
245 | if (!(reg_rule->flags & NL80211_RRF_PASSIVE_SCAN)) | ||
246 | if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN) | ||
247 | ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN; | ||
248 | } | ||
249 | |||
250 | ch = &sband->channels[12]; /* CH 13 */ | ||
251 | r = freq_reg_info(wiphy, ch->center_freq, bandwidth, ®_rule); | ||
252 | if (!r) { | ||
253 | if (!(reg_rule->flags & NL80211_RRF_PASSIVE_SCAN)) | ||
254 | if (ch->flags & IEEE80211_CHAN_PASSIVE_SCAN) | ||
255 | ch->flags &= ~IEEE80211_CHAN_PASSIVE_SCAN; | ||
256 | } | ||
257 | } | ||
258 | |||
259 | /* | ||
260 | *Always apply Radar/DFS rules on | ||
261 | *freq range 5260 MHz - 5700 MHz | ||
262 | */ | ||
263 | static void _rtl_reg_apply_radar_flags(struct wiphy *wiphy) | ||
264 | { | ||
265 | struct ieee80211_supported_band *sband; | ||
266 | struct ieee80211_channel *ch; | ||
267 | unsigned int i; | ||
268 | |||
269 | if (!wiphy->bands[IEEE80211_BAND_5GHZ]) | ||
270 | return; | ||
271 | |||
272 | sband = wiphy->bands[IEEE80211_BAND_5GHZ]; | ||
273 | |||
274 | for (i = 0; i < sband->n_channels; i++) { | ||
275 | ch = &sband->channels[i]; | ||
276 | if (!_rtl_is_radar_freq(ch->center_freq)) | ||
277 | continue; | ||
278 | |||
279 | /* | ||
280 | *We always enable radar detection/DFS on this | ||
281 | *frequency range. Additionally we also apply on | ||
282 | *this frequency range: | ||
283 | *- If STA mode does not yet have DFS supports disable | ||
284 | * active scanning | ||
285 | *- If adhoc mode does not support DFS yet then disable | ||
286 | * adhoc in the frequency. | ||
287 | *- If AP mode does not yet support radar detection/DFS | ||
288 | *do not allow AP mode | ||
289 | */ | ||
290 | if (!(ch->flags & IEEE80211_CHAN_DISABLED)) | ||
291 | ch->flags |= IEEE80211_CHAN_RADAR | | ||
292 | IEEE80211_CHAN_NO_IBSS | | ||
293 | IEEE80211_CHAN_PASSIVE_SCAN; | ||
294 | } | ||
295 | } | ||
296 | |||
297 | static void _rtl_reg_apply_world_flags(struct wiphy *wiphy, | ||
298 | enum nl80211_reg_initiator initiator, | ||
299 | struct rtl_regulatory *reg) | ||
300 | { | ||
301 | _rtl_reg_apply_beaconing_flags(wiphy, initiator); | ||
302 | _rtl_reg_apply_active_scan_flags(wiphy, initiator); | ||
303 | return; | ||
304 | } | ||
305 | |||
306 | static void _rtl_dump_channel_map(struct wiphy *wiphy) | ||
307 | { | ||
308 | enum ieee80211_band band; | ||
309 | struct ieee80211_supported_band *sband; | ||
310 | struct ieee80211_channel *ch; | ||
311 | unsigned int i; | ||
312 | |||
313 | for (band = 0; band < IEEE80211_NUM_BANDS; band++) { | ||
314 | if (!wiphy->bands[band]) | ||
315 | continue; | ||
316 | sband = wiphy->bands[band]; | ||
317 | for (i = 0; i < sband->n_channels; i++) | ||
318 | ch = &sband->channels[i]; | ||
319 | } | ||
320 | } | ||
321 | |||
322 | static int _rtl_reg_notifier_apply(struct wiphy *wiphy, | ||
323 | struct regulatory_request *request, | ||
324 | struct rtl_regulatory *reg) | ||
325 | { | ||
326 | /* We always apply this */ | ||
327 | _rtl_reg_apply_radar_flags(wiphy); | ||
328 | |||
329 | switch (request->initiator) { | ||
330 | case NL80211_REGDOM_SET_BY_DRIVER: | ||
331 | case NL80211_REGDOM_SET_BY_CORE: | ||
332 | case NL80211_REGDOM_SET_BY_USER: | ||
333 | break; | ||
334 | case NL80211_REGDOM_SET_BY_COUNTRY_IE: | ||
335 | _rtl_reg_apply_world_flags(wiphy, request->initiator, reg); | ||
336 | break; | ||
337 | } | ||
338 | |||
339 | _rtl_dump_channel_map(wiphy); | ||
340 | |||
341 | return 0; | ||
342 | } | ||
343 | |||
344 | static const struct ieee80211_regdomain *_rtl_regdomain_select( | ||
345 | struct rtl_regulatory *reg) | ||
346 | { | ||
347 | switch (reg->country_code) { | ||
348 | case COUNTRY_CODE_FCC: | ||
349 | return &rtl_regdom_no_midband; | ||
350 | case COUNTRY_CODE_IC: | ||
351 | return &rtl_regdom_11; | ||
352 | case COUNTRY_CODE_ETSI: | ||
353 | case COUNTRY_CODE_TELEC_NETGEAR: | ||
354 | return &rtl_regdom_60_64; | ||
355 | case COUNTRY_CODE_SPAIN: | ||
356 | case COUNTRY_CODE_FRANCE: | ||
357 | case COUNTRY_CODE_ISRAEL: | ||
358 | case COUNTRY_CODE_WORLD_WIDE_13: | ||
359 | return &rtl_regdom_12_13; | ||
360 | case COUNTRY_CODE_MKK: | ||
361 | case COUNTRY_CODE_MKK1: | ||
362 | case COUNTRY_CODE_TELEC: | ||
363 | case COUNTRY_CODE_MIC: | ||
364 | return &rtl_regdom_14_60_64; | ||
365 | case COUNTRY_CODE_GLOBAL_DOMAIN: | ||
366 | return &rtl_regdom_14; | ||
367 | default: | ||
368 | return &rtl_regdom_no_midband; | ||
369 | } | ||
370 | } | ||
371 | |||
372 | static int _rtl_regd_init_wiphy(struct rtl_regulatory *reg, | ||
373 | struct wiphy *wiphy, | ||
374 | int (*reg_notifier) (struct wiphy *wiphy, | ||
375 | struct regulatory_request * | ||
376 | request)) | ||
377 | { | ||
378 | const struct ieee80211_regdomain *regd; | ||
379 | |||
380 | wiphy->reg_notifier = reg_notifier; | ||
381 | |||
382 | wiphy->flags |= WIPHY_FLAG_CUSTOM_REGULATORY; | ||
383 | wiphy->flags &= ~WIPHY_FLAG_STRICT_REGULATORY; | ||
384 | wiphy->flags &= ~WIPHY_FLAG_DISABLE_BEACON_HINTS; | ||
385 | |||
386 | regd = _rtl_regdomain_select(reg); | ||
387 | wiphy_apply_custom_regulatory(wiphy, regd); | ||
388 | _rtl_reg_apply_radar_flags(wiphy); | ||
389 | _rtl_reg_apply_world_flags(wiphy, NL80211_REGDOM_SET_BY_DRIVER, reg); | ||
390 | return 0; | ||
391 | } | ||
392 | |||
393 | static struct country_code_to_enum_rd *_rtl_regd_find_country(u16 countrycode) | ||
394 | { | ||
395 | int i; | ||
396 | |||
397 | for (i = 0; i < ARRAY_SIZE(allCountries); i++) { | ||
398 | if (allCountries[i].countrycode == countrycode) | ||
399 | return &allCountries[i]; | ||
400 | } | ||
401 | return NULL; | ||
402 | } | ||
403 | |||
404 | int rtl_regd_init(struct ieee80211_hw *hw, | ||
405 | int (*reg_notifier) (struct wiphy *wiphy, | ||
406 | struct regulatory_request *request)) | ||
407 | { | ||
408 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
409 | struct wiphy *wiphy = hw->wiphy; | ||
410 | struct country_code_to_enum_rd *country = NULL; | ||
411 | |||
412 | if (wiphy == NULL || &rtlpriv->regd == NULL) | ||
413 | return -EINVAL; | ||
414 | |||
415 | /* init country_code from efuse channel plan */ | ||
416 | rtlpriv->regd.country_code = rtlpriv->efuse.channel_plan; | ||
417 | |||
418 | RT_TRACE(rtlpriv, COMP_REGD, DBG_TRACE, | ||
419 | (KERN_DEBUG "rtl: EEPROM regdomain: 0x%0x\n", | ||
420 | rtlpriv->regd.country_code)); | ||
421 | |||
422 | if (rtlpriv->regd.country_code >= COUNTRY_CODE_MAX) { | ||
423 | RT_TRACE(rtlpriv, COMP_REGD, DBG_DMESG, | ||
424 | (KERN_DEBUG "rtl: EEPROM indicates invalid contry code" | ||
425 | "world wide 13 should be used\n")); | ||
426 | |||
427 | rtlpriv->regd.country_code = COUNTRY_CODE_WORLD_WIDE_13; | ||
428 | } | ||
429 | |||
430 | country = _rtl_regd_find_country(rtlpriv->regd.country_code); | ||
431 | |||
432 | if (country) { | ||
433 | rtlpriv->regd.alpha2[0] = country->iso_name[0]; | ||
434 | rtlpriv->regd.alpha2[1] = country->iso_name[1]; | ||
435 | } else { | ||
436 | rtlpriv->regd.alpha2[0] = '0'; | ||
437 | rtlpriv->regd.alpha2[1] = '0'; | ||
438 | } | ||
439 | |||
440 | RT_TRACE(rtlpriv, COMP_REGD, DBG_TRACE, | ||
441 | (KERN_DEBUG "rtl: Country alpha2 being used: %c%c\n", | ||
442 | rtlpriv->regd.alpha2[0], rtlpriv->regd.alpha2[1])); | ||
443 | |||
444 | _rtl_regd_init_wiphy(&rtlpriv->regd, wiphy, reg_notifier); | ||
445 | |||
446 | return 0; | ||
447 | } | ||
448 | |||
449 | int rtl_reg_notifier(struct wiphy *wiphy, struct regulatory_request *request) | ||
450 | { | ||
451 | struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy); | ||
452 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
453 | |||
454 | RT_TRACE(rtlpriv, COMP_REGD, DBG_LOUD, ("\n")); | ||
455 | |||
456 | return _rtl_reg_notifier_apply(wiphy, request, &rtlpriv->regd); | ||
457 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/regd.h b/drivers/net/wireless/rtlwifi/regd.h new file mode 100644 index 000000000000..d23118938fac --- /dev/null +++ b/drivers/net/wireless/rtlwifi/regd.h | |||
@@ -0,0 +1,61 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_REGD_H__ | ||
31 | #define __RTL_REGD_H__ | ||
32 | |||
33 | struct country_code_to_enum_rd { | ||
34 | u16 countrycode; | ||
35 | const char *iso_name; | ||
36 | }; | ||
37 | |||
38 | enum country_code_type_t { | ||
39 | COUNTRY_CODE_FCC = 0, | ||
40 | COUNTRY_CODE_IC = 1, | ||
41 | COUNTRY_CODE_ETSI = 2, | ||
42 | COUNTRY_CODE_SPAIN = 3, | ||
43 | COUNTRY_CODE_FRANCE = 4, | ||
44 | COUNTRY_CODE_MKK = 5, | ||
45 | COUNTRY_CODE_MKK1 = 6, | ||
46 | COUNTRY_CODE_ISRAEL = 7, | ||
47 | COUNTRY_CODE_TELEC = 8, | ||
48 | COUNTRY_CODE_MIC = 9, | ||
49 | COUNTRY_CODE_GLOBAL_DOMAIN = 10, | ||
50 | COUNTRY_CODE_WORLD_WIDE_13 = 11, | ||
51 | COUNTRY_CODE_TELEC_NETGEAR = 12, | ||
52 | |||
53 | /*add new channel plan above this line */ | ||
54 | COUNTRY_CODE_MAX | ||
55 | }; | ||
56 | |||
57 | int rtl_regd_init(struct ieee80211_hw *hw, | ||
58 | int (*reg_notifier) (struct wiphy *wiphy, | ||
59 | struct regulatory_request *request)); | ||
60 | int rtl_reg_notifier(struct wiphy *wiphy, struct regulatory_request *request); | ||
61 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/Makefile b/drivers/net/wireless/rtlwifi/rtl8192c/Makefile new file mode 100644 index 000000000000..aee42d7ae8a2 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/Makefile | |||
@@ -0,0 +1,9 @@ | |||
1 | rtl8192c-common-objs := \ | ||
2 | main.o \ | ||
3 | dm_common.o \ | ||
4 | fw_common.o \ | ||
5 | phy_common.o | ||
6 | |||
7 | obj-$(CONFIG_RTL8192C_COMMON) += rtl8192c-common.o | ||
8 | |||
9 | ccflags-y += -D__CHECK_ENDIAN__ | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.c b/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.c new file mode 100644 index 000000000000..97183829b9be --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.c | |||
@@ -0,0 +1,1779 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "dm_common.h" | ||
31 | #include "phy_common.h" | ||
32 | #include "../pci.h" | ||
33 | #include "../base.h" | ||
34 | |||
35 | struct dig_t dm_digtable; | ||
36 | static struct ps_t dm_pstable; | ||
37 | |||
38 | #define BT_RSSI_STATE_NORMAL_POWER BIT_OFFSET_LEN_MASK_32(0, 1) | ||
39 | #define BT_RSSI_STATE_AMDPU_OFF BIT_OFFSET_LEN_MASK_32(1, 1) | ||
40 | #define BT_RSSI_STATE_SPECIAL_LOW BIT_OFFSET_LEN_MASK_32(2, 1) | ||
41 | #define BT_RSSI_STATE_BG_EDCA_LOW BIT_OFFSET_LEN_MASK_32(3, 1) | ||
42 | #define BT_RSSI_STATE_TXPOWER_LOW BIT_OFFSET_LEN_MASK_32(4, 1) | ||
43 | |||
44 | #define RTLPRIV (struct rtl_priv *) | ||
45 | #define GET_UNDECORATED_AVERAGE_RSSI(_priv) \ | ||
46 | ((RTLPRIV(_priv))->mac80211.opmode == \ | ||
47 | NL80211_IFTYPE_ADHOC) ? \ | ||
48 | ((RTLPRIV(_priv))->dm.entry_min_undecoratedsmoothed_pwdb) : \ | ||
49 | ((RTLPRIV(_priv))->dm.undecorated_smoothed_pwdb) | ||
50 | |||
51 | static const u32 ofdmswing_table[OFDM_TABLE_SIZE] = { | ||
52 | 0x7f8001fe, | ||
53 | 0x788001e2, | ||
54 | 0x71c001c7, | ||
55 | 0x6b8001ae, | ||
56 | 0x65400195, | ||
57 | 0x5fc0017f, | ||
58 | 0x5a400169, | ||
59 | 0x55400155, | ||
60 | 0x50800142, | ||
61 | 0x4c000130, | ||
62 | 0x47c0011f, | ||
63 | 0x43c0010f, | ||
64 | 0x40000100, | ||
65 | 0x3c8000f2, | ||
66 | 0x390000e4, | ||
67 | 0x35c000d7, | ||
68 | 0x32c000cb, | ||
69 | 0x300000c0, | ||
70 | 0x2d4000b5, | ||
71 | 0x2ac000ab, | ||
72 | 0x288000a2, | ||
73 | 0x26000098, | ||
74 | 0x24000090, | ||
75 | 0x22000088, | ||
76 | 0x20000080, | ||
77 | 0x1e400079, | ||
78 | 0x1c800072, | ||
79 | 0x1b00006c, | ||
80 | 0x19800066, | ||
81 | 0x18000060, | ||
82 | 0x16c0005b, | ||
83 | 0x15800056, | ||
84 | 0x14400051, | ||
85 | 0x1300004c, | ||
86 | 0x12000048, | ||
87 | 0x11000044, | ||
88 | 0x10000040, | ||
89 | }; | ||
90 | |||
91 | static const u8 cckswing_table_ch1ch13[CCK_TABLE_SIZE][8] = { | ||
92 | {0x36, 0x35, 0x2e, 0x25, 0x1c, 0x12, 0x09, 0x04}, | ||
93 | {0x33, 0x32, 0x2b, 0x23, 0x1a, 0x11, 0x08, 0x04}, | ||
94 | {0x30, 0x2f, 0x29, 0x21, 0x19, 0x10, 0x08, 0x03}, | ||
95 | {0x2d, 0x2d, 0x27, 0x1f, 0x18, 0x0f, 0x08, 0x03}, | ||
96 | {0x2b, 0x2a, 0x25, 0x1e, 0x16, 0x0e, 0x07, 0x03}, | ||
97 | {0x28, 0x28, 0x22, 0x1c, 0x15, 0x0d, 0x07, 0x03}, | ||
98 | {0x26, 0x25, 0x21, 0x1b, 0x14, 0x0d, 0x06, 0x03}, | ||
99 | {0x24, 0x23, 0x1f, 0x19, 0x13, 0x0c, 0x06, 0x03}, | ||
100 | {0x22, 0x21, 0x1d, 0x18, 0x11, 0x0b, 0x06, 0x02}, | ||
101 | {0x20, 0x20, 0x1b, 0x16, 0x11, 0x08, 0x05, 0x02}, | ||
102 | {0x1f, 0x1e, 0x1a, 0x15, 0x10, 0x0a, 0x05, 0x02}, | ||
103 | {0x1d, 0x1c, 0x18, 0x14, 0x0f, 0x0a, 0x05, 0x02}, | ||
104 | {0x1b, 0x1a, 0x17, 0x13, 0x0e, 0x09, 0x04, 0x02}, | ||
105 | {0x1a, 0x19, 0x16, 0x12, 0x0d, 0x09, 0x04, 0x02}, | ||
106 | {0x18, 0x17, 0x15, 0x11, 0x0c, 0x08, 0x04, 0x02}, | ||
107 | {0x17, 0x16, 0x13, 0x10, 0x0c, 0x08, 0x04, 0x02}, | ||
108 | {0x16, 0x15, 0x12, 0x0f, 0x0b, 0x07, 0x04, 0x01}, | ||
109 | {0x14, 0x14, 0x11, 0x0e, 0x0b, 0x07, 0x03, 0x02}, | ||
110 | {0x13, 0x13, 0x10, 0x0d, 0x0a, 0x06, 0x03, 0x01}, | ||
111 | {0x12, 0x12, 0x0f, 0x0c, 0x09, 0x06, 0x03, 0x01}, | ||
112 | {0x11, 0x11, 0x0f, 0x0c, 0x09, 0x06, 0x03, 0x01}, | ||
113 | {0x10, 0x10, 0x0e, 0x0b, 0x08, 0x05, 0x03, 0x01}, | ||
114 | {0x0f, 0x0f, 0x0d, 0x0b, 0x08, 0x05, 0x03, 0x01}, | ||
115 | {0x0e, 0x0e, 0x0c, 0x0a, 0x08, 0x05, 0x02, 0x01}, | ||
116 | {0x0d, 0x0d, 0x0c, 0x0a, 0x07, 0x05, 0x02, 0x01}, | ||
117 | {0x0d, 0x0c, 0x0b, 0x09, 0x07, 0x04, 0x02, 0x01}, | ||
118 | {0x0c, 0x0c, 0x0a, 0x09, 0x06, 0x04, 0x02, 0x01}, | ||
119 | {0x0b, 0x0b, 0x0a, 0x08, 0x06, 0x04, 0x02, 0x01}, | ||
120 | {0x0b, 0x0a, 0x09, 0x08, 0x06, 0x04, 0x02, 0x01}, | ||
121 | {0x0a, 0x0a, 0x09, 0x07, 0x05, 0x03, 0x02, 0x01}, | ||
122 | {0x0a, 0x09, 0x08, 0x07, 0x05, 0x03, 0x02, 0x01}, | ||
123 | {0x09, 0x09, 0x08, 0x06, 0x05, 0x03, 0x01, 0x01}, | ||
124 | {0x09, 0x08, 0x07, 0x06, 0x04, 0x03, 0x01, 0x01} | ||
125 | }; | ||
126 | |||
127 | static const u8 cckswing_table_ch14[CCK_TABLE_SIZE][8] = { | ||
128 | {0x36, 0x35, 0x2e, 0x1b, 0x00, 0x00, 0x00, 0x00}, | ||
129 | {0x33, 0x32, 0x2b, 0x19, 0x00, 0x00, 0x00, 0x00}, | ||
130 | {0x30, 0x2f, 0x29, 0x18, 0x00, 0x00, 0x00, 0x00}, | ||
131 | {0x2d, 0x2d, 0x17, 0x17, 0x00, 0x00, 0x00, 0x00}, | ||
132 | {0x2b, 0x2a, 0x25, 0x15, 0x00, 0x00, 0x00, 0x00}, | ||
133 | {0x28, 0x28, 0x24, 0x14, 0x00, 0x00, 0x00, 0x00}, | ||
134 | {0x26, 0x25, 0x21, 0x13, 0x00, 0x00, 0x00, 0x00}, | ||
135 | {0x24, 0x23, 0x1f, 0x12, 0x00, 0x00, 0x00, 0x00}, | ||
136 | {0x22, 0x21, 0x1d, 0x11, 0x00, 0x00, 0x00, 0x00}, | ||
137 | {0x20, 0x20, 0x1b, 0x10, 0x00, 0x00, 0x00, 0x00}, | ||
138 | {0x1f, 0x1e, 0x1a, 0x0f, 0x00, 0x00, 0x00, 0x00}, | ||
139 | {0x1d, 0x1c, 0x18, 0x0e, 0x00, 0x00, 0x00, 0x00}, | ||
140 | {0x1b, 0x1a, 0x17, 0x0e, 0x00, 0x00, 0x00, 0x00}, | ||
141 | {0x1a, 0x19, 0x16, 0x0d, 0x00, 0x00, 0x00, 0x00}, | ||
142 | {0x18, 0x17, 0x15, 0x0c, 0x00, 0x00, 0x00, 0x00}, | ||
143 | {0x17, 0x16, 0x13, 0x0b, 0x00, 0x00, 0x00, 0x00}, | ||
144 | {0x16, 0x15, 0x12, 0x0b, 0x00, 0x00, 0x00, 0x00}, | ||
145 | {0x14, 0x14, 0x11, 0x0a, 0x00, 0x00, 0x00, 0x00}, | ||
146 | {0x13, 0x13, 0x10, 0x0a, 0x00, 0x00, 0x00, 0x00}, | ||
147 | {0x12, 0x12, 0x0f, 0x09, 0x00, 0x00, 0x00, 0x00}, | ||
148 | {0x11, 0x11, 0x0f, 0x09, 0x00, 0x00, 0x00, 0x00}, | ||
149 | {0x10, 0x10, 0x0e, 0x08, 0x00, 0x00, 0x00, 0x00}, | ||
150 | {0x0f, 0x0f, 0x0d, 0x08, 0x00, 0x00, 0x00, 0x00}, | ||
151 | {0x0e, 0x0e, 0x0c, 0x07, 0x00, 0x00, 0x00, 0x00}, | ||
152 | {0x0d, 0x0d, 0x0c, 0x07, 0x00, 0x00, 0x00, 0x00}, | ||
153 | {0x0d, 0x0c, 0x0b, 0x06, 0x00, 0x00, 0x00, 0x00}, | ||
154 | {0x0c, 0x0c, 0x0a, 0x06, 0x00, 0x00, 0x00, 0x00}, | ||
155 | {0x0b, 0x0b, 0x0a, 0x06, 0x00, 0x00, 0x00, 0x00}, | ||
156 | {0x0b, 0x0a, 0x09, 0x05, 0x00, 0x00, 0x00, 0x00}, | ||
157 | {0x0a, 0x0a, 0x09, 0x05, 0x00, 0x00, 0x00, 0x00}, | ||
158 | {0x0a, 0x09, 0x08, 0x05, 0x00, 0x00, 0x00, 0x00}, | ||
159 | {0x09, 0x09, 0x08, 0x05, 0x00, 0x00, 0x00, 0x00}, | ||
160 | {0x09, 0x08, 0x07, 0x04, 0x00, 0x00, 0x00, 0x00} | ||
161 | }; | ||
162 | |||
163 | static void rtl92c_dm_diginit(struct ieee80211_hw *hw) | ||
164 | { | ||
165 | dm_digtable.dig_enable_flag = true; | ||
166 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_MAX; | ||
167 | dm_digtable.cur_igvalue = 0x20; | ||
168 | dm_digtable.pre_igvalue = 0x0; | ||
169 | dm_digtable.cursta_connectctate = DIG_STA_DISCONNECT; | ||
170 | dm_digtable.presta_connectstate = DIG_STA_DISCONNECT; | ||
171 | dm_digtable.curmultista_connectstate = DIG_MULTISTA_DISCONNECT; | ||
172 | dm_digtable.rssi_lowthresh = DM_DIG_THRESH_LOW; | ||
173 | dm_digtable.rssi_highthresh = DM_DIG_THRESH_HIGH; | ||
174 | dm_digtable.fa_lowthresh = DM_FALSEALARM_THRESH_LOW; | ||
175 | dm_digtable.fa_highthresh = DM_FALSEALARM_THRESH_HIGH; | ||
176 | dm_digtable.rx_gain_range_max = DM_DIG_MAX; | ||
177 | dm_digtable.rx_gain_range_min = DM_DIG_MIN; | ||
178 | dm_digtable.backoff_val = DM_DIG_BACKOFF_DEFAULT; | ||
179 | dm_digtable.backoff_val_range_max = DM_DIG_BACKOFF_MAX; | ||
180 | dm_digtable.backoff_val_range_min = DM_DIG_BACKOFF_MIN; | ||
181 | dm_digtable.pre_cck_pd_state = CCK_PD_STAGE_MAX; | ||
182 | dm_digtable.cur_cck_pd_state = CCK_PD_STAGE_MAX; | ||
183 | } | ||
184 | |||
185 | static u8 rtl92c_dm_initial_gain_min_pwdb(struct ieee80211_hw *hw) | ||
186 | { | ||
187 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
188 | long rssi_val_min = 0; | ||
189 | |||
190 | if ((dm_digtable.curmultista_connectstate == DIG_MULTISTA_CONNECT) && | ||
191 | (dm_digtable.cursta_connectctate == DIG_STA_CONNECT)) { | ||
192 | if (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb != 0) | ||
193 | rssi_val_min = | ||
194 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb > | ||
195 | rtlpriv->dm.undecorated_smoothed_pwdb) ? | ||
196 | rtlpriv->dm.undecorated_smoothed_pwdb : | ||
197 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
198 | else | ||
199 | rssi_val_min = rtlpriv->dm.undecorated_smoothed_pwdb; | ||
200 | } else if (dm_digtable.cursta_connectctate == DIG_STA_CONNECT || | ||
201 | dm_digtable.cursta_connectctate == DIG_STA_BEFORE_CONNECT) { | ||
202 | rssi_val_min = rtlpriv->dm.undecorated_smoothed_pwdb; | ||
203 | } else if (dm_digtable.curmultista_connectstate == | ||
204 | DIG_MULTISTA_CONNECT) { | ||
205 | rssi_val_min = rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
206 | } | ||
207 | |||
208 | return (u8) rssi_val_min; | ||
209 | } | ||
210 | |||
211 | static void rtl92c_dm_false_alarm_counter_statistics(struct ieee80211_hw *hw) | ||
212 | { | ||
213 | u32 ret_value; | ||
214 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
215 | struct false_alarm_statistics *falsealm_cnt = &(rtlpriv->falsealm_cnt); | ||
216 | |||
217 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER1, MASKDWORD); | ||
218 | falsealm_cnt->cnt_parity_fail = ((ret_value & 0xffff0000) >> 16); | ||
219 | |||
220 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER2, MASKDWORD); | ||
221 | falsealm_cnt->cnt_rate_illegal = (ret_value & 0xffff); | ||
222 | falsealm_cnt->cnt_crc8_fail = ((ret_value & 0xffff0000) >> 16); | ||
223 | |||
224 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER3, MASKDWORD); | ||
225 | falsealm_cnt->cnt_mcs_fail = (ret_value & 0xffff); | ||
226 | falsealm_cnt->cnt_ofdm_fail = falsealm_cnt->cnt_parity_fail + | ||
227 | falsealm_cnt->cnt_rate_illegal + | ||
228 | falsealm_cnt->cnt_crc8_fail + falsealm_cnt->cnt_mcs_fail; | ||
229 | |||
230 | rtl_set_bbreg(hw, RCCK0_FALSEALARMREPORT, BIT(14), 1); | ||
231 | ret_value = rtl_get_bbreg(hw, RCCK0_FACOUNTERLOWER, MASKBYTE0); | ||
232 | falsealm_cnt->cnt_cck_fail = ret_value; | ||
233 | |||
234 | ret_value = rtl_get_bbreg(hw, RCCK0_FACOUNTERUPPER, MASKBYTE3); | ||
235 | falsealm_cnt->cnt_cck_fail += (ret_value & 0xff) << 8; | ||
236 | falsealm_cnt->cnt_all = (falsealm_cnt->cnt_parity_fail + | ||
237 | falsealm_cnt->cnt_rate_illegal + | ||
238 | falsealm_cnt->cnt_crc8_fail + | ||
239 | falsealm_cnt->cnt_mcs_fail + | ||
240 | falsealm_cnt->cnt_cck_fail); | ||
241 | |||
242 | rtl_set_bbreg(hw, ROFDM1_LSTF, 0x08000000, 1); | ||
243 | rtl_set_bbreg(hw, ROFDM1_LSTF, 0x08000000, 0); | ||
244 | rtl_set_bbreg(hw, RCCK0_FALSEALARMREPORT, 0x0000c000, 0); | ||
245 | rtl_set_bbreg(hw, RCCK0_FALSEALARMREPORT, 0x0000c000, 2); | ||
246 | |||
247 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
248 | ("cnt_parity_fail = %d, cnt_rate_illegal = %d, " | ||
249 | "cnt_crc8_fail = %d, cnt_mcs_fail = %d\n", | ||
250 | falsealm_cnt->cnt_parity_fail, | ||
251 | falsealm_cnt->cnt_rate_illegal, | ||
252 | falsealm_cnt->cnt_crc8_fail, falsealm_cnt->cnt_mcs_fail)); | ||
253 | |||
254 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
255 | ("cnt_ofdm_fail = %x, cnt_cck_fail = %x, cnt_all = %x\n", | ||
256 | falsealm_cnt->cnt_ofdm_fail, | ||
257 | falsealm_cnt->cnt_cck_fail, falsealm_cnt->cnt_all)); | ||
258 | } | ||
259 | |||
260 | static void rtl92c_dm_ctrl_initgain_by_fa(struct ieee80211_hw *hw) | ||
261 | { | ||
262 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
263 | u8 value_igi = dm_digtable.cur_igvalue; | ||
264 | |||
265 | if (rtlpriv->falsealm_cnt.cnt_all < DM_DIG_FA_TH0) | ||
266 | value_igi--; | ||
267 | else if (rtlpriv->falsealm_cnt.cnt_all < DM_DIG_FA_TH1) | ||
268 | value_igi += 0; | ||
269 | else if (rtlpriv->falsealm_cnt.cnt_all < DM_DIG_FA_TH2) | ||
270 | value_igi++; | ||
271 | else if (rtlpriv->falsealm_cnt.cnt_all >= DM_DIG_FA_TH2) | ||
272 | value_igi += 2; | ||
273 | if (value_igi > DM_DIG_FA_UPPER) | ||
274 | value_igi = DM_DIG_FA_UPPER; | ||
275 | else if (value_igi < DM_DIG_FA_LOWER) | ||
276 | value_igi = DM_DIG_FA_LOWER; | ||
277 | if (rtlpriv->falsealm_cnt.cnt_all > 10000) | ||
278 | value_igi = 0x32; | ||
279 | |||
280 | dm_digtable.cur_igvalue = value_igi; | ||
281 | rtl92c_dm_write_dig(hw); | ||
282 | } | ||
283 | |||
284 | static void rtl92c_dm_ctrl_initgain_by_rssi(struct ieee80211_hw *hw) | ||
285 | { | ||
286 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
287 | |||
288 | if (rtlpriv->falsealm_cnt.cnt_all > dm_digtable.fa_highthresh) { | ||
289 | if ((dm_digtable.backoff_val - 2) < | ||
290 | dm_digtable.backoff_val_range_min) | ||
291 | dm_digtable.backoff_val = | ||
292 | dm_digtable.backoff_val_range_min; | ||
293 | else | ||
294 | dm_digtable.backoff_val -= 2; | ||
295 | } else if (rtlpriv->falsealm_cnt.cnt_all < dm_digtable.fa_lowthresh) { | ||
296 | if ((dm_digtable.backoff_val + 2) > | ||
297 | dm_digtable.backoff_val_range_max) | ||
298 | dm_digtable.backoff_val = | ||
299 | dm_digtable.backoff_val_range_max; | ||
300 | else | ||
301 | dm_digtable.backoff_val += 2; | ||
302 | } | ||
303 | |||
304 | if ((dm_digtable.rssi_val_min + 10 - dm_digtable.backoff_val) > | ||
305 | dm_digtable.rx_gain_range_max) | ||
306 | dm_digtable.cur_igvalue = dm_digtable.rx_gain_range_max; | ||
307 | else if ((dm_digtable.rssi_val_min + 10 - | ||
308 | dm_digtable.backoff_val) < dm_digtable.rx_gain_range_min) | ||
309 | dm_digtable.cur_igvalue = dm_digtable.rx_gain_range_min; | ||
310 | else | ||
311 | dm_digtable.cur_igvalue = dm_digtable.rssi_val_min + 10 - | ||
312 | dm_digtable.backoff_val; | ||
313 | |||
314 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
315 | ("rssi_val_min = %x backoff_val %x\n", | ||
316 | dm_digtable.rssi_val_min, dm_digtable.backoff_val)); | ||
317 | |||
318 | rtl92c_dm_write_dig(hw); | ||
319 | } | ||
320 | |||
321 | static void rtl92c_dm_initial_gain_multi_sta(struct ieee80211_hw *hw) | ||
322 | { | ||
323 | static u8 initialized; /* initialized to false */ | ||
324 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
325 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
326 | long rssi_strength = rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
327 | bool multi_sta = false; | ||
328 | |||
329 | if (mac->opmode == NL80211_IFTYPE_ADHOC) | ||
330 | multi_sta = true; | ||
331 | |||
332 | if ((multi_sta == false) || (dm_digtable.cursta_connectctate != | ||
333 | DIG_STA_DISCONNECT)) { | ||
334 | initialized = false; | ||
335 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_MAX; | ||
336 | return; | ||
337 | } else if (initialized == false) { | ||
338 | initialized = true; | ||
339 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_0; | ||
340 | dm_digtable.cur_igvalue = 0x20; | ||
341 | rtl92c_dm_write_dig(hw); | ||
342 | } | ||
343 | |||
344 | if (dm_digtable.curmultista_connectstate == DIG_MULTISTA_CONNECT) { | ||
345 | if ((rssi_strength < dm_digtable.rssi_lowthresh) && | ||
346 | (dm_digtable.dig_ext_port_stage != DIG_EXT_PORT_STAGE_1)) { | ||
347 | |||
348 | if (dm_digtable.dig_ext_port_stage == | ||
349 | DIG_EXT_PORT_STAGE_2) { | ||
350 | dm_digtable.cur_igvalue = 0x20; | ||
351 | rtl92c_dm_write_dig(hw); | ||
352 | } | ||
353 | |||
354 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_1; | ||
355 | } else if (rssi_strength > dm_digtable.rssi_highthresh) { | ||
356 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_2; | ||
357 | rtl92c_dm_ctrl_initgain_by_fa(hw); | ||
358 | } | ||
359 | } else if (dm_digtable.dig_ext_port_stage != DIG_EXT_PORT_STAGE_0) { | ||
360 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_0; | ||
361 | dm_digtable.cur_igvalue = 0x20; | ||
362 | rtl92c_dm_write_dig(hw); | ||
363 | } | ||
364 | |||
365 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
366 | ("curmultista_connectstate = " | ||
367 | "%x dig_ext_port_stage %x\n", | ||
368 | dm_digtable.curmultista_connectstate, | ||
369 | dm_digtable.dig_ext_port_stage)); | ||
370 | } | ||
371 | |||
372 | static void rtl92c_dm_initial_gain_sta(struct ieee80211_hw *hw) | ||
373 | { | ||
374 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
375 | |||
376 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
377 | ("presta_connectstate = %x," | ||
378 | " cursta_connectctate = %x\n", | ||
379 | dm_digtable.presta_connectstate, | ||
380 | dm_digtable.cursta_connectctate)); | ||
381 | |||
382 | if (dm_digtable.presta_connectstate == dm_digtable.cursta_connectctate | ||
383 | || dm_digtable.cursta_connectctate == DIG_STA_BEFORE_CONNECT | ||
384 | || dm_digtable.cursta_connectctate == DIG_STA_CONNECT) { | ||
385 | |||
386 | if (dm_digtable.cursta_connectctate != DIG_STA_DISCONNECT) { | ||
387 | dm_digtable.rssi_val_min = | ||
388 | rtl92c_dm_initial_gain_min_pwdb(hw); | ||
389 | rtl92c_dm_ctrl_initgain_by_rssi(hw); | ||
390 | } | ||
391 | } else { | ||
392 | dm_digtable.rssi_val_min = 0; | ||
393 | dm_digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_MAX; | ||
394 | dm_digtable.backoff_val = DM_DIG_BACKOFF_DEFAULT; | ||
395 | dm_digtable.cur_igvalue = 0x20; | ||
396 | dm_digtable.pre_igvalue = 0; | ||
397 | rtl92c_dm_write_dig(hw); | ||
398 | } | ||
399 | } | ||
400 | |||
401 | static void rtl92c_dm_cck_packet_detection_thresh(struct ieee80211_hw *hw) | ||
402 | { | ||
403 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
404 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
405 | |||
406 | if (dm_digtable.cursta_connectctate == DIG_STA_CONNECT) { | ||
407 | dm_digtable.rssi_val_min = rtl92c_dm_initial_gain_min_pwdb(hw); | ||
408 | |||
409 | if (dm_digtable.pre_cck_pd_state == CCK_PD_STAGE_LowRssi) { | ||
410 | if (dm_digtable.rssi_val_min <= 25) | ||
411 | dm_digtable.cur_cck_pd_state = | ||
412 | CCK_PD_STAGE_LowRssi; | ||
413 | else | ||
414 | dm_digtable.cur_cck_pd_state = | ||
415 | CCK_PD_STAGE_HighRssi; | ||
416 | } else { | ||
417 | if (dm_digtable.rssi_val_min <= 20) | ||
418 | dm_digtable.cur_cck_pd_state = | ||
419 | CCK_PD_STAGE_LowRssi; | ||
420 | else | ||
421 | dm_digtable.cur_cck_pd_state = | ||
422 | CCK_PD_STAGE_HighRssi; | ||
423 | } | ||
424 | } else { | ||
425 | dm_digtable.cur_cck_pd_state = CCK_PD_STAGE_MAX; | ||
426 | } | ||
427 | |||
428 | if (dm_digtable.pre_cck_pd_state != dm_digtable.cur_cck_pd_state) { | ||
429 | if (dm_digtable.cur_cck_pd_state == CCK_PD_STAGE_LowRssi) { | ||
430 | if (rtlpriv->falsealm_cnt.cnt_cck_fail > 800) | ||
431 | dm_digtable.cur_cck_fa_state = | ||
432 | CCK_FA_STAGE_High; | ||
433 | else | ||
434 | dm_digtable.cur_cck_fa_state = CCK_FA_STAGE_Low; | ||
435 | |||
436 | if (dm_digtable.pre_cck_fa_state != | ||
437 | dm_digtable.cur_cck_fa_state) { | ||
438 | if (dm_digtable.cur_cck_fa_state == | ||
439 | CCK_FA_STAGE_Low) | ||
440 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, | ||
441 | 0x83); | ||
442 | else | ||
443 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, | ||
444 | 0xcd); | ||
445 | |||
446 | dm_digtable.pre_cck_fa_state = | ||
447 | dm_digtable.cur_cck_fa_state; | ||
448 | } | ||
449 | |||
450 | rtl_set_bbreg(hw, RCCK0_SYSTEM, MASKBYTE1, 0x40); | ||
451 | |||
452 | if (IS_92C_SERIAL(rtlhal->version)) | ||
453 | rtl_set_bbreg(hw, RCCK0_FALSEALARMREPORT, | ||
454 | MASKBYTE2, 0xd7); | ||
455 | } else { | ||
456 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, 0xcd); | ||
457 | rtl_set_bbreg(hw, RCCK0_SYSTEM, MASKBYTE1, 0x47); | ||
458 | |||
459 | if (IS_92C_SERIAL(rtlhal->version)) | ||
460 | rtl_set_bbreg(hw, RCCK0_FALSEALARMREPORT, | ||
461 | MASKBYTE2, 0xd3); | ||
462 | } | ||
463 | dm_digtable.pre_cck_pd_state = dm_digtable.cur_cck_pd_state; | ||
464 | } | ||
465 | |||
466 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
467 | ("CCKPDStage=%x\n", dm_digtable.cur_cck_pd_state)); | ||
468 | |||
469 | RT_TRACE(rtlpriv, COMP_DIG, DBG_TRACE, | ||
470 | ("is92C=%x\n", IS_92C_SERIAL(rtlhal->version))); | ||
471 | } | ||
472 | |||
473 | static void rtl92c_dm_ctrl_initgain_by_twoport(struct ieee80211_hw *hw) | ||
474 | { | ||
475 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
476 | |||
477 | if (mac->act_scanning == true) | ||
478 | return; | ||
479 | |||
480 | if (mac->link_state >= MAC80211_LINKED) | ||
481 | dm_digtable.cursta_connectctate = DIG_STA_CONNECT; | ||
482 | else | ||
483 | dm_digtable.cursta_connectctate = DIG_STA_DISCONNECT; | ||
484 | |||
485 | rtl92c_dm_initial_gain_sta(hw); | ||
486 | rtl92c_dm_initial_gain_multi_sta(hw); | ||
487 | rtl92c_dm_cck_packet_detection_thresh(hw); | ||
488 | |||
489 | dm_digtable.presta_connectstate = dm_digtable.cursta_connectctate; | ||
490 | |||
491 | } | ||
492 | |||
493 | static void rtl92c_dm_dig(struct ieee80211_hw *hw) | ||
494 | { | ||
495 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
496 | |||
497 | if (rtlpriv->dm.dm_initialgain_enable == false) | ||
498 | return; | ||
499 | if (dm_digtable.dig_enable_flag == false) | ||
500 | return; | ||
501 | |||
502 | rtl92c_dm_ctrl_initgain_by_twoport(hw); | ||
503 | |||
504 | } | ||
505 | |||
506 | static void rtl92c_dm_init_dynamic_txpower(struct ieee80211_hw *hw) | ||
507 | { | ||
508 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
509 | |||
510 | rtlpriv->dm.dynamic_txpower_enable = false; | ||
511 | |||
512 | rtlpriv->dm.last_dtp_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
513 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
514 | } | ||
515 | |||
516 | void rtl92c_dm_write_dig(struct ieee80211_hw *hw) | ||
517 | { | ||
518 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
519 | |||
520 | RT_TRACE(rtlpriv, COMP_DIG, DBG_LOUD, | ||
521 | ("cur_igvalue = 0x%x, " | ||
522 | "pre_igvalue = 0x%x, backoff_val = %d\n", | ||
523 | dm_digtable.cur_igvalue, dm_digtable.pre_igvalue, | ||
524 | dm_digtable.backoff_val)); | ||
525 | |||
526 | if (dm_digtable.pre_igvalue != dm_digtable.cur_igvalue) { | ||
527 | rtl_set_bbreg(hw, ROFDM0_XAAGCCORE1, 0x7f, | ||
528 | dm_digtable.cur_igvalue); | ||
529 | rtl_set_bbreg(hw, ROFDM0_XBAGCCORE1, 0x7f, | ||
530 | dm_digtable.cur_igvalue); | ||
531 | |||
532 | dm_digtable.pre_igvalue = dm_digtable.cur_igvalue; | ||
533 | } | ||
534 | } | ||
535 | EXPORT_SYMBOL(rtl92c_dm_write_dig); | ||
536 | |||
537 | static void rtl92c_dm_pwdb_monitor(struct ieee80211_hw *hw) | ||
538 | { | ||
539 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
540 | long tmpentry_max_pwdb = 0, tmpentry_min_pwdb = 0xff; | ||
541 | |||
542 | u8 h2c_parameter[3] = { 0 }; | ||
543 | |||
544 | return; | ||
545 | |||
546 | if (tmpentry_max_pwdb != 0) { | ||
547 | rtlpriv->dm.entry_max_undecoratedsmoothed_pwdb = | ||
548 | tmpentry_max_pwdb; | ||
549 | } else { | ||
550 | rtlpriv->dm.entry_max_undecoratedsmoothed_pwdb = 0; | ||
551 | } | ||
552 | |||
553 | if (tmpentry_min_pwdb != 0xff) { | ||
554 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb = | ||
555 | tmpentry_min_pwdb; | ||
556 | } else { | ||
557 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb = 0; | ||
558 | } | ||
559 | |||
560 | h2c_parameter[2] = (u8) (rtlpriv->dm.undecorated_smoothed_pwdb & 0xFF); | ||
561 | h2c_parameter[0] = 0; | ||
562 | |||
563 | rtl92c_fill_h2c_cmd(hw, H2C_RSSI_REPORT, 3, h2c_parameter); | ||
564 | } | ||
565 | |||
566 | void rtl92c_dm_init_edca_turbo(struct ieee80211_hw *hw) | ||
567 | { | ||
568 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
569 | rtlpriv->dm.current_turbo_edca = false; | ||
570 | rtlpriv->dm.is_any_nonbepkts = false; | ||
571 | rtlpriv->dm.is_cur_rdlstate = false; | ||
572 | } | ||
573 | EXPORT_SYMBOL(rtl92c_dm_init_edca_turbo); | ||
574 | |||
575 | static void rtl92c_dm_check_edca_turbo(struct ieee80211_hw *hw) | ||
576 | { | ||
577 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
578 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
579 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
580 | |||
581 | static u64 last_txok_cnt; | ||
582 | static u64 last_rxok_cnt; | ||
583 | static u32 last_bt_edca_ul; | ||
584 | static u32 last_bt_edca_dl; | ||
585 | u64 cur_txok_cnt = 0; | ||
586 | u64 cur_rxok_cnt = 0; | ||
587 | u32 edca_be_ul = 0x5ea42b; | ||
588 | u32 edca_be_dl = 0x5ea42b; | ||
589 | bool bt_change_edca = false; | ||
590 | |||
591 | if ((last_bt_edca_ul != rtlpcipriv->bt_coexist.bt_edca_ul) || | ||
592 | (last_bt_edca_dl != rtlpcipriv->bt_coexist.bt_edca_dl)) { | ||
593 | rtlpriv->dm.current_turbo_edca = false; | ||
594 | last_bt_edca_ul = rtlpcipriv->bt_coexist.bt_edca_ul; | ||
595 | last_bt_edca_dl = rtlpcipriv->bt_coexist.bt_edca_dl; | ||
596 | } | ||
597 | |||
598 | if (rtlpcipriv->bt_coexist.bt_edca_ul != 0) { | ||
599 | edca_be_ul = rtlpcipriv->bt_coexist.bt_edca_ul; | ||
600 | bt_change_edca = true; | ||
601 | } | ||
602 | |||
603 | if (rtlpcipriv->bt_coexist.bt_edca_dl != 0) { | ||
604 | edca_be_ul = rtlpcipriv->bt_coexist.bt_edca_dl; | ||
605 | bt_change_edca = true; | ||
606 | } | ||
607 | |||
608 | if (mac->link_state != MAC80211_LINKED) { | ||
609 | rtlpriv->dm.current_turbo_edca = false; | ||
610 | return; | ||
611 | } | ||
612 | |||
613 | if ((!mac->ht_enable) && (!rtlpcipriv->bt_coexist.bt_coexistence)) { | ||
614 | if (!(edca_be_ul & 0xffff0000)) | ||
615 | edca_be_ul |= 0x005e0000; | ||
616 | |||
617 | if (!(edca_be_dl & 0xffff0000)) | ||
618 | edca_be_dl |= 0x005e0000; | ||
619 | } | ||
620 | |||
621 | if ((bt_change_edca) || ((!rtlpriv->dm.is_any_nonbepkts) && | ||
622 | (!rtlpriv->dm.disable_framebursting))) { | ||
623 | |||
624 | cur_txok_cnt = rtlpriv->stats.txbytesunicast - last_txok_cnt; | ||
625 | cur_rxok_cnt = rtlpriv->stats.rxbytesunicast - last_rxok_cnt; | ||
626 | |||
627 | if (cur_rxok_cnt > 4 * cur_txok_cnt) { | ||
628 | if (!rtlpriv->dm.is_cur_rdlstate || | ||
629 | !rtlpriv->dm.current_turbo_edca) { | ||
630 | rtl_write_dword(rtlpriv, | ||
631 | REG_EDCA_BE_PARAM, | ||
632 | edca_be_dl); | ||
633 | rtlpriv->dm.is_cur_rdlstate = true; | ||
634 | } | ||
635 | } else { | ||
636 | if (rtlpriv->dm.is_cur_rdlstate || | ||
637 | !rtlpriv->dm.current_turbo_edca) { | ||
638 | rtl_write_dword(rtlpriv, | ||
639 | REG_EDCA_BE_PARAM, | ||
640 | edca_be_ul); | ||
641 | rtlpriv->dm.is_cur_rdlstate = false; | ||
642 | } | ||
643 | } | ||
644 | rtlpriv->dm.current_turbo_edca = true; | ||
645 | } else { | ||
646 | if (rtlpriv->dm.current_turbo_edca) { | ||
647 | u8 tmp = AC0_BE; | ||
648 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
649 | HW_VAR_AC_PARAM, | ||
650 | (u8 *) (&tmp)); | ||
651 | rtlpriv->dm.current_turbo_edca = false; | ||
652 | } | ||
653 | } | ||
654 | |||
655 | rtlpriv->dm.is_any_nonbepkts = false; | ||
656 | last_txok_cnt = rtlpriv->stats.txbytesunicast; | ||
657 | last_rxok_cnt = rtlpriv->stats.rxbytesunicast; | ||
658 | } | ||
659 | |||
660 | static void rtl92c_dm_txpower_tracking_callback_thermalmeter(struct ieee80211_hw | ||
661 | *hw) | ||
662 | { | ||
663 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
664 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
665 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
666 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
667 | u8 thermalvalue, delta, delta_lck, delta_iqk; | ||
668 | long ele_a, ele_d, temp_cck, val_x, value32; | ||
669 | long val_y, ele_c = 0; | ||
670 | u8 ofdm_index[2], cck_index = 0, ofdm_index_old[2], cck_index_old = 0; | ||
671 | int i; | ||
672 | bool is2t = IS_92C_SERIAL(rtlhal->version); | ||
673 | u8 txpwr_level[2] = {0, 0}; | ||
674 | u8 ofdm_min_index = 6, rf; | ||
675 | |||
676 | rtlpriv->dm.txpower_trackinginit = true; | ||
677 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
678 | ("rtl92c_dm_txpower_tracking_callback_thermalmeter\n")); | ||
679 | |||
680 | thermalvalue = (u8) rtl_get_rfreg(hw, RF90_PATH_A, RF_T_METER, 0x1f); | ||
681 | |||
682 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
683 | ("Readback Thermal Meter = 0x%x pre thermal meter 0x%x " | ||
684 | "eeprom_thermalmeter 0x%x\n", | ||
685 | thermalvalue, rtlpriv->dm.thermalvalue, | ||
686 | rtlefuse->eeprom_thermalmeter)); | ||
687 | |||
688 | rtl92c_phy_ap_calibrate(hw, (thermalvalue - | ||
689 | rtlefuse->eeprom_thermalmeter)); | ||
690 | if (is2t) | ||
691 | rf = 2; | ||
692 | else | ||
693 | rf = 1; | ||
694 | |||
695 | if (thermalvalue) { | ||
696 | ele_d = rtl_get_bbreg(hw, ROFDM0_XATXIQIMBALANCE, | ||
697 | MASKDWORD) & MASKOFDM_D; | ||
698 | |||
699 | for (i = 0; i < OFDM_TABLE_LENGTH; i++) { | ||
700 | if (ele_d == (ofdmswing_table[i] & MASKOFDM_D)) { | ||
701 | ofdm_index_old[0] = (u8) i; | ||
702 | |||
703 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
704 | ("Initial pathA ele_d reg0x%x = 0x%lx, " | ||
705 | "ofdm_index=0x%x\n", | ||
706 | ROFDM0_XATXIQIMBALANCE, | ||
707 | ele_d, ofdm_index_old[0])); | ||
708 | break; | ||
709 | } | ||
710 | } | ||
711 | |||
712 | if (is2t) { | ||
713 | ele_d = rtl_get_bbreg(hw, ROFDM0_XBTXIQIMBALANCE, | ||
714 | MASKDWORD) & MASKOFDM_D; | ||
715 | |||
716 | for (i = 0; i < OFDM_TABLE_LENGTH; i++) { | ||
717 | if (ele_d == (ofdmswing_table[i] & | ||
718 | MASKOFDM_D)) { | ||
719 | |||
720 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, | ||
721 | DBG_LOUD, | ||
722 | ("Initial pathB ele_d reg0x%x = " | ||
723 | "0x%lx, ofdm_index=0x%x\n", | ||
724 | ROFDM0_XBTXIQIMBALANCE, ele_d, | ||
725 | ofdm_index_old[1])); | ||
726 | break; | ||
727 | } | ||
728 | } | ||
729 | } | ||
730 | |||
731 | temp_cck = | ||
732 | rtl_get_bbreg(hw, RCCK0_TXFILTER2, MASKDWORD) & MASKCCK; | ||
733 | |||
734 | for (i = 0; i < CCK_TABLE_LENGTH; i++) { | ||
735 | if (rtlpriv->dm.cck_inch14) { | ||
736 | if (memcmp((void *)&temp_cck, | ||
737 | (void *)&cckswing_table_ch14[i][2], | ||
738 | 4) == 0) { | ||
739 | cck_index_old = (u8) i; | ||
740 | |||
741 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, | ||
742 | DBG_LOUD, | ||
743 | ("Initial reg0x%x = 0x%lx, " | ||
744 | "cck_index=0x%x, ch 14 %d\n", | ||
745 | RCCK0_TXFILTER2, temp_cck, | ||
746 | cck_index_old, | ||
747 | rtlpriv->dm.cck_inch14)); | ||
748 | break; | ||
749 | } | ||
750 | } else { | ||
751 | if (memcmp((void *)&temp_cck, | ||
752 | (void *) | ||
753 | &cckswing_table_ch1ch13[i][2], | ||
754 | 4) == 0) { | ||
755 | cck_index_old = (u8) i; | ||
756 | |||
757 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, | ||
758 | DBG_LOUD, | ||
759 | ("Initial reg0x%x = 0x%lx, " | ||
760 | "cck_index=0x%x, ch14 %d\n", | ||
761 | RCCK0_TXFILTER2, temp_cck, | ||
762 | cck_index_old, | ||
763 | rtlpriv->dm.cck_inch14)); | ||
764 | break; | ||
765 | } | ||
766 | } | ||
767 | } | ||
768 | |||
769 | if (!rtlpriv->dm.thermalvalue) { | ||
770 | rtlpriv->dm.thermalvalue = | ||
771 | rtlefuse->eeprom_thermalmeter; | ||
772 | rtlpriv->dm.thermalvalue_lck = thermalvalue; | ||
773 | rtlpriv->dm.thermalvalue_iqk = thermalvalue; | ||
774 | for (i = 0; i < rf; i++) | ||
775 | rtlpriv->dm.ofdm_index[i] = ofdm_index_old[i]; | ||
776 | rtlpriv->dm.cck_index = cck_index_old; | ||
777 | } | ||
778 | |||
779 | delta = (thermalvalue > rtlpriv->dm.thermalvalue) ? | ||
780 | (thermalvalue - rtlpriv->dm.thermalvalue) : | ||
781 | (rtlpriv->dm.thermalvalue - thermalvalue); | ||
782 | |||
783 | delta_lck = (thermalvalue > rtlpriv->dm.thermalvalue_lck) ? | ||
784 | (thermalvalue - rtlpriv->dm.thermalvalue_lck) : | ||
785 | (rtlpriv->dm.thermalvalue_lck - thermalvalue); | ||
786 | |||
787 | delta_iqk = (thermalvalue > rtlpriv->dm.thermalvalue_iqk) ? | ||
788 | (thermalvalue - rtlpriv->dm.thermalvalue_iqk) : | ||
789 | (rtlpriv->dm.thermalvalue_iqk - thermalvalue); | ||
790 | |||
791 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
792 | ("Readback Thermal Meter = 0x%x pre thermal meter 0x%x " | ||
793 | "eeprom_thermalmeter 0x%x delta 0x%x " | ||
794 | "delta_lck 0x%x delta_iqk 0x%x\n", | ||
795 | thermalvalue, rtlpriv->dm.thermalvalue, | ||
796 | rtlefuse->eeprom_thermalmeter, delta, delta_lck, | ||
797 | delta_iqk)); | ||
798 | |||
799 | if (delta_lck > 1) { | ||
800 | rtlpriv->dm.thermalvalue_lck = thermalvalue; | ||
801 | rtl92c_phy_lc_calibrate(hw); | ||
802 | } | ||
803 | |||
804 | if (delta > 0 && rtlpriv->dm.txpower_track_control) { | ||
805 | if (thermalvalue > rtlpriv->dm.thermalvalue) { | ||
806 | for (i = 0; i < rf; i++) | ||
807 | rtlpriv->dm.ofdm_index[i] -= delta; | ||
808 | rtlpriv->dm.cck_index -= delta; | ||
809 | } else { | ||
810 | for (i = 0; i < rf; i++) | ||
811 | rtlpriv->dm.ofdm_index[i] += delta; | ||
812 | rtlpriv->dm.cck_index += delta; | ||
813 | } | ||
814 | |||
815 | if (is2t) { | ||
816 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
817 | ("temp OFDM_A_index=0x%x, " | ||
818 | "OFDM_B_index=0x%x," | ||
819 | "cck_index=0x%x\n", | ||
820 | rtlpriv->dm.ofdm_index[0], | ||
821 | rtlpriv->dm.ofdm_index[1], | ||
822 | rtlpriv->dm.cck_index)); | ||
823 | } else { | ||
824 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
825 | ("temp OFDM_A_index=0x%x," | ||
826 | "cck_index=0x%x\n", | ||
827 | rtlpriv->dm.ofdm_index[0], | ||
828 | rtlpriv->dm.cck_index)); | ||
829 | } | ||
830 | |||
831 | if (thermalvalue > rtlefuse->eeprom_thermalmeter) { | ||
832 | for (i = 0; i < rf; i++) | ||
833 | ofdm_index[i] = | ||
834 | rtlpriv->dm.ofdm_index[i] | ||
835 | + 1; | ||
836 | cck_index = rtlpriv->dm.cck_index + 1; | ||
837 | } else { | ||
838 | for (i = 0; i < rf; i++) | ||
839 | ofdm_index[i] = | ||
840 | rtlpriv->dm.ofdm_index[i]; | ||
841 | cck_index = rtlpriv->dm.cck_index; | ||
842 | } | ||
843 | |||
844 | for (i = 0; i < rf; i++) { | ||
845 | if (txpwr_level[i] >= 0 && | ||
846 | txpwr_level[i] <= 26) { | ||
847 | if (thermalvalue > | ||
848 | rtlefuse->eeprom_thermalmeter) { | ||
849 | if (delta < 5) | ||
850 | ofdm_index[i] -= 1; | ||
851 | |||
852 | else | ||
853 | ofdm_index[i] -= 2; | ||
854 | } else if (delta > 5 && thermalvalue < | ||
855 | rtlefuse-> | ||
856 | eeprom_thermalmeter) { | ||
857 | ofdm_index[i] += 1; | ||
858 | } | ||
859 | } else if (txpwr_level[i] >= 27 && | ||
860 | txpwr_level[i] <= 32 | ||
861 | && thermalvalue > | ||
862 | rtlefuse->eeprom_thermalmeter) { | ||
863 | if (delta < 5) | ||
864 | ofdm_index[i] -= 1; | ||
865 | |||
866 | else | ||
867 | ofdm_index[i] -= 2; | ||
868 | } else if (txpwr_level[i] >= 32 && | ||
869 | txpwr_level[i] <= 38 && | ||
870 | thermalvalue > | ||
871 | rtlefuse->eeprom_thermalmeter | ||
872 | && delta > 5) { | ||
873 | ofdm_index[i] -= 1; | ||
874 | } | ||
875 | } | ||
876 | |||
877 | if (txpwr_level[i] >= 0 && txpwr_level[i] <= 26) { | ||
878 | if (thermalvalue > | ||
879 | rtlefuse->eeprom_thermalmeter) { | ||
880 | if (delta < 5) | ||
881 | cck_index -= 1; | ||
882 | |||
883 | else | ||
884 | cck_index -= 2; | ||
885 | } else if (delta > 5 && thermalvalue < | ||
886 | rtlefuse->eeprom_thermalmeter) { | ||
887 | cck_index += 1; | ||
888 | } | ||
889 | } else if (txpwr_level[i] >= 27 && | ||
890 | txpwr_level[i] <= 32 && | ||
891 | thermalvalue > | ||
892 | rtlefuse->eeprom_thermalmeter) { | ||
893 | if (delta < 5) | ||
894 | cck_index -= 1; | ||
895 | |||
896 | else | ||
897 | cck_index -= 2; | ||
898 | } else if (txpwr_level[i] >= 32 && | ||
899 | txpwr_level[i] <= 38 && | ||
900 | thermalvalue > rtlefuse->eeprom_thermalmeter | ||
901 | && delta > 5) { | ||
902 | cck_index -= 1; | ||
903 | } | ||
904 | |||
905 | for (i = 0; i < rf; i++) { | ||
906 | if (ofdm_index[i] > OFDM_TABLE_SIZE - 1) | ||
907 | ofdm_index[i] = OFDM_TABLE_SIZE - 1; | ||
908 | |||
909 | else if (ofdm_index[i] < ofdm_min_index) | ||
910 | ofdm_index[i] = ofdm_min_index; | ||
911 | } | ||
912 | |||
913 | if (cck_index > CCK_TABLE_SIZE - 1) | ||
914 | cck_index = CCK_TABLE_SIZE - 1; | ||
915 | else if (cck_index < 0) | ||
916 | cck_index = 0; | ||
917 | |||
918 | if (is2t) { | ||
919 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
920 | ("new OFDM_A_index=0x%x, " | ||
921 | "OFDM_B_index=0x%x," | ||
922 | "cck_index=0x%x\n", | ||
923 | ofdm_index[0], ofdm_index[1], | ||
924 | cck_index)); | ||
925 | } else { | ||
926 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
927 | ("new OFDM_A_index=0x%x," | ||
928 | "cck_index=0x%x\n", | ||
929 | ofdm_index[0], cck_index)); | ||
930 | } | ||
931 | } | ||
932 | |||
933 | if (rtlpriv->dm.txpower_track_control && delta != 0) { | ||
934 | ele_d = | ||
935 | (ofdmswing_table[ofdm_index[0]] & 0xFFC00000) >> 22; | ||
936 | val_x = rtlphy->reg_e94; | ||
937 | val_y = rtlphy->reg_e9c; | ||
938 | |||
939 | if (val_x != 0) { | ||
940 | if ((val_x & 0x00000200) != 0) | ||
941 | val_x = val_x | 0xFFFFFC00; | ||
942 | ele_a = ((val_x * ele_d) >> 8) & 0x000003FF; | ||
943 | |||
944 | if ((val_y & 0x00000200) != 0) | ||
945 | val_y = val_y | 0xFFFFFC00; | ||
946 | ele_c = ((val_y * ele_d) >> 8) & 0x000003FF; | ||
947 | |||
948 | value32 = (ele_d << 22) | | ||
949 | ((ele_c & 0x3F) << 16) | ele_a; | ||
950 | |||
951 | rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, | ||
952 | MASKDWORD, value32); | ||
953 | |||
954 | value32 = (ele_c & 0x000003C0) >> 6; | ||
955 | rtl_set_bbreg(hw, ROFDM0_XCTXAFE, MASKH4BITS, | ||
956 | value32); | ||
957 | |||
958 | value32 = ((val_x * ele_d) >> 7) & 0x01; | ||
959 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
960 | BIT(31), value32); | ||
961 | |||
962 | value32 = ((val_y * ele_d) >> 7) & 0x01; | ||
963 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
964 | BIT(29), value32); | ||
965 | } else { | ||
966 | rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, | ||
967 | MASKDWORD, | ||
968 | ofdmswing_table[ofdm_index[0]]); | ||
969 | |||
970 | rtl_set_bbreg(hw, ROFDM0_XCTXAFE, MASKH4BITS, | ||
971 | 0x00); | ||
972 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
973 | BIT(31) | BIT(29), 0x00); | ||
974 | } | ||
975 | |||
976 | if (!rtlpriv->dm.cck_inch14) { | ||
977 | rtl_write_byte(rtlpriv, 0xa22, | ||
978 | cckswing_table_ch1ch13[cck_index] | ||
979 | [0]); | ||
980 | rtl_write_byte(rtlpriv, 0xa23, | ||
981 | cckswing_table_ch1ch13[cck_index] | ||
982 | [1]); | ||
983 | rtl_write_byte(rtlpriv, 0xa24, | ||
984 | cckswing_table_ch1ch13[cck_index] | ||
985 | [2]); | ||
986 | rtl_write_byte(rtlpriv, 0xa25, | ||
987 | cckswing_table_ch1ch13[cck_index] | ||
988 | [3]); | ||
989 | rtl_write_byte(rtlpriv, 0xa26, | ||
990 | cckswing_table_ch1ch13[cck_index] | ||
991 | [4]); | ||
992 | rtl_write_byte(rtlpriv, 0xa27, | ||
993 | cckswing_table_ch1ch13[cck_index] | ||
994 | [5]); | ||
995 | rtl_write_byte(rtlpriv, 0xa28, | ||
996 | cckswing_table_ch1ch13[cck_index] | ||
997 | [6]); | ||
998 | rtl_write_byte(rtlpriv, 0xa29, | ||
999 | cckswing_table_ch1ch13[cck_index] | ||
1000 | [7]); | ||
1001 | } else { | ||
1002 | rtl_write_byte(rtlpriv, 0xa22, | ||
1003 | cckswing_table_ch14[cck_index] | ||
1004 | [0]); | ||
1005 | rtl_write_byte(rtlpriv, 0xa23, | ||
1006 | cckswing_table_ch14[cck_index] | ||
1007 | [1]); | ||
1008 | rtl_write_byte(rtlpriv, 0xa24, | ||
1009 | cckswing_table_ch14[cck_index] | ||
1010 | [2]); | ||
1011 | rtl_write_byte(rtlpriv, 0xa25, | ||
1012 | cckswing_table_ch14[cck_index] | ||
1013 | [3]); | ||
1014 | rtl_write_byte(rtlpriv, 0xa26, | ||
1015 | cckswing_table_ch14[cck_index] | ||
1016 | [4]); | ||
1017 | rtl_write_byte(rtlpriv, 0xa27, | ||
1018 | cckswing_table_ch14[cck_index] | ||
1019 | [5]); | ||
1020 | rtl_write_byte(rtlpriv, 0xa28, | ||
1021 | cckswing_table_ch14[cck_index] | ||
1022 | [6]); | ||
1023 | rtl_write_byte(rtlpriv, 0xa29, | ||
1024 | cckswing_table_ch14[cck_index] | ||
1025 | [7]); | ||
1026 | } | ||
1027 | |||
1028 | if (is2t) { | ||
1029 | ele_d = (ofdmswing_table[ofdm_index[1]] & | ||
1030 | 0xFFC00000) >> 22; | ||
1031 | |||
1032 | val_x = rtlphy->reg_eb4; | ||
1033 | val_y = rtlphy->reg_ebc; | ||
1034 | |||
1035 | if (val_x != 0) { | ||
1036 | if ((val_x & 0x00000200) != 0) | ||
1037 | val_x = val_x | 0xFFFFFC00; | ||
1038 | ele_a = ((val_x * ele_d) >> 8) & | ||
1039 | 0x000003FF; | ||
1040 | |||
1041 | if ((val_y & 0x00000200) != 0) | ||
1042 | val_y = val_y | 0xFFFFFC00; | ||
1043 | ele_c = ((val_y * ele_d) >> 8) & | ||
1044 | 0x00003FF; | ||
1045 | |||
1046 | value32 = (ele_d << 22) | | ||
1047 | ((ele_c & 0x3F) << 16) | ele_a; | ||
1048 | rtl_set_bbreg(hw, | ||
1049 | ROFDM0_XBTXIQIMBALANCE, | ||
1050 | MASKDWORD, value32); | ||
1051 | |||
1052 | value32 = (ele_c & 0x000003C0) >> 6; | ||
1053 | rtl_set_bbreg(hw, ROFDM0_XDTXAFE, | ||
1054 | MASKH4BITS, value32); | ||
1055 | |||
1056 | value32 = ((val_x * ele_d) >> 7) & 0x01; | ||
1057 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
1058 | BIT(27), value32); | ||
1059 | |||
1060 | value32 = ((val_y * ele_d) >> 7) & 0x01; | ||
1061 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
1062 | BIT(25), value32); | ||
1063 | } else { | ||
1064 | rtl_set_bbreg(hw, | ||
1065 | ROFDM0_XBTXIQIMBALANCE, | ||
1066 | MASKDWORD, | ||
1067 | ofdmswing_table[ofdm_index | ||
1068 | [1]]); | ||
1069 | rtl_set_bbreg(hw, ROFDM0_XDTXAFE, | ||
1070 | MASKH4BITS, 0x00); | ||
1071 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, | ||
1072 | BIT(27) | BIT(25), 0x00); | ||
1073 | } | ||
1074 | |||
1075 | } | ||
1076 | } | ||
1077 | |||
1078 | if (delta_iqk > 3) { | ||
1079 | rtlpriv->dm.thermalvalue_iqk = thermalvalue; | ||
1080 | rtl92c_phy_iq_calibrate(hw, false); | ||
1081 | } | ||
1082 | |||
1083 | if (rtlpriv->dm.txpower_track_control) | ||
1084 | rtlpriv->dm.thermalvalue = thermalvalue; | ||
1085 | } | ||
1086 | |||
1087 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, ("<===\n")); | ||
1088 | |||
1089 | } | ||
1090 | |||
1091 | static void rtl92c_dm_initialize_txpower_tracking_thermalmeter( | ||
1092 | struct ieee80211_hw *hw) | ||
1093 | { | ||
1094 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1095 | |||
1096 | rtlpriv->dm.txpower_tracking = true; | ||
1097 | rtlpriv->dm.txpower_trackinginit = false; | ||
1098 | |||
1099 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
1100 | ("pMgntInfo->txpower_tracking = %d\n", | ||
1101 | rtlpriv->dm.txpower_tracking)); | ||
1102 | } | ||
1103 | |||
1104 | static void rtl92c_dm_initialize_txpower_tracking(struct ieee80211_hw *hw) | ||
1105 | { | ||
1106 | rtl92c_dm_initialize_txpower_tracking_thermalmeter(hw); | ||
1107 | } | ||
1108 | |||
1109 | static void rtl92c_dm_txpower_tracking_directcall(struct ieee80211_hw *hw) | ||
1110 | { | ||
1111 | rtl92c_dm_txpower_tracking_callback_thermalmeter(hw); | ||
1112 | } | ||
1113 | |||
1114 | static void rtl92c_dm_check_txpower_tracking_thermal_meter( | ||
1115 | struct ieee80211_hw *hw) | ||
1116 | { | ||
1117 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1118 | static u8 tm_trigger; | ||
1119 | |||
1120 | if (!rtlpriv->dm.txpower_tracking) | ||
1121 | return; | ||
1122 | |||
1123 | if (!tm_trigger) { | ||
1124 | rtl_set_rfreg(hw, RF90_PATH_A, RF_T_METER, RFREG_OFFSET_MASK, | ||
1125 | 0x60); | ||
1126 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
1127 | ("Trigger 92S Thermal Meter!!\n")); | ||
1128 | tm_trigger = 1; | ||
1129 | return; | ||
1130 | } else { | ||
1131 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
1132 | ("Schedule TxPowerTracking direct call!!\n")); | ||
1133 | rtl92c_dm_txpower_tracking_directcall(hw); | ||
1134 | tm_trigger = 0; | ||
1135 | } | ||
1136 | } | ||
1137 | |||
1138 | void rtl92c_dm_check_txpower_tracking(struct ieee80211_hw *hw) | ||
1139 | { | ||
1140 | rtl92c_dm_check_txpower_tracking_thermal_meter(hw); | ||
1141 | } | ||
1142 | EXPORT_SYMBOL(rtl92c_dm_check_txpower_tracking); | ||
1143 | |||
1144 | void rtl92c_dm_init_rate_adaptive_mask(struct ieee80211_hw *hw) | ||
1145 | { | ||
1146 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1147 | struct rate_adaptive *p_ra = &(rtlpriv->ra); | ||
1148 | |||
1149 | p_ra->ratr_state = DM_RATR_STA_INIT; | ||
1150 | p_ra->pre_ratr_state = DM_RATR_STA_INIT; | ||
1151 | |||
1152 | if (rtlpriv->dm.dm_type == DM_TYPE_BYDRIVER) | ||
1153 | rtlpriv->dm.useramask = true; | ||
1154 | else | ||
1155 | rtlpriv->dm.useramask = false; | ||
1156 | |||
1157 | } | ||
1158 | EXPORT_SYMBOL(rtl92c_dm_init_rate_adaptive_mask); | ||
1159 | |||
1160 | static void rtl92c_dm_refresh_rate_adaptive_mask(struct ieee80211_hw *hw) | ||
1161 | { | ||
1162 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1163 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1164 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1165 | struct rate_adaptive *p_ra = &(rtlpriv->ra); | ||
1166 | u32 low_rssithresh_for_ra, high_rssithresh_for_ra; | ||
1167 | struct ieee80211_sta *sta = NULL; | ||
1168 | |||
1169 | if (is_hal_stop(rtlhal)) { | ||
1170 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, | ||
1171 | ("<---- driver is going to unload\n")); | ||
1172 | return; | ||
1173 | } | ||
1174 | |||
1175 | if (!rtlpriv->dm.useramask) { | ||
1176 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, | ||
1177 | ("<---- driver does not control rate adaptive mask\n")); | ||
1178 | return; | ||
1179 | } | ||
1180 | |||
1181 | if (mac->link_state == MAC80211_LINKED && | ||
1182 | mac->opmode == NL80211_IFTYPE_STATION) { | ||
1183 | switch (p_ra->pre_ratr_state) { | ||
1184 | case DM_RATR_STA_HIGH: | ||
1185 | high_rssithresh_for_ra = 50; | ||
1186 | low_rssithresh_for_ra = 20; | ||
1187 | break; | ||
1188 | case DM_RATR_STA_MIDDLE: | ||
1189 | high_rssithresh_for_ra = 55; | ||
1190 | low_rssithresh_for_ra = 20; | ||
1191 | break; | ||
1192 | case DM_RATR_STA_LOW: | ||
1193 | high_rssithresh_for_ra = 50; | ||
1194 | low_rssithresh_for_ra = 25; | ||
1195 | break; | ||
1196 | default: | ||
1197 | high_rssithresh_for_ra = 50; | ||
1198 | low_rssithresh_for_ra = 20; | ||
1199 | break; | ||
1200 | } | ||
1201 | |||
1202 | if (rtlpriv->dm.undecorated_smoothed_pwdb > | ||
1203 | (long)high_rssithresh_for_ra) | ||
1204 | p_ra->ratr_state = DM_RATR_STA_HIGH; | ||
1205 | else if (rtlpriv->dm.undecorated_smoothed_pwdb > | ||
1206 | (long)low_rssithresh_for_ra) | ||
1207 | p_ra->ratr_state = DM_RATR_STA_MIDDLE; | ||
1208 | else | ||
1209 | p_ra->ratr_state = DM_RATR_STA_LOW; | ||
1210 | |||
1211 | if (p_ra->pre_ratr_state != p_ra->ratr_state) { | ||
1212 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, | ||
1213 | ("RSSI = %ld\n", | ||
1214 | rtlpriv->dm.undecorated_smoothed_pwdb)); | ||
1215 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, | ||
1216 | ("RSSI_LEVEL = %d\n", p_ra->ratr_state)); | ||
1217 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, | ||
1218 | ("PreState = %d, CurState = %d\n", | ||
1219 | p_ra->pre_ratr_state, p_ra->ratr_state)); | ||
1220 | |||
1221 | rcu_read_lock(); | ||
1222 | sta = ieee80211_find_sta(mac->vif, mac->bssid); | ||
1223 | rtlpriv->cfg->ops->update_rate_tbl(hw, sta, | ||
1224 | p_ra->ratr_state); | ||
1225 | |||
1226 | p_ra->pre_ratr_state = p_ra->ratr_state; | ||
1227 | rcu_read_unlock(); | ||
1228 | } | ||
1229 | } | ||
1230 | } | ||
1231 | |||
1232 | static void rtl92c_dm_init_dynamic_bb_powersaving(struct ieee80211_hw *hw) | ||
1233 | { | ||
1234 | dm_pstable.pre_ccastate = CCA_MAX; | ||
1235 | dm_pstable.cur_ccasate = CCA_MAX; | ||
1236 | dm_pstable.pre_rfstate = RF_MAX; | ||
1237 | dm_pstable.cur_rfstate = RF_MAX; | ||
1238 | dm_pstable.rssi_val_min = 0; | ||
1239 | } | ||
1240 | |||
1241 | void rtl92c_dm_rf_saving(struct ieee80211_hw *hw, u8 bforce_in_normal) | ||
1242 | { | ||
1243 | static u8 initialize; | ||
1244 | static u32 reg_874, reg_c70, reg_85c, reg_a74; | ||
1245 | |||
1246 | if (initialize == 0) { | ||
1247 | reg_874 = (rtl_get_bbreg(hw, RFPGA0_XCD_RFINTERFACESW, | ||
1248 | MASKDWORD) & 0x1CC000) >> 14; | ||
1249 | |||
1250 | reg_c70 = (rtl_get_bbreg(hw, ROFDM0_AGCPARAMETER1, | ||
1251 | MASKDWORD) & BIT(3)) >> 3; | ||
1252 | |||
1253 | reg_85c = (rtl_get_bbreg(hw, RFPGA0_XCD_SWITCHCONTROL, | ||
1254 | MASKDWORD) & 0xFF000000) >> 24; | ||
1255 | |||
1256 | reg_a74 = (rtl_get_bbreg(hw, 0xa74, MASKDWORD) & 0xF000) >> 12; | ||
1257 | |||
1258 | initialize = 1; | ||
1259 | } | ||
1260 | |||
1261 | if (!bforce_in_normal) { | ||
1262 | if (dm_pstable.rssi_val_min != 0) { | ||
1263 | if (dm_pstable.pre_rfstate == RF_NORMAL) { | ||
1264 | if (dm_pstable.rssi_val_min >= 30) | ||
1265 | dm_pstable.cur_rfstate = RF_SAVE; | ||
1266 | else | ||
1267 | dm_pstable.cur_rfstate = RF_NORMAL; | ||
1268 | } else { | ||
1269 | if (dm_pstable.rssi_val_min <= 25) | ||
1270 | dm_pstable.cur_rfstate = RF_NORMAL; | ||
1271 | else | ||
1272 | dm_pstable.cur_rfstate = RF_SAVE; | ||
1273 | } | ||
1274 | } else { | ||
1275 | dm_pstable.cur_rfstate = RF_MAX; | ||
1276 | } | ||
1277 | } else { | ||
1278 | dm_pstable.cur_rfstate = RF_NORMAL; | ||
1279 | } | ||
1280 | |||
1281 | if (dm_pstable.pre_rfstate != dm_pstable.cur_rfstate) { | ||
1282 | if (dm_pstable.cur_rfstate == RF_SAVE) { | ||
1283 | rtl_set_bbreg(hw, RFPGA0_XCD_RFINTERFACESW, | ||
1284 | 0x1C0000, 0x2); | ||
1285 | rtl_set_bbreg(hw, ROFDM0_AGCPARAMETER1, BIT(3), 0); | ||
1286 | rtl_set_bbreg(hw, RFPGA0_XCD_SWITCHCONTROL, | ||
1287 | 0xFF000000, 0x63); | ||
1288 | rtl_set_bbreg(hw, RFPGA0_XCD_RFINTERFACESW, | ||
1289 | 0xC000, 0x2); | ||
1290 | rtl_set_bbreg(hw, 0xa74, 0xF000, 0x3); | ||
1291 | rtl_set_bbreg(hw, 0x818, BIT(28), 0x0); | ||
1292 | rtl_set_bbreg(hw, 0x818, BIT(28), 0x1); | ||
1293 | } else { | ||
1294 | rtl_set_bbreg(hw, RFPGA0_XCD_RFINTERFACESW, | ||
1295 | 0x1CC000, reg_874); | ||
1296 | rtl_set_bbreg(hw, ROFDM0_AGCPARAMETER1, BIT(3), | ||
1297 | reg_c70); | ||
1298 | rtl_set_bbreg(hw, RFPGA0_XCD_SWITCHCONTROL, 0xFF000000, | ||
1299 | reg_85c); | ||
1300 | rtl_set_bbreg(hw, 0xa74, 0xF000, reg_a74); | ||
1301 | rtl_set_bbreg(hw, 0x818, BIT(28), 0x0); | ||
1302 | } | ||
1303 | |||
1304 | dm_pstable.pre_rfstate = dm_pstable.cur_rfstate; | ||
1305 | } | ||
1306 | } | ||
1307 | EXPORT_SYMBOL(rtl92c_dm_rf_saving); | ||
1308 | |||
1309 | static void rtl92c_dm_dynamic_bb_powersaving(struct ieee80211_hw *hw) | ||
1310 | { | ||
1311 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1312 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1313 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1314 | |||
1315 | if (((mac->link_state == MAC80211_NOLINK)) && | ||
1316 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0)) { | ||
1317 | dm_pstable.rssi_val_min = 0; | ||
1318 | RT_TRACE(rtlpriv, DBG_LOUD, DBG_LOUD, | ||
1319 | ("Not connected to any\n")); | ||
1320 | } | ||
1321 | |||
1322 | if (mac->link_state == MAC80211_LINKED) { | ||
1323 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
1324 | dm_pstable.rssi_val_min = | ||
1325 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
1326 | RT_TRACE(rtlpriv, DBG_LOUD, DBG_LOUD, | ||
1327 | ("AP Client PWDB = 0x%lx\n", | ||
1328 | dm_pstable.rssi_val_min)); | ||
1329 | } else { | ||
1330 | dm_pstable.rssi_val_min = | ||
1331 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
1332 | RT_TRACE(rtlpriv, DBG_LOUD, DBG_LOUD, | ||
1333 | ("STA Default Port PWDB = 0x%lx\n", | ||
1334 | dm_pstable.rssi_val_min)); | ||
1335 | } | ||
1336 | } else { | ||
1337 | dm_pstable.rssi_val_min = | ||
1338 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
1339 | |||
1340 | RT_TRACE(rtlpriv, DBG_LOUD, DBG_LOUD, | ||
1341 | ("AP Ext Port PWDB = 0x%lx\n", | ||
1342 | dm_pstable.rssi_val_min)); | ||
1343 | } | ||
1344 | |||
1345 | if (IS_92C_SERIAL(rtlhal->version)) | ||
1346 | ;/* rtl92c_dm_1r_cca(hw); */ | ||
1347 | else | ||
1348 | rtl92c_dm_rf_saving(hw, false); | ||
1349 | } | ||
1350 | |||
1351 | void rtl92c_dm_init(struct ieee80211_hw *hw) | ||
1352 | { | ||
1353 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1354 | |||
1355 | rtlpriv->dm.dm_type = DM_TYPE_BYDRIVER; | ||
1356 | rtl92c_dm_diginit(hw); | ||
1357 | rtl92c_dm_init_dynamic_txpower(hw); | ||
1358 | rtl92c_dm_init_edca_turbo(hw); | ||
1359 | rtl92c_dm_init_rate_adaptive_mask(hw); | ||
1360 | rtl92c_dm_initialize_txpower_tracking(hw); | ||
1361 | rtl92c_dm_init_dynamic_bb_powersaving(hw); | ||
1362 | } | ||
1363 | EXPORT_SYMBOL(rtl92c_dm_init); | ||
1364 | |||
1365 | void rtl92c_dm_dynamic_txpower(struct ieee80211_hw *hw) | ||
1366 | { | ||
1367 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1368 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1369 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1370 | long undecorated_smoothed_pwdb; | ||
1371 | |||
1372 | if (!rtlpriv->dm.dynamic_txpower_enable) | ||
1373 | return; | ||
1374 | |||
1375 | if (rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) { | ||
1376 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
1377 | return; | ||
1378 | } | ||
1379 | |||
1380 | if ((mac->link_state < MAC80211_LINKED) && | ||
1381 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0)) { | ||
1382 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
1383 | ("Not connected to any\n")); | ||
1384 | |||
1385 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
1386 | |||
1387 | rtlpriv->dm.last_dtp_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
1388 | return; | ||
1389 | } | ||
1390 | |||
1391 | if (mac->link_state >= MAC80211_LINKED) { | ||
1392 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
1393 | undecorated_smoothed_pwdb = | ||
1394 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
1395 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1396 | ("AP Client PWDB = 0x%lx\n", | ||
1397 | undecorated_smoothed_pwdb)); | ||
1398 | } else { | ||
1399 | undecorated_smoothed_pwdb = | ||
1400 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
1401 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1402 | ("STA Default Port PWDB = 0x%lx\n", | ||
1403 | undecorated_smoothed_pwdb)); | ||
1404 | } | ||
1405 | } else { | ||
1406 | undecorated_smoothed_pwdb = | ||
1407 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
1408 | |||
1409 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1410 | ("AP Ext Port PWDB = 0x%lx\n", | ||
1411 | undecorated_smoothed_pwdb)); | ||
1412 | } | ||
1413 | |||
1414 | if (undecorated_smoothed_pwdb >= TX_POWER_NEAR_FIELD_THRESH_LVL2) { | ||
1415 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
1416 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1417 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x0)\n")); | ||
1418 | } else if ((undecorated_smoothed_pwdb < | ||
1419 | (TX_POWER_NEAR_FIELD_THRESH_LVL2 - 3)) && | ||
1420 | (undecorated_smoothed_pwdb >= | ||
1421 | TX_POWER_NEAR_FIELD_THRESH_LVL1)) { | ||
1422 | |||
1423 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
1424 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1425 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x10)\n")); | ||
1426 | } else if (undecorated_smoothed_pwdb < | ||
1427 | (TX_POWER_NEAR_FIELD_THRESH_LVL1 - 5)) { | ||
1428 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
1429 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1430 | ("TXHIGHPWRLEVEL_NORMAL\n")); | ||
1431 | } | ||
1432 | |||
1433 | if ((rtlpriv->dm.dynamic_txhighpower_lvl != rtlpriv->dm.last_dtp_lvl)) { | ||
1434 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1435 | ("PHY_SetTxPowerLevel8192S() Channel = %d\n", | ||
1436 | rtlphy->current_channel)); | ||
1437 | rtl92c_phy_set_txpower_level(hw, rtlphy->current_channel); | ||
1438 | } | ||
1439 | |||
1440 | rtlpriv->dm.last_dtp_lvl = rtlpriv->dm.dynamic_txhighpower_lvl; | ||
1441 | } | ||
1442 | |||
1443 | void rtl92c_dm_watchdog(struct ieee80211_hw *hw) | ||
1444 | { | ||
1445 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1446 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1447 | bool fw_current_inpsmode = false; | ||
1448 | bool fw_ps_awake = true; | ||
1449 | |||
1450 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_FW_PSMODE_STATUS, | ||
1451 | (u8 *) (&fw_current_inpsmode)); | ||
1452 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_FWLPS_RF_ON, | ||
1453 | (u8 *) (&fw_ps_awake)); | ||
1454 | |||
1455 | if ((ppsc->rfpwr_state == ERFON) && ((!fw_current_inpsmode) && | ||
1456 | fw_ps_awake) | ||
1457 | && (!ppsc->rfchange_inprogress)) { | ||
1458 | rtl92c_dm_pwdb_monitor(hw); | ||
1459 | rtl92c_dm_dig(hw); | ||
1460 | rtl92c_dm_false_alarm_counter_statistics(hw); | ||
1461 | rtl92c_dm_dynamic_bb_powersaving(hw); | ||
1462 | rtl92c_dm_dynamic_txpower(hw); | ||
1463 | rtl92c_dm_check_txpower_tracking(hw); | ||
1464 | rtl92c_dm_refresh_rate_adaptive_mask(hw); | ||
1465 | rtl92c_dm_bt_coexist(hw); | ||
1466 | rtl92c_dm_check_edca_turbo(hw); | ||
1467 | } | ||
1468 | } | ||
1469 | EXPORT_SYMBOL(rtl92c_dm_watchdog); | ||
1470 | |||
1471 | u8 rtl92c_bt_rssi_state_change(struct ieee80211_hw *hw) | ||
1472 | { | ||
1473 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1474 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1475 | long undecorated_smoothed_pwdb; | ||
1476 | u8 curr_bt_rssi_state = 0x00; | ||
1477 | |||
1478 | if (rtlpriv->mac80211.link_state == MAC80211_LINKED) { | ||
1479 | undecorated_smoothed_pwdb = | ||
1480 | GET_UNDECORATED_AVERAGE_RSSI(rtlpriv); | ||
1481 | } else { | ||
1482 | if (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0) | ||
1483 | undecorated_smoothed_pwdb = 100; | ||
1484 | else | ||
1485 | undecorated_smoothed_pwdb = | ||
1486 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
1487 | } | ||
1488 | |||
1489 | /* Check RSSI to determine HighPower/NormalPower state for | ||
1490 | * BT coexistence. */ | ||
1491 | if (undecorated_smoothed_pwdb >= 67) | ||
1492 | curr_bt_rssi_state &= (~BT_RSSI_STATE_NORMAL_POWER); | ||
1493 | else if (undecorated_smoothed_pwdb < 62) | ||
1494 | curr_bt_rssi_state |= BT_RSSI_STATE_NORMAL_POWER; | ||
1495 | |||
1496 | /* Check RSSI to determine AMPDU setting for BT coexistence. */ | ||
1497 | if (undecorated_smoothed_pwdb >= 40) | ||
1498 | curr_bt_rssi_state &= (~BT_RSSI_STATE_AMDPU_OFF); | ||
1499 | else if (undecorated_smoothed_pwdb <= 32) | ||
1500 | curr_bt_rssi_state |= BT_RSSI_STATE_AMDPU_OFF; | ||
1501 | |||
1502 | /* Marked RSSI state. It will be used to determine BT coexistence | ||
1503 | * setting later. */ | ||
1504 | if (undecorated_smoothed_pwdb < 35) | ||
1505 | curr_bt_rssi_state |= BT_RSSI_STATE_SPECIAL_LOW; | ||
1506 | else | ||
1507 | curr_bt_rssi_state &= (~BT_RSSI_STATE_SPECIAL_LOW); | ||
1508 | |||
1509 | /* Set Tx Power according to BT status. */ | ||
1510 | if (undecorated_smoothed_pwdb >= 30) | ||
1511 | curr_bt_rssi_state |= BT_RSSI_STATE_TXPOWER_LOW; | ||
1512 | else if (undecorated_smoothed_pwdb < 25) | ||
1513 | curr_bt_rssi_state &= (~BT_RSSI_STATE_TXPOWER_LOW); | ||
1514 | |||
1515 | /* Check BT state related to BT_Idle in B/G mode. */ | ||
1516 | if (undecorated_smoothed_pwdb < 15) | ||
1517 | curr_bt_rssi_state |= BT_RSSI_STATE_BG_EDCA_LOW; | ||
1518 | else | ||
1519 | curr_bt_rssi_state &= (~BT_RSSI_STATE_BG_EDCA_LOW); | ||
1520 | |||
1521 | if (curr_bt_rssi_state != rtlpcipriv->bt_coexist.bt_rssi_state) { | ||
1522 | rtlpcipriv->bt_coexist.bt_rssi_state = curr_bt_rssi_state; | ||
1523 | return true; | ||
1524 | } else { | ||
1525 | return false; | ||
1526 | } | ||
1527 | } | ||
1528 | EXPORT_SYMBOL(rtl92c_bt_rssi_state_change); | ||
1529 | |||
1530 | static bool rtl92c_bt_state_change(struct ieee80211_hw *hw) | ||
1531 | { | ||
1532 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1533 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1534 | |||
1535 | u32 polling, ratio_tx, ratio_pri; | ||
1536 | u32 bt_tx, bt_pri; | ||
1537 | u8 bt_state; | ||
1538 | u8 cur_service_type; | ||
1539 | |||
1540 | if (rtlpriv->mac80211.link_state < MAC80211_LINKED) | ||
1541 | return false; | ||
1542 | |||
1543 | bt_state = rtl_read_byte(rtlpriv, 0x4fd); | ||
1544 | bt_tx = rtl_read_dword(rtlpriv, 0x488); | ||
1545 | bt_tx = bt_tx & 0x00ffffff; | ||
1546 | bt_pri = rtl_read_dword(rtlpriv, 0x48c); | ||
1547 | bt_pri = bt_pri & 0x00ffffff; | ||
1548 | polling = rtl_read_dword(rtlpriv, 0x490); | ||
1549 | |||
1550 | if (bt_tx == 0xffffffff && bt_pri == 0xffffffff && | ||
1551 | polling == 0xffffffff && bt_state == 0xff) | ||
1552 | return false; | ||
1553 | |||
1554 | bt_state &= BIT_OFFSET_LEN_MASK_32(0, 1); | ||
1555 | if (bt_state != rtlpcipriv->bt_coexist.bt_cur_state) { | ||
1556 | rtlpcipriv->bt_coexist.bt_cur_state = bt_state; | ||
1557 | |||
1558 | if (rtlpcipriv->bt_coexist.reg_bt_sco == 3) { | ||
1559 | rtlpcipriv->bt_coexist.bt_service = BT_IDLE; | ||
1560 | |||
1561 | bt_state = bt_state | | ||
1562 | ((rtlpcipriv->bt_coexist.bt_ant_isolation == 1) ? | ||
1563 | 0 : BIT_OFFSET_LEN_MASK_32(1, 1)) | | ||
1564 | BIT_OFFSET_LEN_MASK_32(2, 1); | ||
1565 | rtl_write_byte(rtlpriv, 0x4fd, bt_state); | ||
1566 | } | ||
1567 | return true; | ||
1568 | } | ||
1569 | |||
1570 | ratio_tx = bt_tx * 1000 / polling; | ||
1571 | ratio_pri = bt_pri * 1000 / polling; | ||
1572 | rtlpcipriv->bt_coexist.ratio_tx = ratio_tx; | ||
1573 | rtlpcipriv->bt_coexist.ratio_pri = ratio_pri; | ||
1574 | |||
1575 | if (bt_state && rtlpcipriv->bt_coexist.reg_bt_sco == 3) { | ||
1576 | |||
1577 | if ((ratio_tx < 30) && (ratio_pri < 30)) | ||
1578 | cur_service_type = BT_IDLE; | ||
1579 | else if ((ratio_pri > 110) && (ratio_pri < 250)) | ||
1580 | cur_service_type = BT_SCO; | ||
1581 | else if ((ratio_tx >= 200) && (ratio_pri >= 200)) | ||
1582 | cur_service_type = BT_BUSY; | ||
1583 | else if ((ratio_tx >= 350) && (ratio_tx < 500)) | ||
1584 | cur_service_type = BT_OTHERBUSY; | ||
1585 | else if (ratio_tx >= 500) | ||
1586 | cur_service_type = BT_PAN; | ||
1587 | else | ||
1588 | cur_service_type = BT_OTHER_ACTION; | ||
1589 | |||
1590 | if (cur_service_type != rtlpcipriv->bt_coexist.bt_service) { | ||
1591 | rtlpcipriv->bt_coexist.bt_service = cur_service_type; | ||
1592 | bt_state = bt_state | | ||
1593 | ((rtlpcipriv->bt_coexist.bt_ant_isolation == 1) ? | ||
1594 | 0 : BIT_OFFSET_LEN_MASK_32(1, 1)) | | ||
1595 | ((rtlpcipriv->bt_coexist.bt_service != BT_IDLE) ? | ||
1596 | 0 : BIT_OFFSET_LEN_MASK_32(2, 1)); | ||
1597 | |||
1598 | /* Add interrupt migration when bt is not ini | ||
1599 | * idle state (no traffic). */ | ||
1600 | if (rtlpcipriv->bt_coexist.bt_service != BT_IDLE) { | ||
1601 | rtl_write_word(rtlpriv, 0x504, 0x0ccc); | ||
1602 | rtl_write_byte(rtlpriv, 0x506, 0x54); | ||
1603 | rtl_write_byte(rtlpriv, 0x507, 0x54); | ||
1604 | } else { | ||
1605 | rtl_write_byte(rtlpriv, 0x506, 0x00); | ||
1606 | rtl_write_byte(rtlpriv, 0x507, 0x00); | ||
1607 | } | ||
1608 | |||
1609 | rtl_write_byte(rtlpriv, 0x4fd, bt_state); | ||
1610 | return true; | ||
1611 | } | ||
1612 | } | ||
1613 | |||
1614 | return false; | ||
1615 | |||
1616 | } | ||
1617 | |||
1618 | static bool rtl92c_bt_wifi_connect_change(struct ieee80211_hw *hw) | ||
1619 | { | ||
1620 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1621 | static bool media_connect; | ||
1622 | |||
1623 | if (rtlpriv->mac80211.link_state < MAC80211_LINKED) { | ||
1624 | media_connect = false; | ||
1625 | } else { | ||
1626 | if (!media_connect) { | ||
1627 | media_connect = true; | ||
1628 | return true; | ||
1629 | } | ||
1630 | media_connect = true; | ||
1631 | } | ||
1632 | |||
1633 | return false; | ||
1634 | } | ||
1635 | |||
1636 | static void rtl92c_bt_set_normal(struct ieee80211_hw *hw) | ||
1637 | { | ||
1638 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1639 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1640 | |||
1641 | |||
1642 | if (rtlpcipriv->bt_coexist.bt_service == BT_OTHERBUSY) { | ||
1643 | rtlpcipriv->bt_coexist.bt_edca_ul = 0x5ea72b; | ||
1644 | rtlpcipriv->bt_coexist.bt_edca_dl = 0x5ea72b; | ||
1645 | } else if (rtlpcipriv->bt_coexist.bt_service == BT_BUSY) { | ||
1646 | rtlpcipriv->bt_coexist.bt_edca_ul = 0x5eb82f; | ||
1647 | rtlpcipriv->bt_coexist.bt_edca_dl = 0x5eb82f; | ||
1648 | } else if (rtlpcipriv->bt_coexist.bt_service == BT_SCO) { | ||
1649 | if (rtlpcipriv->bt_coexist.ratio_tx > 160) { | ||
1650 | rtlpcipriv->bt_coexist.bt_edca_ul = 0x5ea72f; | ||
1651 | rtlpcipriv->bt_coexist.bt_edca_dl = 0x5ea72f; | ||
1652 | } else { | ||
1653 | rtlpcipriv->bt_coexist.bt_edca_ul = 0x5ea32b; | ||
1654 | rtlpcipriv->bt_coexist.bt_edca_dl = 0x5ea42b; | ||
1655 | } | ||
1656 | } else { | ||
1657 | rtlpcipriv->bt_coexist.bt_edca_ul = 0; | ||
1658 | rtlpcipriv->bt_coexist.bt_edca_dl = 0; | ||
1659 | } | ||
1660 | |||
1661 | if ((rtlpcipriv->bt_coexist.bt_service != BT_IDLE) && | ||
1662 | (rtlpriv->mac80211.mode == WIRELESS_MODE_G || | ||
1663 | (rtlpriv->mac80211.mode == (WIRELESS_MODE_G | WIRELESS_MODE_B))) && | ||
1664 | (rtlpcipriv->bt_coexist.bt_rssi_state & | ||
1665 | BT_RSSI_STATE_BG_EDCA_LOW)) { | ||
1666 | rtlpcipriv->bt_coexist.bt_edca_ul = 0x5eb82b; | ||
1667 | rtlpcipriv->bt_coexist.bt_edca_dl = 0x5eb82b; | ||
1668 | } | ||
1669 | } | ||
1670 | |||
1671 | static void rtl92c_bt_ant_isolation(struct ieee80211_hw *hw) | ||
1672 | { | ||
1673 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1674 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1675 | |||
1676 | |||
1677 | /* Only enable HW BT coexist when BT in "Busy" state. */ | ||
1678 | if (rtlpriv->mac80211.vendor == PEER_CISCO && | ||
1679 | rtlpcipriv->bt_coexist.bt_service == BT_OTHER_ACTION) { | ||
1680 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0xa0); | ||
1681 | } else { | ||
1682 | if ((rtlpcipriv->bt_coexist.bt_service == BT_BUSY) && | ||
1683 | (rtlpcipriv->bt_coexist.bt_rssi_state & | ||
1684 | BT_RSSI_STATE_NORMAL_POWER)) { | ||
1685 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0xa0); | ||
1686 | } else if ((rtlpcipriv->bt_coexist.bt_service == | ||
1687 | BT_OTHER_ACTION) && (rtlpriv->mac80211.mode < | ||
1688 | WIRELESS_MODE_N_24G) && | ||
1689 | (rtlpcipriv->bt_coexist.bt_rssi_state & | ||
1690 | BT_RSSI_STATE_SPECIAL_LOW)) { | ||
1691 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0xa0); | ||
1692 | } else if (rtlpcipriv->bt_coexist.bt_service == BT_PAN) { | ||
1693 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0x00); | ||
1694 | } else { | ||
1695 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0x00); | ||
1696 | } | ||
1697 | } | ||
1698 | |||
1699 | if (rtlpcipriv->bt_coexist.bt_service == BT_PAN) | ||
1700 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, 0x10100); | ||
1701 | else | ||
1702 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, 0x0); | ||
1703 | |||
1704 | if (rtlpcipriv->bt_coexist.bt_rssi_state & | ||
1705 | BT_RSSI_STATE_NORMAL_POWER) { | ||
1706 | rtl92c_bt_set_normal(hw); | ||
1707 | } else { | ||
1708 | rtlpcipriv->bt_coexist.bt_edca_ul = 0; | ||
1709 | rtlpcipriv->bt_coexist.bt_edca_dl = 0; | ||
1710 | } | ||
1711 | |||
1712 | if (rtlpcipriv->bt_coexist.bt_service != BT_IDLE) { | ||
1713 | rtlpriv->cfg->ops->set_rfreg(hw, | ||
1714 | RF90_PATH_A, | ||
1715 | 0x1e, | ||
1716 | 0xf0, 0xf); | ||
1717 | } else { | ||
1718 | rtlpriv->cfg->ops->set_rfreg(hw, | ||
1719 | RF90_PATH_A, 0x1e, 0xf0, | ||
1720 | rtlpcipriv->bt_coexist.bt_rfreg_origin_1e); | ||
1721 | } | ||
1722 | |||
1723 | if (!rtlpriv->dm.dynamic_txpower_enable) { | ||
1724 | if (rtlpcipriv->bt_coexist.bt_service != BT_IDLE) { | ||
1725 | if (rtlpcipriv->bt_coexist.bt_rssi_state & | ||
1726 | BT_RSSI_STATE_TXPOWER_LOW) { | ||
1727 | rtlpriv->dm.dynamic_txhighpower_lvl = | ||
1728 | TXHIGHPWRLEVEL_BT2; | ||
1729 | } else { | ||
1730 | rtlpriv->dm.dynamic_txhighpower_lvl = | ||
1731 | TXHIGHPWRLEVEL_BT1; | ||
1732 | } | ||
1733 | } else { | ||
1734 | rtlpriv->dm.dynamic_txhighpower_lvl = | ||
1735 | TXHIGHPWRLEVEL_NORMAL; | ||
1736 | } | ||
1737 | rtl92c_phy_set_txpower_level(hw, | ||
1738 | rtlpriv->phy.current_channel); | ||
1739 | } | ||
1740 | } | ||
1741 | |||
1742 | static void rtl92c_check_bt_change(struct ieee80211_hw *hw) | ||
1743 | { | ||
1744 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1745 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1746 | |||
1747 | if (rtlpcipriv->bt_coexist.bt_cur_state) { | ||
1748 | if (rtlpcipriv->bt_coexist.bt_ant_isolation) | ||
1749 | rtl92c_bt_ant_isolation(hw); | ||
1750 | } else { | ||
1751 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0x00); | ||
1752 | rtlpriv->cfg->ops->set_rfreg(hw, RF90_PATH_A, 0x1e, 0xf0, | ||
1753 | rtlpcipriv->bt_coexist.bt_rfreg_origin_1e); | ||
1754 | |||
1755 | rtlpcipriv->bt_coexist.bt_edca_ul = 0; | ||
1756 | rtlpcipriv->bt_coexist.bt_edca_dl = 0; | ||
1757 | } | ||
1758 | } | ||
1759 | |||
1760 | void rtl92c_dm_bt_coexist(struct ieee80211_hw *hw) | ||
1761 | { | ||
1762 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1763 | |||
1764 | bool wifi_connect_change; | ||
1765 | bool bt_state_change; | ||
1766 | bool rssi_state_change; | ||
1767 | |||
1768 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
1769 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4)) { | ||
1770 | |||
1771 | wifi_connect_change = rtl92c_bt_wifi_connect_change(hw); | ||
1772 | bt_state_change = rtl92c_bt_state_change(hw); | ||
1773 | rssi_state_change = rtl92c_bt_rssi_state_change(hw); | ||
1774 | |||
1775 | if (wifi_connect_change || bt_state_change || rssi_state_change) | ||
1776 | rtl92c_check_bt_change(hw); | ||
1777 | } | ||
1778 | } | ||
1779 | EXPORT_SYMBOL(rtl92c_dm_bt_coexist); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.h b/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.h new file mode 100644 index 000000000000..b9736d3e9a39 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/dm_common.h | |||
@@ -0,0 +1,206 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92COMMON_DM_H__ | ||
31 | #define __RTL92COMMON_DM_H__ | ||
32 | |||
33 | #include "../wifi.h" | ||
34 | #include "../rtl8192ce/def.h" | ||
35 | #include "../rtl8192ce/reg.h" | ||
36 | #include "fw_common.h" | ||
37 | |||
38 | #define HAL_DM_DIG_DISABLE BIT(0) | ||
39 | #define HAL_DM_HIPWR_DISABLE BIT(1) | ||
40 | |||
41 | #define OFDM_TABLE_LENGTH 37 | ||
42 | #define CCK_TABLE_LENGTH 33 | ||
43 | |||
44 | #define OFDM_TABLE_SIZE 37 | ||
45 | #define CCK_TABLE_SIZE 33 | ||
46 | |||
47 | #define BW_AUTO_SWITCH_HIGH_LOW 25 | ||
48 | #define BW_AUTO_SWITCH_LOW_HIGH 30 | ||
49 | |||
50 | #define DM_DIG_THRESH_HIGH 40 | ||
51 | #define DM_DIG_THRESH_LOW 35 | ||
52 | |||
53 | #define DM_FALSEALARM_THRESH_LOW 400 | ||
54 | #define DM_FALSEALARM_THRESH_HIGH 1000 | ||
55 | |||
56 | #define DM_DIG_MAX 0x3e | ||
57 | #define DM_DIG_MIN 0x1e | ||
58 | |||
59 | #define DM_DIG_FA_UPPER 0x32 | ||
60 | #define DM_DIG_FA_LOWER 0x20 | ||
61 | #define DM_DIG_FA_TH0 0x20 | ||
62 | #define DM_DIG_FA_TH1 0x100 | ||
63 | #define DM_DIG_FA_TH2 0x200 | ||
64 | |||
65 | #define DM_DIG_BACKOFF_MAX 12 | ||
66 | #define DM_DIG_BACKOFF_MIN -4 | ||
67 | #define DM_DIG_BACKOFF_DEFAULT 10 | ||
68 | |||
69 | #define RXPATHSELECTION_SS_TH_lOW 30 | ||
70 | #define RXPATHSELECTION_DIFF_TH 18 | ||
71 | |||
72 | #define DM_RATR_STA_INIT 0 | ||
73 | #define DM_RATR_STA_HIGH 1 | ||
74 | #define DM_RATR_STA_MIDDLE 2 | ||
75 | #define DM_RATR_STA_LOW 3 | ||
76 | |||
77 | #define CTS2SELF_THVAL 30 | ||
78 | #define REGC38_TH 20 | ||
79 | |||
80 | #define WAIOTTHVal 25 | ||
81 | |||
82 | #define TXHIGHPWRLEVEL_NORMAL 0 | ||
83 | #define TXHIGHPWRLEVEL_LEVEL1 1 | ||
84 | #define TXHIGHPWRLEVEL_LEVEL2 2 | ||
85 | #define TXHIGHPWRLEVEL_BT1 3 | ||
86 | #define TXHIGHPWRLEVEL_BT2 4 | ||
87 | |||
88 | #define DM_TYPE_BYFW 0 | ||
89 | #define DM_TYPE_BYDRIVER 1 | ||
90 | |||
91 | #define TX_POWER_NEAR_FIELD_THRESH_LVL2 74 | ||
92 | #define TX_POWER_NEAR_FIELD_THRESH_LVL1 67 | ||
93 | |||
94 | struct ps_t { | ||
95 | u8 pre_ccastate; | ||
96 | u8 cur_ccasate; | ||
97 | u8 pre_rfstate; | ||
98 | u8 cur_rfstate; | ||
99 | long rssi_val_min; | ||
100 | }; | ||
101 | |||
102 | struct dig_t { | ||
103 | u8 dig_enable_flag; | ||
104 | u8 dig_ext_port_stage; | ||
105 | u32 rssi_lowthresh; | ||
106 | u32 rssi_highthresh; | ||
107 | u32 fa_lowthresh; | ||
108 | u32 fa_highthresh; | ||
109 | u8 cursta_connectctate; | ||
110 | u8 presta_connectstate; | ||
111 | u8 curmultista_connectstate; | ||
112 | u8 pre_igvalue; | ||
113 | u8 cur_igvalue; | ||
114 | char backoff_val; | ||
115 | char backoff_val_range_max; | ||
116 | char backoff_val_range_min; | ||
117 | u8 rx_gain_range_max; | ||
118 | u8 rx_gain_range_min; | ||
119 | u8 rssi_val_min; | ||
120 | u8 pre_cck_pd_state; | ||
121 | u8 cur_cck_pd_state; | ||
122 | u8 pre_cck_fa_state; | ||
123 | u8 cur_cck_fa_state; | ||
124 | u8 pre_ccastate; | ||
125 | u8 cur_ccasate; | ||
126 | }; | ||
127 | |||
128 | struct swat_t { | ||
129 | u8 failure_cnt; | ||
130 | u8 try_flag; | ||
131 | u8 stop_trying; | ||
132 | long pre_rssi; | ||
133 | long trying_threshold; | ||
134 | u8 cur_antenna; | ||
135 | u8 pre_antenna; | ||
136 | }; | ||
137 | |||
138 | enum tag_dynamic_init_gain_operation_type_definition { | ||
139 | DIG_TYPE_THRESH_HIGH = 0, | ||
140 | DIG_TYPE_THRESH_LOW = 1, | ||
141 | DIG_TYPE_BACKOFF = 2, | ||
142 | DIG_TYPE_RX_GAIN_MIN = 3, | ||
143 | DIG_TYPE_RX_GAIN_MAX = 4, | ||
144 | DIG_TYPE_ENABLE = 5, | ||
145 | DIG_TYPE_DISABLE = 6, | ||
146 | DIG_OP_TYPE_MAX | ||
147 | }; | ||
148 | |||
149 | enum tag_cck_packet_detection_threshold_type_definition { | ||
150 | CCK_PD_STAGE_LowRssi = 0, | ||
151 | CCK_PD_STAGE_HighRssi = 1, | ||
152 | CCK_FA_STAGE_Low = 2, | ||
153 | CCK_FA_STAGE_High = 3, | ||
154 | CCK_PD_STAGE_MAX = 4, | ||
155 | }; | ||
156 | |||
157 | enum dm_1r_cca_e { | ||
158 | CCA_1R = 0, | ||
159 | CCA_2R = 1, | ||
160 | CCA_MAX = 2, | ||
161 | }; | ||
162 | |||
163 | enum dm_rf_e { | ||
164 | RF_SAVE = 0, | ||
165 | RF_NORMAL = 1, | ||
166 | RF_MAX = 2, | ||
167 | }; | ||
168 | |||
169 | enum dm_sw_ant_switch_e { | ||
170 | ANS_ANTENNA_B = 1, | ||
171 | ANS_ANTENNA_A = 2, | ||
172 | ANS_ANTENNA_MAX = 3, | ||
173 | }; | ||
174 | |||
175 | enum dm_dig_ext_port_alg_e { | ||
176 | DIG_EXT_PORT_STAGE_0 = 0, | ||
177 | DIG_EXT_PORT_STAGE_1 = 1, | ||
178 | DIG_EXT_PORT_STAGE_2 = 2, | ||
179 | DIG_EXT_PORT_STAGE_3 = 3, | ||
180 | DIG_EXT_PORT_STAGE_MAX = 4, | ||
181 | }; | ||
182 | |||
183 | enum dm_dig_connect_e { | ||
184 | DIG_STA_DISCONNECT = 0, | ||
185 | DIG_STA_CONNECT = 1, | ||
186 | DIG_STA_BEFORE_CONNECT = 2, | ||
187 | DIG_MULTISTA_DISCONNECT = 3, | ||
188 | DIG_MULTISTA_CONNECT = 4, | ||
189 | DIG_CONNECT_MAX | ||
190 | }; | ||
191 | |||
192 | extern struct dig_t dm_digtable; | ||
193 | void rtl92c_dm_init(struct ieee80211_hw *hw); | ||
194 | void rtl92c_dm_watchdog(struct ieee80211_hw *hw); | ||
195 | void rtl92c_dm_write_dig(struct ieee80211_hw *hw); | ||
196 | void rtl92c_dm_init_edca_turbo(struct ieee80211_hw *hw); | ||
197 | void rtl92c_dm_check_txpower_tracking(struct ieee80211_hw *hw); | ||
198 | void rtl92c_dm_init_rate_adaptive_mask(struct ieee80211_hw *hw); | ||
199 | void rtl92c_dm_rf_saving(struct ieee80211_hw *hw, u8 bforce_in_normal); | ||
200 | void rtl92c_phy_ap_calibrate(struct ieee80211_hw *hw, char delta); | ||
201 | void rtl92c_phy_lc_calibrate(struct ieee80211_hw *hw); | ||
202 | void rtl92c_phy_iq_calibrate(struct ieee80211_hw *hw, bool recovery); | ||
203 | void rtl92c_dm_dynamic_txpower(struct ieee80211_hw *hw); | ||
204 | void rtl92c_dm_bt_coexist(struct ieee80211_hw *hw); | ||
205 | |||
206 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.c b/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.c new file mode 100644 index 000000000000..50303e1adff1 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.c | |||
@@ -0,0 +1,785 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include <linux/firmware.h> | ||
31 | #include "../wifi.h" | ||
32 | #include "../pci.h" | ||
33 | #include "../base.h" | ||
34 | #include "../rtl8192ce/reg.h" | ||
35 | #include "../rtl8192ce/def.h" | ||
36 | #include "fw_common.h" | ||
37 | |||
38 | static void _rtl92c_enable_fw_download(struct ieee80211_hw *hw, bool enable) | ||
39 | { | ||
40 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
41 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
42 | |||
43 | if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192CU) { | ||
44 | u32 value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); | ||
45 | if (enable) | ||
46 | value32 |= MCUFWDL_EN; | ||
47 | else | ||
48 | value32 &= ~MCUFWDL_EN; | ||
49 | rtl_write_dword(rtlpriv, REG_MCUFWDL, value32); | ||
50 | } else if (rtlhal->hw_type == HARDWARE_TYPE_RTL8192CE) { | ||
51 | u8 tmp; | ||
52 | if (enable) { | ||
53 | |||
54 | tmp = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
55 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, | ||
56 | tmp | 0x04); | ||
57 | |||
58 | tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL); | ||
59 | rtl_write_byte(rtlpriv, REG_MCUFWDL, tmp | 0x01); | ||
60 | |||
61 | tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL + 2); | ||
62 | rtl_write_byte(rtlpriv, REG_MCUFWDL + 2, tmp & 0xf7); | ||
63 | } else { | ||
64 | |||
65 | tmp = rtl_read_byte(rtlpriv, REG_MCUFWDL); | ||
66 | rtl_write_byte(rtlpriv, REG_MCUFWDL, tmp & 0xfe); | ||
67 | |||
68 | rtl_write_byte(rtlpriv, REG_MCUFWDL + 1, 0x00); | ||
69 | } | ||
70 | } | ||
71 | } | ||
72 | |||
73 | static void _rtl92c_fw_block_write(struct ieee80211_hw *hw, | ||
74 | const u8 *buffer, u32 size) | ||
75 | { | ||
76 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
77 | u32 blockSize = sizeof(u32); | ||
78 | u8 *bufferPtr = (u8 *) buffer; | ||
79 | u32 *pu4BytePtr = (u32 *) buffer; | ||
80 | u32 i, offset, blockCount, remainSize; | ||
81 | |||
82 | blockCount = size / blockSize; | ||
83 | remainSize = size % blockSize; | ||
84 | |||
85 | for (i = 0; i < blockCount; i++) { | ||
86 | offset = i * blockSize; | ||
87 | rtl_write_dword(rtlpriv, (FW_8192C_START_ADDRESS + offset), | ||
88 | *(pu4BytePtr + i)); | ||
89 | } | ||
90 | |||
91 | if (remainSize) { | ||
92 | offset = blockCount * blockSize; | ||
93 | bufferPtr += offset; | ||
94 | for (i = 0; i < remainSize; i++) { | ||
95 | rtl_write_byte(rtlpriv, (FW_8192C_START_ADDRESS + | ||
96 | offset + i), *(bufferPtr + i)); | ||
97 | } | ||
98 | } | ||
99 | } | ||
100 | |||
101 | static void _rtl92c_fw_page_write(struct ieee80211_hw *hw, | ||
102 | u32 page, const u8 *buffer, u32 size) | ||
103 | { | ||
104 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
105 | u8 value8; | ||
106 | u8 u8page = (u8) (page & 0x07); | ||
107 | |||
108 | value8 = (rtl_read_byte(rtlpriv, REG_MCUFWDL + 2) & 0xF8) | u8page; | ||
109 | |||
110 | rtl_write_byte(rtlpriv, (REG_MCUFWDL + 2), value8); | ||
111 | _rtl92c_fw_block_write(hw, buffer, size); | ||
112 | } | ||
113 | |||
114 | static void _rtl92c_fill_dummy(u8 *pfwbuf, u32 *pfwlen) | ||
115 | { | ||
116 | u32 fwlen = *pfwlen; | ||
117 | u8 remain = (u8) (fwlen % 4); | ||
118 | |||
119 | remain = (remain == 0) ? 0 : (4 - remain); | ||
120 | |||
121 | while (remain > 0) { | ||
122 | pfwbuf[fwlen] = 0; | ||
123 | fwlen++; | ||
124 | remain--; | ||
125 | } | ||
126 | |||
127 | *pfwlen = fwlen; | ||
128 | } | ||
129 | |||
130 | static void _rtl92c_write_fw(struct ieee80211_hw *hw, | ||
131 | enum version_8192c version, u8 *buffer, u32 size) | ||
132 | { | ||
133 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
134 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
135 | u8 *bufferPtr = (u8 *) buffer; | ||
136 | |||
137 | RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, ("FW size is %d bytes,\n", size)); | ||
138 | |||
139 | if (IS_CHIP_VER_B(version)) { | ||
140 | u32 pageNums, remainSize; | ||
141 | u32 page, offset; | ||
142 | |||
143 | if (IS_HARDWARE_TYPE_8192CE(rtlhal)) | ||
144 | _rtl92c_fill_dummy(bufferPtr, &size); | ||
145 | |||
146 | pageNums = size / FW_8192C_PAGE_SIZE; | ||
147 | remainSize = size % FW_8192C_PAGE_SIZE; | ||
148 | |||
149 | if (pageNums > 4) { | ||
150 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
151 | ("Page numbers should not greater then 4\n")); | ||
152 | } | ||
153 | |||
154 | for (page = 0; page < pageNums; page++) { | ||
155 | offset = page * FW_8192C_PAGE_SIZE; | ||
156 | _rtl92c_fw_page_write(hw, page, (bufferPtr + offset), | ||
157 | FW_8192C_PAGE_SIZE); | ||
158 | } | ||
159 | |||
160 | if (remainSize) { | ||
161 | offset = pageNums * FW_8192C_PAGE_SIZE; | ||
162 | page = pageNums; | ||
163 | _rtl92c_fw_page_write(hw, page, (bufferPtr + offset), | ||
164 | remainSize); | ||
165 | } | ||
166 | } else { | ||
167 | _rtl92c_fw_block_write(hw, buffer, size); | ||
168 | } | ||
169 | } | ||
170 | |||
171 | static int _rtl92c_fw_free_to_go(struct ieee80211_hw *hw) | ||
172 | { | ||
173 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
174 | u32 counter = 0; | ||
175 | u32 value32; | ||
176 | |||
177 | do { | ||
178 | value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); | ||
179 | } while ((counter++ < FW_8192C_POLLING_TIMEOUT_COUNT) && | ||
180 | (!(value32 & FWDL_ChkSum_rpt))); | ||
181 | |||
182 | if (counter >= FW_8192C_POLLING_TIMEOUT_COUNT) { | ||
183 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
184 | ("chksum report faill ! REG_MCUFWDL:0x%08x .\n", | ||
185 | value32)); | ||
186 | return -EIO; | ||
187 | } | ||
188 | |||
189 | RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, | ||
190 | ("Checksum report OK ! REG_MCUFWDL:0x%08x .\n", value32)); | ||
191 | |||
192 | value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); | ||
193 | value32 |= MCUFWDL_RDY; | ||
194 | value32 &= ~WINTINI_RDY; | ||
195 | rtl_write_dword(rtlpriv, REG_MCUFWDL, value32); | ||
196 | |||
197 | counter = 0; | ||
198 | |||
199 | do { | ||
200 | value32 = rtl_read_dword(rtlpriv, REG_MCUFWDL); | ||
201 | if (value32 & WINTINI_RDY) { | ||
202 | RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, | ||
203 | ("Polling FW ready success!!" | ||
204 | " REG_MCUFWDL:0x%08x .\n", | ||
205 | value32)); | ||
206 | return 0; | ||
207 | } | ||
208 | |||
209 | mdelay(FW_8192C_POLLING_DELAY); | ||
210 | |||
211 | } while (counter++ < FW_8192C_POLLING_TIMEOUT_COUNT); | ||
212 | |||
213 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
214 | ("Polling FW ready fail!! REG_MCUFWDL:0x%08x .\n", value32)); | ||
215 | return -EIO; | ||
216 | } | ||
217 | |||
218 | int rtl92c_download_fw(struct ieee80211_hw *hw) | ||
219 | { | ||
220 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
221 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
222 | struct rtl92c_firmware_header *pfwheader; | ||
223 | u8 *pfwdata; | ||
224 | u32 fwsize; | ||
225 | enum version_8192c version = rtlhal->version; | ||
226 | |||
227 | printk(KERN_INFO "rtl8192c: Loading firmware file %s\n", | ||
228 | rtlpriv->cfg->fw_name); | ||
229 | if (!rtlhal->pfirmware) | ||
230 | return 1; | ||
231 | |||
232 | pfwheader = (struct rtl92c_firmware_header *)rtlhal->pfirmware; | ||
233 | pfwdata = (u8 *) rtlhal->pfirmware; | ||
234 | fwsize = rtlhal->fwsize; | ||
235 | |||
236 | if (IS_FW_HEADER_EXIST(pfwheader)) { | ||
237 | RT_TRACE(rtlpriv, COMP_FW, DBG_DMESG, | ||
238 | ("Firmware Version(%d), Signature(%#x),Size(%d)\n", | ||
239 | pfwheader->version, pfwheader->signature, | ||
240 | (uint)sizeof(struct rtl92c_firmware_header))); | ||
241 | |||
242 | pfwdata = pfwdata + sizeof(struct rtl92c_firmware_header); | ||
243 | fwsize = fwsize - sizeof(struct rtl92c_firmware_header); | ||
244 | } | ||
245 | |||
246 | _rtl92c_enable_fw_download(hw, true); | ||
247 | _rtl92c_write_fw(hw, version, pfwdata, fwsize); | ||
248 | _rtl92c_enable_fw_download(hw, false); | ||
249 | |||
250 | if (_rtl92c_fw_free_to_go(hw)) { | ||
251 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
252 | ("Firmware is not ready to run!\n")); | ||
253 | } else { | ||
254 | RT_TRACE(rtlpriv, COMP_FW, DBG_TRACE, | ||
255 | ("Firmware is ready to run!\n")); | ||
256 | } | ||
257 | |||
258 | return 0; | ||
259 | } | ||
260 | EXPORT_SYMBOL(rtl92c_download_fw); | ||
261 | |||
262 | static bool _rtl92c_check_fw_read_last_h2c(struct ieee80211_hw *hw, u8 boxnum) | ||
263 | { | ||
264 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
265 | u8 val_hmetfr, val_mcutst_1; | ||
266 | bool result = false; | ||
267 | |||
268 | val_hmetfr = rtl_read_byte(rtlpriv, REG_HMETFR); | ||
269 | val_mcutst_1 = rtl_read_byte(rtlpriv, (REG_MCUTST_1 + boxnum)); | ||
270 | |||
271 | if (((val_hmetfr >> boxnum) & BIT(0)) == 0 && val_mcutst_1 == 0) | ||
272 | result = true; | ||
273 | return result; | ||
274 | } | ||
275 | |||
276 | static void _rtl92c_fill_h2c_command(struct ieee80211_hw *hw, | ||
277 | u8 element_id, u32 cmd_len, u8 *p_cmdbuffer) | ||
278 | { | ||
279 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
280 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
281 | u8 boxnum; | ||
282 | u16 box_reg = 0, box_extreg = 0; | ||
283 | u8 u1b_tmp; | ||
284 | bool isfw_read = false; | ||
285 | bool bwrite_sucess = false; | ||
286 | u8 wait_h2c_limmit = 100; | ||
287 | u8 wait_writeh2c_limmit = 100; | ||
288 | u8 boxcontent[4], boxextcontent[2]; | ||
289 | u32 h2c_waitcounter = 0; | ||
290 | unsigned long flag; | ||
291 | u8 idx; | ||
292 | |||
293 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, ("come in\n")); | ||
294 | |||
295 | while (true) { | ||
296 | spin_lock_irqsave(&rtlpriv->locks.h2c_lock, flag); | ||
297 | if (rtlhal->h2c_setinprogress) { | ||
298 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
299 | ("H2C set in progress! Wait to set.." | ||
300 | "element_id(%d).\n", element_id)); | ||
301 | |||
302 | while (rtlhal->h2c_setinprogress) { | ||
303 | spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, | ||
304 | flag); | ||
305 | h2c_waitcounter++; | ||
306 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
307 | ("Wait 100 us (%d times)...\n", | ||
308 | h2c_waitcounter)); | ||
309 | udelay(100); | ||
310 | |||
311 | if (h2c_waitcounter > 1000) | ||
312 | return; | ||
313 | spin_lock_irqsave(&rtlpriv->locks.h2c_lock, | ||
314 | flag); | ||
315 | } | ||
316 | spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); | ||
317 | } else { | ||
318 | rtlhal->h2c_setinprogress = true; | ||
319 | spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); | ||
320 | break; | ||
321 | } | ||
322 | } | ||
323 | |||
324 | while (!bwrite_sucess) { | ||
325 | wait_writeh2c_limmit--; | ||
326 | if (wait_writeh2c_limmit == 0) { | ||
327 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
328 | ("Write H2C fail because no trigger " | ||
329 | "for FW INT!\n")); | ||
330 | break; | ||
331 | } | ||
332 | |||
333 | boxnum = rtlhal->last_hmeboxnum; | ||
334 | switch (boxnum) { | ||
335 | case 0: | ||
336 | box_reg = REG_HMEBOX_0; | ||
337 | box_extreg = REG_HMEBOX_EXT_0; | ||
338 | break; | ||
339 | case 1: | ||
340 | box_reg = REG_HMEBOX_1; | ||
341 | box_extreg = REG_HMEBOX_EXT_1; | ||
342 | break; | ||
343 | case 2: | ||
344 | box_reg = REG_HMEBOX_2; | ||
345 | box_extreg = REG_HMEBOX_EXT_2; | ||
346 | break; | ||
347 | case 3: | ||
348 | box_reg = REG_HMEBOX_3; | ||
349 | box_extreg = REG_HMEBOX_EXT_3; | ||
350 | break; | ||
351 | default: | ||
352 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
353 | ("switch case not process\n")); | ||
354 | break; | ||
355 | } | ||
356 | |||
357 | isfw_read = _rtl92c_check_fw_read_last_h2c(hw, boxnum); | ||
358 | while (!isfw_read) { | ||
359 | |||
360 | wait_h2c_limmit--; | ||
361 | if (wait_h2c_limmit == 0) { | ||
362 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
363 | ("Wating too long for FW read " | ||
364 | "clear HMEBox(%d)!\n", boxnum)); | ||
365 | break; | ||
366 | } | ||
367 | |||
368 | udelay(10); | ||
369 | |||
370 | isfw_read = _rtl92c_check_fw_read_last_h2c(hw, boxnum); | ||
371 | u1b_tmp = rtl_read_byte(rtlpriv, 0x1BF); | ||
372 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
373 | ("Wating for FW read clear HMEBox(%d)!!! " | ||
374 | "0x1BF = %2x\n", boxnum, u1b_tmp)); | ||
375 | } | ||
376 | |||
377 | if (!isfw_read) { | ||
378 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
379 | ("Write H2C register BOX[%d] fail!!!!! " | ||
380 | "Fw do not read.\n", boxnum)); | ||
381 | break; | ||
382 | } | ||
383 | |||
384 | memset(boxcontent, 0, sizeof(boxcontent)); | ||
385 | memset(boxextcontent, 0, sizeof(boxextcontent)); | ||
386 | boxcontent[0] = element_id; | ||
387 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
388 | ("Write element_id box_reg(%4x) = %2x\n", | ||
389 | box_reg, element_id)); | ||
390 | |||
391 | switch (cmd_len) { | ||
392 | case 1: | ||
393 | boxcontent[0] &= ~(BIT(7)); | ||
394 | memcpy((u8 *) (boxcontent) + 1, | ||
395 | p_cmdbuffer, 1); | ||
396 | |||
397 | for (idx = 0; idx < 4; idx++) { | ||
398 | rtl_write_byte(rtlpriv, box_reg + idx, | ||
399 | boxcontent[idx]); | ||
400 | } | ||
401 | break; | ||
402 | case 2: | ||
403 | boxcontent[0] &= ~(BIT(7)); | ||
404 | memcpy((u8 *) (boxcontent) + 1, | ||
405 | p_cmdbuffer, 2); | ||
406 | |||
407 | for (idx = 0; idx < 4; idx++) { | ||
408 | rtl_write_byte(rtlpriv, box_reg + idx, | ||
409 | boxcontent[idx]); | ||
410 | } | ||
411 | break; | ||
412 | case 3: | ||
413 | boxcontent[0] &= ~(BIT(7)); | ||
414 | memcpy((u8 *) (boxcontent) + 1, | ||
415 | p_cmdbuffer, 3); | ||
416 | |||
417 | for (idx = 0; idx < 4; idx++) { | ||
418 | rtl_write_byte(rtlpriv, box_reg + idx, | ||
419 | boxcontent[idx]); | ||
420 | } | ||
421 | break; | ||
422 | case 4: | ||
423 | boxcontent[0] |= (BIT(7)); | ||
424 | memcpy((u8 *) (boxextcontent), | ||
425 | p_cmdbuffer, 2); | ||
426 | memcpy((u8 *) (boxcontent) + 1, | ||
427 | p_cmdbuffer + 2, 2); | ||
428 | |||
429 | for (idx = 0; idx < 2; idx++) { | ||
430 | rtl_write_byte(rtlpriv, box_extreg + idx, | ||
431 | boxextcontent[idx]); | ||
432 | } | ||
433 | |||
434 | for (idx = 0; idx < 4; idx++) { | ||
435 | rtl_write_byte(rtlpriv, box_reg + idx, | ||
436 | boxcontent[idx]); | ||
437 | } | ||
438 | break; | ||
439 | case 5: | ||
440 | boxcontent[0] |= (BIT(7)); | ||
441 | memcpy((u8 *) (boxextcontent), | ||
442 | p_cmdbuffer, 2); | ||
443 | memcpy((u8 *) (boxcontent) + 1, | ||
444 | p_cmdbuffer + 2, 3); | ||
445 | |||
446 | for (idx = 0; idx < 2; idx++) { | ||
447 | rtl_write_byte(rtlpriv, box_extreg + idx, | ||
448 | boxextcontent[idx]); | ||
449 | } | ||
450 | |||
451 | for (idx = 0; idx < 4; idx++) { | ||
452 | rtl_write_byte(rtlpriv, box_reg + idx, | ||
453 | boxcontent[idx]); | ||
454 | } | ||
455 | break; | ||
456 | default: | ||
457 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
458 | ("switch case not process\n")); | ||
459 | break; | ||
460 | } | ||
461 | |||
462 | bwrite_sucess = true; | ||
463 | |||
464 | rtlhal->last_hmeboxnum = boxnum + 1; | ||
465 | if (rtlhal->last_hmeboxnum == 4) | ||
466 | rtlhal->last_hmeboxnum = 0; | ||
467 | |||
468 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
469 | ("pHalData->last_hmeboxnum = %d\n", | ||
470 | rtlhal->last_hmeboxnum)); | ||
471 | } | ||
472 | |||
473 | spin_lock_irqsave(&rtlpriv->locks.h2c_lock, flag); | ||
474 | rtlhal->h2c_setinprogress = false; | ||
475 | spin_unlock_irqrestore(&rtlpriv->locks.h2c_lock, flag); | ||
476 | |||
477 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, ("go out\n")); | ||
478 | } | ||
479 | |||
480 | void rtl92c_fill_h2c_cmd(struct ieee80211_hw *hw, | ||
481 | u8 element_id, u32 cmd_len, u8 *p_cmdbuffer) | ||
482 | { | ||
483 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
484 | u32 tmp_cmdbuf[2]; | ||
485 | |||
486 | if (rtlhal->fw_ready == false) { | ||
487 | RT_ASSERT(false, ("return H2C cmd because of Fw " | ||
488 | "download fail!!!\n")); | ||
489 | return; | ||
490 | } | ||
491 | |||
492 | memset(tmp_cmdbuf, 0, 8); | ||
493 | memcpy(tmp_cmdbuf, p_cmdbuffer, cmd_len); | ||
494 | _rtl92c_fill_h2c_command(hw, element_id, cmd_len, (u8 *)&tmp_cmdbuf); | ||
495 | |||
496 | return; | ||
497 | } | ||
498 | EXPORT_SYMBOL(rtl92c_fill_h2c_cmd); | ||
499 | |||
500 | void rtl92c_firmware_selfreset(struct ieee80211_hw *hw) | ||
501 | { | ||
502 | u8 u1b_tmp; | ||
503 | u8 delay = 100; | ||
504 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
505 | |||
506 | rtl_write_byte(rtlpriv, REG_HMETFR + 3, 0x20); | ||
507 | u1b_tmp = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
508 | |||
509 | while (u1b_tmp & BIT(2)) { | ||
510 | delay--; | ||
511 | if (delay == 0) { | ||
512 | RT_ASSERT(false, ("8051 reset fail.\n")); | ||
513 | break; | ||
514 | } | ||
515 | udelay(50); | ||
516 | u1b_tmp = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
517 | } | ||
518 | } | ||
519 | EXPORT_SYMBOL(rtl92c_firmware_selfreset); | ||
520 | |||
521 | void rtl92c_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 mode) | ||
522 | { | ||
523 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
524 | u8 u1_h2c_set_pwrmode[3] = {0}; | ||
525 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
526 | |||
527 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("FW LPS mode = %d\n", mode)); | ||
528 | |||
529 | SET_H2CCMD_PWRMODE_PARM_MODE(u1_h2c_set_pwrmode, mode); | ||
530 | SET_H2CCMD_PWRMODE_PARM_SMART_PS(u1_h2c_set_pwrmode, 1); | ||
531 | SET_H2CCMD_PWRMODE_PARM_BCN_PASS_TIME(u1_h2c_set_pwrmode, | ||
532 | ppsc->reg_max_lps_awakeintvl); | ||
533 | |||
534 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, | ||
535 | "rtl92c_set_fw_rsvdpagepkt(): u1_h2c_set_pwrmode\n", | ||
536 | u1_h2c_set_pwrmode, 3); | ||
537 | rtl92c_fill_h2c_cmd(hw, H2C_SETPWRMODE, 3, u1_h2c_set_pwrmode); | ||
538 | |||
539 | } | ||
540 | EXPORT_SYMBOL(rtl92c_set_fw_pwrmode_cmd); | ||
541 | |||
542 | static bool _rtl92c_cmd_send_packet(struct ieee80211_hw *hw, | ||
543 | struct sk_buff *skb) | ||
544 | { | ||
545 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
546 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
547 | struct rtl8192_tx_ring *ring; | ||
548 | struct rtl_tx_desc *pdesc; | ||
549 | u8 own; | ||
550 | unsigned long flags; | ||
551 | struct sk_buff *pskb = NULL; | ||
552 | |||
553 | ring = &rtlpci->tx_ring[BEACON_QUEUE]; | ||
554 | |||
555 | pskb = __skb_dequeue(&ring->queue); | ||
556 | if (pskb) | ||
557 | kfree_skb(pskb); | ||
558 | |||
559 | spin_lock_irqsave(&rtlpriv->locks.irq_th_lock, flags); | ||
560 | |||
561 | pdesc = &ring->desc[0]; | ||
562 | own = (u8) rtlpriv->cfg->ops->get_desc((u8 *) pdesc, true, HW_DESC_OWN); | ||
563 | |||
564 | rtlpriv->cfg->ops->fill_tx_cmddesc(hw, (u8 *) pdesc, 1, 1, skb); | ||
565 | |||
566 | __skb_queue_tail(&ring->queue, skb); | ||
567 | |||
568 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
569 | |||
570 | rtlpriv->cfg->ops->tx_polling(hw, BEACON_QUEUE); | ||
571 | |||
572 | return true; | ||
573 | } | ||
574 | |||
575 | #define BEACON_PG 0 /*->1*/ | ||
576 | #define PSPOLL_PG 2 | ||
577 | #define NULL_PG 3 | ||
578 | #define PROBERSP_PG 4 /*->5*/ | ||
579 | |||
580 | #define TOTAL_RESERVED_PKT_LEN 768 | ||
581 | |||
582 | static u8 reserved_page_packet[TOTAL_RESERVED_PKT_LEN] = { | ||
583 | /* page 0 beacon */ | ||
584 | 0x80, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, | ||
585 | 0xFF, 0xFF, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, | ||
586 | 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x50, 0x08, | ||
587 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
588 | 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69, | ||
589 | 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C, | ||
590 | 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96, | ||
591 | 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A, | ||
592 | 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C, | ||
593 | 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18, | ||
594 | 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
595 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
596 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
597 | 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02, | ||
598 | 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
599 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
600 | |||
601 | /* page 1 beacon */ | ||
602 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
603 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
604 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
605 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
606 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
607 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
608 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
609 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
610 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
611 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
612 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
613 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
614 | 0x10, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x10, 0x00, | ||
615 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
616 | 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
617 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
618 | |||
619 | /* page 2 ps-poll */ | ||
620 | 0xA4, 0x10, 0x01, 0xC0, 0x00, 0x40, 0x10, 0x10, | ||
621 | 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, | ||
622 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
623 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
624 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
625 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
626 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
627 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
628 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
629 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
630 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
631 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
632 | 0x18, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00, | ||
633 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, | ||
634 | 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
635 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
636 | |||
637 | /* page 3 null */ | ||
638 | 0x48, 0x01, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10, | ||
639 | 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, | ||
640 | 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00, | ||
641 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
642 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
643 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
644 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
645 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
646 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
647 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
648 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
649 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
650 | 0x72, 0x00, 0x20, 0x8C, 0x00, 0x12, 0x00, 0x00, | ||
651 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, | ||
652 | 0x80, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
653 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
654 | |||
655 | /* page 4 probe_resp */ | ||
656 | 0x50, 0x00, 0x00, 0x00, 0x00, 0x40, 0x10, 0x10, | ||
657 | 0x00, 0x03, 0x00, 0xE0, 0x4C, 0x76, 0x00, 0x42, | ||
658 | 0x00, 0x40, 0x10, 0x10, 0x00, 0x03, 0x00, 0x00, | ||
659 | 0x9E, 0x46, 0x15, 0x32, 0x27, 0xF2, 0x2D, 0x00, | ||
660 | 0x64, 0x00, 0x00, 0x04, 0x00, 0x0C, 0x6C, 0x69, | ||
661 | 0x6E, 0x6B, 0x73, 0x79, 0x73, 0x5F, 0x77, 0x6C, | ||
662 | 0x61, 0x6E, 0x01, 0x04, 0x82, 0x84, 0x8B, 0x96, | ||
663 | 0x03, 0x01, 0x01, 0x06, 0x02, 0x00, 0x00, 0x2A, | ||
664 | 0x01, 0x00, 0x32, 0x08, 0x24, 0x30, 0x48, 0x6C, | ||
665 | 0x0C, 0x12, 0x18, 0x60, 0x2D, 0x1A, 0x6C, 0x18, | ||
666 | 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
667 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
668 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
669 | 0x3D, 0x00, 0xDD, 0x06, 0x00, 0xE0, 0x4C, 0x02, | ||
670 | 0x01, 0x70, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
671 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
672 | |||
673 | /* page 5 probe_resp */ | ||
674 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
675 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
676 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
677 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
678 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
679 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
680 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
681 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
682 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
683 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
684 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
685 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
686 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
687 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
688 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
689 | 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, | ||
690 | }; | ||
691 | |||
692 | void rtl92c_set_fw_rsvdpagepkt(struct ieee80211_hw *hw, bool dl_finished) | ||
693 | { | ||
694 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
695 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
696 | struct sk_buff *skb = NULL; | ||
697 | |||
698 | u32 totalpacketlen; | ||
699 | bool rtstatus; | ||
700 | u8 u1RsvdPageLoc[3] = {0}; | ||
701 | bool dlok = false; | ||
702 | |||
703 | u8 *beacon; | ||
704 | u8 *pspoll; | ||
705 | u8 *nullfunc; | ||
706 | u8 *probersp; | ||
707 | /*--------------------------------------------------------- | ||
708 | (1) beacon | ||
709 | ---------------------------------------------------------*/ | ||
710 | beacon = &reserved_page_packet[BEACON_PG * 128]; | ||
711 | SET_80211_HDR_ADDRESS2(beacon, mac->mac_addr); | ||
712 | SET_80211_HDR_ADDRESS3(beacon, mac->bssid); | ||
713 | |||
714 | /*------------------------------------------------------- | ||
715 | (2) ps-poll | ||
716 | --------------------------------------------------------*/ | ||
717 | pspoll = &reserved_page_packet[PSPOLL_PG * 128]; | ||
718 | SET_80211_PS_POLL_AID(pspoll, (mac->assoc_id | 0xc000)); | ||
719 | SET_80211_PS_POLL_BSSID(pspoll, mac->bssid); | ||
720 | SET_80211_PS_POLL_TA(pspoll, mac->mac_addr); | ||
721 | |||
722 | SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(u1RsvdPageLoc, PSPOLL_PG); | ||
723 | |||
724 | /*-------------------------------------------------------- | ||
725 | (3) null data | ||
726 | ---------------------------------------------------------*/ | ||
727 | nullfunc = &reserved_page_packet[NULL_PG * 128]; | ||
728 | SET_80211_HDR_ADDRESS1(nullfunc, mac->bssid); | ||
729 | SET_80211_HDR_ADDRESS2(nullfunc, mac->mac_addr); | ||
730 | SET_80211_HDR_ADDRESS3(nullfunc, mac->bssid); | ||
731 | |||
732 | SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(u1RsvdPageLoc, NULL_PG); | ||
733 | |||
734 | /*--------------------------------------------------------- | ||
735 | (4) probe response | ||
736 | ----------------------------------------------------------*/ | ||
737 | probersp = &reserved_page_packet[PROBERSP_PG * 128]; | ||
738 | SET_80211_HDR_ADDRESS1(probersp, mac->bssid); | ||
739 | SET_80211_HDR_ADDRESS2(probersp, mac->mac_addr); | ||
740 | SET_80211_HDR_ADDRESS3(probersp, mac->bssid); | ||
741 | |||
742 | SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(u1RsvdPageLoc, PROBERSP_PG); | ||
743 | |||
744 | totalpacketlen = TOTAL_RESERVED_PKT_LEN; | ||
745 | |||
746 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_LOUD, | ||
747 | "rtl92c_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n", | ||
748 | &reserved_page_packet[0], totalpacketlen); | ||
749 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, | ||
750 | "rtl92c_set_fw_rsvdpagepkt(): HW_VAR_SET_TX_CMD: ALL\n", | ||
751 | u1RsvdPageLoc, 3); | ||
752 | |||
753 | |||
754 | skb = dev_alloc_skb(totalpacketlen); | ||
755 | memcpy((u8 *) skb_put(skb, totalpacketlen), | ||
756 | &reserved_page_packet, totalpacketlen); | ||
757 | |||
758 | rtstatus = _rtl92c_cmd_send_packet(hw, skb); | ||
759 | |||
760 | if (rtstatus) | ||
761 | dlok = true; | ||
762 | |||
763 | if (dlok) { | ||
764 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
765 | ("Set RSVD page location to Fw.\n")); | ||
766 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_DMESG, | ||
767 | "H2C_RSVDPAGE:\n", | ||
768 | u1RsvdPageLoc, 3); | ||
769 | rtl92c_fill_h2c_cmd(hw, H2C_RSVDPAGE, | ||
770 | sizeof(u1RsvdPageLoc), u1RsvdPageLoc); | ||
771 | } else | ||
772 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
773 | ("Set RSVD page location to Fw FAIL!!!!!!.\n")); | ||
774 | } | ||
775 | EXPORT_SYMBOL(rtl92c_set_fw_rsvdpagepkt); | ||
776 | |||
777 | void rtl92c_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, u8 mstatus) | ||
778 | { | ||
779 | u8 u1_joinbssrpt_parm[1] = {0}; | ||
780 | |||
781 | SET_H2CCMD_JOINBSSRPT_PARM_OPMODE(u1_joinbssrpt_parm, mstatus); | ||
782 | |||
783 | rtl92c_fill_h2c_cmd(hw, H2C_JOINBSSRPT, 1, u1_joinbssrpt_parm); | ||
784 | } | ||
785 | EXPORT_SYMBOL(rtl92c_set_fw_joinbss_report_cmd); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.h b/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.h new file mode 100644 index 000000000000..3d5823c12621 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/fw_common.h | |||
@@ -0,0 +1,98 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C__FW__COMMON__H__ | ||
31 | #define __RTL92C__FW__COMMON__H__ | ||
32 | |||
33 | #define FW_8192C_SIZE 0x3000 | ||
34 | #define FW_8192C_START_ADDRESS 0x1000 | ||
35 | #define FW_8192C_END_ADDRESS 0x3FFF | ||
36 | #define FW_8192C_PAGE_SIZE 4096 | ||
37 | #define FW_8192C_POLLING_DELAY 5 | ||
38 | #define FW_8192C_POLLING_TIMEOUT_COUNT 100 | ||
39 | |||
40 | #define IS_FW_HEADER_EXIST(_pfwhdr) \ | ||
41 | ((_pfwhdr->signature&0xFFF0) == 0x92C0 ||\ | ||
42 | (_pfwhdr->signature&0xFFF0) == 0x88C0) | ||
43 | |||
44 | struct rtl92c_firmware_header { | ||
45 | u16 signature; | ||
46 | u8 category; | ||
47 | u8 function; | ||
48 | u16 version; | ||
49 | u8 subversion; | ||
50 | u8 rsvd1; | ||
51 | u8 month; | ||
52 | u8 date; | ||
53 | u8 hour; | ||
54 | u8 minute; | ||
55 | u16 ramcodeSize; | ||
56 | u16 rsvd2; | ||
57 | u32 svnindex; | ||
58 | u32 rsvd3; | ||
59 | u32 rsvd4; | ||
60 | u32 rsvd5; | ||
61 | }; | ||
62 | |||
63 | enum rtl8192c_h2c_cmd { | ||
64 | H2C_AP_OFFLOAD = 0, | ||
65 | H2C_SETPWRMODE = 1, | ||
66 | H2C_JOINBSSRPT = 2, | ||
67 | H2C_RSVDPAGE = 3, | ||
68 | H2C_RSSI_REPORT = 5, | ||
69 | H2C_RA_MASK = 6, | ||
70 | MAX_H2CCMD | ||
71 | }; | ||
72 | |||
73 | #define pagenum_128(_len) (u32)(((_len)>>7) + ((_len)&0x7F ? 1 : 0)) | ||
74 | |||
75 | #define SET_H2CCMD_PWRMODE_PARM_MODE(__ph2ccmd, __val) \ | ||
76 | SET_BITS_TO_LE_1BYTE(__ph2ccmd, 0, 8, __val) | ||
77 | #define SET_H2CCMD_PWRMODE_PARM_SMART_PS(__ph2ccmd, __val) \ | ||
78 | SET_BITS_TO_LE_1BYTE((__ph2ccmd)+1, 0, 8, __val) | ||
79 | #define SET_H2CCMD_PWRMODE_PARM_BCN_PASS_TIME(__ph2ccmd, __val) \ | ||
80 | SET_BITS_TO_LE_1BYTE((__ph2ccmd)+2, 0, 8, __val) | ||
81 | #define SET_H2CCMD_JOINBSSRPT_PARM_OPMODE(__ph2ccmd, __val) \ | ||
82 | SET_BITS_TO_LE_1BYTE(__ph2ccmd, 0, 8, __val) | ||
83 | #define SET_H2CCMD_RSVDPAGE_LOC_PROBE_RSP(__ph2ccmd, __val) \ | ||
84 | SET_BITS_TO_LE_1BYTE(__ph2ccmd, 0, 8, __val) | ||
85 | #define SET_H2CCMD_RSVDPAGE_LOC_PSPOLL(__ph2ccmd, __val) \ | ||
86 | SET_BITS_TO_LE_1BYTE((__ph2ccmd)+1, 0, 8, __val) | ||
87 | #define SET_H2CCMD_RSVDPAGE_LOC_NULL_DATA(__ph2ccmd, __val) \ | ||
88 | SET_BITS_TO_LE_1BYTE((__ph2ccmd)+2, 0, 8, __val) | ||
89 | |||
90 | int rtl92c_download_fw(struct ieee80211_hw *hw); | ||
91 | void rtl92c_fill_h2c_cmd(struct ieee80211_hw *hw, u8 element_id, | ||
92 | u32 cmd_len, u8 *p_cmdbuffer); | ||
93 | void rtl92c_firmware_selfreset(struct ieee80211_hw *hw); | ||
94 | void rtl92c_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 mode); | ||
95 | void rtl92c_set_fw_rsvdpagepkt(struct ieee80211_hw *hw, bool b_dl_finished); | ||
96 | void rtl92c_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, u8 mstatus); | ||
97 | |||
98 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/main.c b/drivers/net/wireless/rtlwifi/rtl8192c/main.c new file mode 100644 index 000000000000..2f624fc27499 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/main.c | |||
@@ -0,0 +1,39 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | |||
32 | |||
33 | MODULE_AUTHOR("lizhaoming <chaoming_li@realsil.com.cn>"); | ||
34 | MODULE_AUTHOR("Realtek WlanFAE <wlanfae@realtek.com>"); | ||
35 | MODULE_AUTHOR("Georgia <georgia@realtek.com>"); | ||
36 | MODULE_AUTHOR("Ziv Huang <ziv_huang@realtek.com>"); | ||
37 | MODULE_AUTHOR("Larry Finger <Larry.Finger@lwfinger.net>"); | ||
38 | MODULE_LICENSE("GPL"); | ||
39 | MODULE_DESCRIPTION("Realtek 8192C/8188C 802.11n PCI wireless"); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.c b/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.c new file mode 100644 index 000000000000..d2cc81586a6a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.c | |||
@@ -0,0 +1,2032 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../rtl8192ce/reg.h" | ||
32 | #include "../rtl8192ce/def.h" | ||
33 | #include "dm_common.h" | ||
34 | #include "phy_common.h" | ||
35 | |||
36 | /* Define macro to shorten lines */ | ||
37 | #define MCS_TXPWR mcs_txpwrlevel_origoffset | ||
38 | |||
39 | u32 rtl92c_phy_query_bb_reg(struct ieee80211_hw *hw, u32 regaddr, u32 bitmask) | ||
40 | { | ||
41 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
42 | u32 returnvalue, originalvalue, bitshift; | ||
43 | |||
44 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), " | ||
45 | "bitmask(%#x)\n", regaddr, | ||
46 | bitmask)); | ||
47 | originalvalue = rtl_read_dword(rtlpriv, regaddr); | ||
48 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
49 | returnvalue = (originalvalue & bitmask) >> bitshift; | ||
50 | |||
51 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("BBR MASK=0x%x " | ||
52 | "Addr[0x%x]=0x%x\n", bitmask, | ||
53 | regaddr, originalvalue)); | ||
54 | |||
55 | return returnvalue; | ||
56 | |||
57 | } | ||
58 | EXPORT_SYMBOL(rtl92c_phy_query_bb_reg); | ||
59 | |||
60 | void rtl92c_phy_set_bb_reg(struct ieee80211_hw *hw, | ||
61 | u32 regaddr, u32 bitmask, u32 data) | ||
62 | { | ||
63 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
64 | u32 originalvalue, bitshift; | ||
65 | |||
66 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)," | ||
67 | " data(%#x)\n", regaddr, bitmask, | ||
68 | data)); | ||
69 | |||
70 | if (bitmask != MASKDWORD) { | ||
71 | originalvalue = rtl_read_dword(rtlpriv, regaddr); | ||
72 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
73 | data = ((originalvalue & (~bitmask)) | (data << bitshift)); | ||
74 | } | ||
75 | |||
76 | rtl_write_dword(rtlpriv, regaddr, data); | ||
77 | |||
78 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)," | ||
79 | " data(%#x)\n", regaddr, bitmask, | ||
80 | data)); | ||
81 | |||
82 | } | ||
83 | EXPORT_SYMBOL(rtl92c_phy_set_bb_reg); | ||
84 | |||
85 | u32 _rtl92c_phy_fw_rf_serial_read(struct ieee80211_hw *hw, | ||
86 | enum radio_path rfpath, u32 offset) | ||
87 | { | ||
88 | RT_ASSERT(false, ("deprecated!\n")); | ||
89 | return 0; | ||
90 | |||
91 | } | ||
92 | EXPORT_SYMBOL(_rtl92c_phy_fw_rf_serial_read); | ||
93 | |||
94 | void _rtl92c_phy_fw_rf_serial_write(struct ieee80211_hw *hw, | ||
95 | enum radio_path rfpath, u32 offset, | ||
96 | u32 data) | ||
97 | { | ||
98 | RT_ASSERT(false, ("deprecated!\n")); | ||
99 | } | ||
100 | EXPORT_SYMBOL(_rtl92c_phy_fw_rf_serial_write); | ||
101 | |||
102 | u32 _rtl92c_phy_rf_serial_read(struct ieee80211_hw *hw, | ||
103 | enum radio_path rfpath, u32 offset) | ||
104 | { | ||
105 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
106 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
107 | struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
108 | u32 newoffset; | ||
109 | u32 tmplong, tmplong2; | ||
110 | u8 rfpi_enable = 0; | ||
111 | u32 retvalue; | ||
112 | |||
113 | offset &= 0x3f; | ||
114 | newoffset = offset; | ||
115 | if (RT_CANNOT_IO(hw)) { | ||
116 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("return all one\n")); | ||
117 | return 0xFFFFFFFF; | ||
118 | } | ||
119 | tmplong = rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD); | ||
120 | if (rfpath == RF90_PATH_A) | ||
121 | tmplong2 = tmplong; | ||
122 | else | ||
123 | tmplong2 = rtl_get_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD); | ||
124 | tmplong2 = (tmplong2 & (~BLSSIREADADDRESS)) | | ||
125 | (newoffset << 23) | BLSSIREADEDGE; | ||
126 | rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, | ||
127 | tmplong & (~BLSSIREADEDGE)); | ||
128 | mdelay(1); | ||
129 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD, tmplong2); | ||
130 | mdelay(1); | ||
131 | rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, | ||
132 | tmplong | BLSSIREADEDGE); | ||
133 | mdelay(1); | ||
134 | if (rfpath == RF90_PATH_A) | ||
135 | rfpi_enable = (u8) rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER1, | ||
136 | BIT(8)); | ||
137 | else if (rfpath == RF90_PATH_B) | ||
138 | rfpi_enable = (u8) rtl_get_bbreg(hw, RFPGA0_XB_HSSIPARAMETER1, | ||
139 | BIT(8)); | ||
140 | if (rfpi_enable) | ||
141 | retvalue = rtl_get_bbreg(hw, pphyreg->rflssi_readbackpi, | ||
142 | BLSSIREADBACKDATA); | ||
143 | else | ||
144 | retvalue = rtl_get_bbreg(hw, pphyreg->rflssi_readback, | ||
145 | BLSSIREADBACKDATA); | ||
146 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("RFR-%d Addr[0x%x]=0x%x\n", | ||
147 | rfpath, pphyreg->rflssi_readback, | ||
148 | retvalue)); | ||
149 | return retvalue; | ||
150 | } | ||
151 | EXPORT_SYMBOL(_rtl92c_phy_rf_serial_read); | ||
152 | |||
153 | void _rtl92c_phy_rf_serial_write(struct ieee80211_hw *hw, | ||
154 | enum radio_path rfpath, u32 offset, | ||
155 | u32 data) | ||
156 | { | ||
157 | u32 data_and_addr; | ||
158 | u32 newoffset; | ||
159 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
160 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
161 | struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
162 | |||
163 | if (RT_CANNOT_IO(hw)) { | ||
164 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("stop\n")); | ||
165 | return; | ||
166 | } | ||
167 | offset &= 0x3f; | ||
168 | newoffset = offset; | ||
169 | data_and_addr = ((newoffset << 20) | (data & 0x000fffff)) & 0x0fffffff; | ||
170 | rtl_set_bbreg(hw, pphyreg->rf3wire_offset, MASKDWORD, data_and_addr); | ||
171 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("RFW-%d Addr[0x%x]=0x%x\n", | ||
172 | rfpath, pphyreg->rf3wire_offset, | ||
173 | data_and_addr)); | ||
174 | } | ||
175 | EXPORT_SYMBOL(_rtl92c_phy_rf_serial_write); | ||
176 | |||
177 | u32 _rtl92c_phy_calculate_bit_shift(u32 bitmask) | ||
178 | { | ||
179 | u32 i; | ||
180 | |||
181 | for (i = 0; i <= 31; i++) { | ||
182 | if (((bitmask >> i) & 0x1) == 1) | ||
183 | break; | ||
184 | } | ||
185 | return i; | ||
186 | } | ||
187 | EXPORT_SYMBOL(_rtl92c_phy_calculate_bit_shift); | ||
188 | |||
189 | static void _rtl92c_phy_bb_config_1t(struct ieee80211_hw *hw) | ||
190 | { | ||
191 | rtl_set_bbreg(hw, RFPGA0_TXINFO, 0x3, 0x2); | ||
192 | rtl_set_bbreg(hw, RFPGA1_TXINFO, 0x300033, 0x200022); | ||
193 | rtl_set_bbreg(hw, RCCK0_AFESETTING, MASKBYTE3, 0x45); | ||
194 | rtl_set_bbreg(hw, ROFDM0_TRXPATHENABLE, MASKBYTE0, 0x23); | ||
195 | rtl_set_bbreg(hw, ROFDM0_AGCPARAMETER1, 0x30, 0x1); | ||
196 | rtl_set_bbreg(hw, 0xe74, 0x0c000000, 0x2); | ||
197 | rtl_set_bbreg(hw, 0xe78, 0x0c000000, 0x2); | ||
198 | rtl_set_bbreg(hw, 0xe7c, 0x0c000000, 0x2); | ||
199 | rtl_set_bbreg(hw, 0xe80, 0x0c000000, 0x2); | ||
200 | rtl_set_bbreg(hw, 0xe88, 0x0c000000, 0x2); | ||
201 | } | ||
202 | |||
203 | bool rtl92c_phy_rf_config(struct ieee80211_hw *hw) | ||
204 | { | ||
205 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
206 | |||
207 | return rtlpriv->cfg->ops->phy_rf6052_config(hw); | ||
208 | } | ||
209 | EXPORT_SYMBOL(rtl92c_phy_rf_config); | ||
210 | |||
211 | bool _rtl92c_phy_bb8192c_config_parafile(struct ieee80211_hw *hw) | ||
212 | { | ||
213 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
214 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
215 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
216 | bool rtstatus; | ||
217 | |||
218 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("==>\n")); | ||
219 | rtstatus = rtlpriv->cfg->ops->config_bb_with_headerfile(hw, | ||
220 | BASEBAND_CONFIG_PHY_REG); | ||
221 | if (rtstatus != true) { | ||
222 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("Write BB Reg Fail!!")); | ||
223 | return false; | ||
224 | } | ||
225 | if (rtlphy->rf_type == RF_1T2R) { | ||
226 | _rtl92c_phy_bb_config_1t(hw); | ||
227 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Config to 1T!!\n")); | ||
228 | } | ||
229 | if (rtlefuse->autoload_failflag == false) { | ||
230 | rtlphy->pwrgroup_cnt = 0; | ||
231 | rtstatus = rtlpriv->cfg->ops->config_bb_with_pgheaderfile(hw, | ||
232 | BASEBAND_CONFIG_PHY_REG); | ||
233 | } | ||
234 | if (rtstatus != true) { | ||
235 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("BB_PG Reg Fail!!")); | ||
236 | return false; | ||
237 | } | ||
238 | rtstatus = rtlpriv->cfg->ops->config_bb_with_headerfile(hw, | ||
239 | BASEBAND_CONFIG_AGC_TAB); | ||
240 | if (rtstatus != true) { | ||
241 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("AGC Table Fail\n")); | ||
242 | return false; | ||
243 | } | ||
244 | rtlphy->cck_high_power = (bool) (rtl_get_bbreg(hw, | ||
245 | RFPGA0_XA_HSSIPARAMETER2, | ||
246 | 0x200)); | ||
247 | |||
248 | return true; | ||
249 | } | ||
250 | EXPORT_SYMBOL(_rtl92c_phy_bb8192c_config_parafile); | ||
251 | |||
252 | void _rtl92c_store_pwrIndex_diffrate_offset(struct ieee80211_hw *hw, | ||
253 | u32 regaddr, u32 bitmask, | ||
254 | u32 data) | ||
255 | { | ||
256 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
257 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
258 | |||
259 | if (regaddr == RTXAGC_A_RATE18_06) { | ||
260 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][0] = data; | ||
261 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
262 | ("MCSTxPowerLevelOriginalOffset[%d][0] = 0x%x\n", | ||
263 | rtlphy->pwrgroup_cnt, | ||
264 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][0])); | ||
265 | } | ||
266 | if (regaddr == RTXAGC_A_RATE54_24) { | ||
267 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][1] = data; | ||
268 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
269 | ("MCSTxPowerLevelOriginalOffset[%d][1] = 0x%x\n", | ||
270 | rtlphy->pwrgroup_cnt, | ||
271 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][1])); | ||
272 | } | ||
273 | if (regaddr == RTXAGC_A_CCK1_MCS32) { | ||
274 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][6] = data; | ||
275 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
276 | ("MCSTxPowerLevelOriginalOffset[%d][6] = 0x%x\n", | ||
277 | rtlphy->pwrgroup_cnt, | ||
278 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][6])); | ||
279 | } | ||
280 | if (regaddr == RTXAGC_B_CCK11_A_CCK2_11 && bitmask == 0xffffff00) { | ||
281 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][7] = data; | ||
282 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
283 | ("MCSTxPowerLevelOriginalOffset[%d][7] = 0x%x\n", | ||
284 | rtlphy->pwrgroup_cnt, | ||
285 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][7])); | ||
286 | } | ||
287 | if (regaddr == RTXAGC_A_MCS03_MCS00) { | ||
288 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][2] = data; | ||
289 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
290 | ("MCSTxPowerLevelOriginalOffset[%d][2] = 0x%x\n", | ||
291 | rtlphy->pwrgroup_cnt, | ||
292 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][2])); | ||
293 | } | ||
294 | if (regaddr == RTXAGC_A_MCS07_MCS04) { | ||
295 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][3] = data; | ||
296 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
297 | ("MCSTxPowerLevelOriginalOffset[%d][3] = 0x%x\n", | ||
298 | rtlphy->pwrgroup_cnt, | ||
299 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][3])); | ||
300 | } | ||
301 | if (regaddr == RTXAGC_A_MCS11_MCS08) { | ||
302 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][4] = data; | ||
303 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
304 | ("MCSTxPowerLevelOriginalOffset[%d][4] = 0x%x\n", | ||
305 | rtlphy->pwrgroup_cnt, | ||
306 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][4])); | ||
307 | } | ||
308 | if (regaddr == RTXAGC_A_MCS15_MCS12) { | ||
309 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][5] = data; | ||
310 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
311 | ("MCSTxPowerLevelOriginalOffset[%d][5] = 0x%x\n", | ||
312 | rtlphy->pwrgroup_cnt, | ||
313 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][5])); | ||
314 | } | ||
315 | if (regaddr == RTXAGC_B_RATE18_06) { | ||
316 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][8] = data; | ||
317 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
318 | ("MCSTxPowerLevelOriginalOffset[%d][8] = 0x%x\n", | ||
319 | rtlphy->pwrgroup_cnt, | ||
320 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][8])); | ||
321 | } | ||
322 | if (regaddr == RTXAGC_B_RATE54_24) { | ||
323 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][9] = data; | ||
324 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
325 | ("MCSTxPowerLevelOriginalOffset[%d][9] = 0x%x\n", | ||
326 | rtlphy->pwrgroup_cnt, | ||
327 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][9])); | ||
328 | } | ||
329 | if (regaddr == RTXAGC_B_CCK1_55_MCS32) { | ||
330 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][14] = data; | ||
331 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
332 | ("MCSTxPowerLevelOriginalOffset[%d][14] = 0x%x\n", | ||
333 | rtlphy->pwrgroup_cnt, | ||
334 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][14])); | ||
335 | } | ||
336 | if (regaddr == RTXAGC_B_CCK11_A_CCK2_11 && bitmask == 0x000000ff) { | ||
337 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][15] = data; | ||
338 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
339 | ("MCSTxPowerLevelOriginalOffset[%d][15] = 0x%x\n", | ||
340 | rtlphy->pwrgroup_cnt, | ||
341 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][15])); | ||
342 | } | ||
343 | if (regaddr == RTXAGC_B_MCS03_MCS00) { | ||
344 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][10] = data; | ||
345 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
346 | ("MCSTxPowerLevelOriginalOffset[%d][10] = 0x%x\n", | ||
347 | rtlphy->pwrgroup_cnt, | ||
348 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][10])); | ||
349 | } | ||
350 | if (regaddr == RTXAGC_B_MCS07_MCS04) { | ||
351 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][11] = data; | ||
352 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
353 | ("MCSTxPowerLevelOriginalOffset[%d][11] = 0x%x\n", | ||
354 | rtlphy->pwrgroup_cnt, | ||
355 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][11])); | ||
356 | } | ||
357 | if (regaddr == RTXAGC_B_MCS11_MCS08) { | ||
358 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][12] = data; | ||
359 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
360 | ("MCSTxPowerLevelOriginalOffset[%d][12] = 0x%x\n", | ||
361 | rtlphy->pwrgroup_cnt, | ||
362 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][12])); | ||
363 | } | ||
364 | if (regaddr == RTXAGC_B_MCS15_MCS12) { | ||
365 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][13] = data; | ||
366 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
367 | ("MCSTxPowerLevelOriginalOffset[%d][13] = 0x%x\n", | ||
368 | rtlphy->pwrgroup_cnt, | ||
369 | rtlphy->MCS_TXPWR[rtlphy->pwrgroup_cnt][13])); | ||
370 | |||
371 | rtlphy->pwrgroup_cnt++; | ||
372 | } | ||
373 | } | ||
374 | EXPORT_SYMBOL(_rtl92c_store_pwrIndex_diffrate_offset); | ||
375 | |||
376 | void rtl92c_phy_get_hw_reg_originalvalue(struct ieee80211_hw *hw) | ||
377 | { | ||
378 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
379 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
380 | |||
381 | rtlphy->default_initialgain[0] = | ||
382 | (u8) rtl_get_bbreg(hw, ROFDM0_XAAGCCORE1, MASKBYTE0); | ||
383 | rtlphy->default_initialgain[1] = | ||
384 | (u8) rtl_get_bbreg(hw, ROFDM0_XBAGCCORE1, MASKBYTE0); | ||
385 | rtlphy->default_initialgain[2] = | ||
386 | (u8) rtl_get_bbreg(hw, ROFDM0_XCAGCCORE1, MASKBYTE0); | ||
387 | rtlphy->default_initialgain[3] = | ||
388 | (u8) rtl_get_bbreg(hw, ROFDM0_XDAGCCORE1, MASKBYTE0); | ||
389 | |||
390 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
391 | ("Default initial gain (c50=0x%x, " | ||
392 | "c58=0x%x, c60=0x%x, c68=0x%x\n", | ||
393 | rtlphy->default_initialgain[0], | ||
394 | rtlphy->default_initialgain[1], | ||
395 | rtlphy->default_initialgain[2], | ||
396 | rtlphy->default_initialgain[3])); | ||
397 | |||
398 | rtlphy->framesync = (u8) rtl_get_bbreg(hw, | ||
399 | ROFDM0_RXDETECTOR3, MASKBYTE0); | ||
400 | rtlphy->framesync_c34 = rtl_get_bbreg(hw, | ||
401 | ROFDM0_RXDETECTOR2, MASKDWORD); | ||
402 | |||
403 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
404 | ("Default framesync (0x%x) = 0x%x\n", | ||
405 | ROFDM0_RXDETECTOR3, rtlphy->framesync)); | ||
406 | } | ||
407 | |||
408 | void _rtl92c_phy_init_bb_rf_register_definition(struct ieee80211_hw *hw) | ||
409 | { | ||
410 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
411 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
412 | |||
413 | rtlphy->phyreg_def[RF90_PATH_A].rfintfs = RFPGA0_XAB_RFINTERFACESW; | ||
414 | rtlphy->phyreg_def[RF90_PATH_B].rfintfs = RFPGA0_XAB_RFINTERFACESW; | ||
415 | rtlphy->phyreg_def[RF90_PATH_C].rfintfs = RFPGA0_XCD_RFINTERFACESW; | ||
416 | rtlphy->phyreg_def[RF90_PATH_D].rfintfs = RFPGA0_XCD_RFINTERFACESW; | ||
417 | |||
418 | rtlphy->phyreg_def[RF90_PATH_A].rfintfi = RFPGA0_XAB_RFINTERFACERB; | ||
419 | rtlphy->phyreg_def[RF90_PATH_B].rfintfi = RFPGA0_XAB_RFINTERFACERB; | ||
420 | rtlphy->phyreg_def[RF90_PATH_C].rfintfi = RFPGA0_XCD_RFINTERFACERB; | ||
421 | rtlphy->phyreg_def[RF90_PATH_D].rfintfi = RFPGA0_XCD_RFINTERFACERB; | ||
422 | |||
423 | rtlphy->phyreg_def[RF90_PATH_A].rfintfo = RFPGA0_XA_RFINTERFACEOE; | ||
424 | rtlphy->phyreg_def[RF90_PATH_B].rfintfo = RFPGA0_XB_RFINTERFACEOE; | ||
425 | |||
426 | rtlphy->phyreg_def[RF90_PATH_A].rfintfe = RFPGA0_XA_RFINTERFACEOE; | ||
427 | rtlphy->phyreg_def[RF90_PATH_B].rfintfe = RFPGA0_XB_RFINTERFACEOE; | ||
428 | |||
429 | rtlphy->phyreg_def[RF90_PATH_A].rf3wire_offset = | ||
430 | RFPGA0_XA_LSSIPARAMETER; | ||
431 | rtlphy->phyreg_def[RF90_PATH_B].rf3wire_offset = | ||
432 | RFPGA0_XB_LSSIPARAMETER; | ||
433 | |||
434 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_select = rFPGA0_XAB_RFPARAMETER; | ||
435 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_select = rFPGA0_XAB_RFPARAMETER; | ||
436 | rtlphy->phyreg_def[RF90_PATH_C].rflssi_select = rFPGA0_XCD_RFPARAMETER; | ||
437 | rtlphy->phyreg_def[RF90_PATH_D].rflssi_select = rFPGA0_XCD_RFPARAMETER; | ||
438 | |||
439 | rtlphy->phyreg_def[RF90_PATH_A].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
440 | rtlphy->phyreg_def[RF90_PATH_B].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
441 | rtlphy->phyreg_def[RF90_PATH_C].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
442 | rtlphy->phyreg_def[RF90_PATH_D].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
443 | |||
444 | rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para1 = RFPGA0_XA_HSSIPARAMETER1; | ||
445 | rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para1 = RFPGA0_XB_HSSIPARAMETER1; | ||
446 | |||
447 | rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para2 = RFPGA0_XA_HSSIPARAMETER2; | ||
448 | rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para2 = RFPGA0_XB_HSSIPARAMETER2; | ||
449 | |||
450 | rtlphy->phyreg_def[RF90_PATH_A].rfswitch_control = | ||
451 | RFPGA0_XAB_SWITCHCONTROL; | ||
452 | rtlphy->phyreg_def[RF90_PATH_B].rfswitch_control = | ||
453 | RFPGA0_XAB_SWITCHCONTROL; | ||
454 | rtlphy->phyreg_def[RF90_PATH_C].rfswitch_control = | ||
455 | RFPGA0_XCD_SWITCHCONTROL; | ||
456 | rtlphy->phyreg_def[RF90_PATH_D].rfswitch_control = | ||
457 | RFPGA0_XCD_SWITCHCONTROL; | ||
458 | |||
459 | rtlphy->phyreg_def[RF90_PATH_A].rfagc_control1 = ROFDM0_XAAGCCORE1; | ||
460 | rtlphy->phyreg_def[RF90_PATH_B].rfagc_control1 = ROFDM0_XBAGCCORE1; | ||
461 | rtlphy->phyreg_def[RF90_PATH_C].rfagc_control1 = ROFDM0_XCAGCCORE1; | ||
462 | rtlphy->phyreg_def[RF90_PATH_D].rfagc_control1 = ROFDM0_XDAGCCORE1; | ||
463 | |||
464 | rtlphy->phyreg_def[RF90_PATH_A].rfagc_control2 = ROFDM0_XAAGCCORE2; | ||
465 | rtlphy->phyreg_def[RF90_PATH_B].rfagc_control2 = ROFDM0_XBAGCCORE2; | ||
466 | rtlphy->phyreg_def[RF90_PATH_C].rfagc_control2 = ROFDM0_XCAGCCORE2; | ||
467 | rtlphy->phyreg_def[RF90_PATH_D].rfagc_control2 = ROFDM0_XDAGCCORE2; | ||
468 | |||
469 | rtlphy->phyreg_def[RF90_PATH_A].rfrxiq_imbalance = | ||
470 | ROFDM0_XARXIQIMBALANCE; | ||
471 | rtlphy->phyreg_def[RF90_PATH_B].rfrxiq_imbalance = | ||
472 | ROFDM0_XBRXIQIMBALANCE; | ||
473 | rtlphy->phyreg_def[RF90_PATH_C].rfrxiq_imbalance = | ||
474 | ROFDM0_XCRXIQIMBANLANCE; | ||
475 | rtlphy->phyreg_def[RF90_PATH_D].rfrxiq_imbalance = | ||
476 | ROFDM0_XDRXIQIMBALANCE; | ||
477 | |||
478 | rtlphy->phyreg_def[RF90_PATH_A].rfrx_afe = ROFDM0_XARXAFE; | ||
479 | rtlphy->phyreg_def[RF90_PATH_B].rfrx_afe = ROFDM0_XBRXAFE; | ||
480 | rtlphy->phyreg_def[RF90_PATH_C].rfrx_afe = ROFDM0_XCRXAFE; | ||
481 | rtlphy->phyreg_def[RF90_PATH_D].rfrx_afe = ROFDM0_XDRXAFE; | ||
482 | |||
483 | rtlphy->phyreg_def[RF90_PATH_A].rftxiq_imbalance = | ||
484 | ROFDM0_XATXIQIMBALANCE; | ||
485 | rtlphy->phyreg_def[RF90_PATH_B].rftxiq_imbalance = | ||
486 | ROFDM0_XBTXIQIMBALANCE; | ||
487 | rtlphy->phyreg_def[RF90_PATH_C].rftxiq_imbalance = | ||
488 | ROFDM0_XCTXIQIMBALANCE; | ||
489 | rtlphy->phyreg_def[RF90_PATH_D].rftxiq_imbalance = | ||
490 | ROFDM0_XDTXIQIMBALANCE; | ||
491 | |||
492 | rtlphy->phyreg_def[RF90_PATH_A].rftx_afe = ROFDM0_XATXAFE; | ||
493 | rtlphy->phyreg_def[RF90_PATH_B].rftx_afe = ROFDM0_XBTXAFE; | ||
494 | rtlphy->phyreg_def[RF90_PATH_C].rftx_afe = ROFDM0_XCTXAFE; | ||
495 | rtlphy->phyreg_def[RF90_PATH_D].rftx_afe = ROFDM0_XDTXAFE; | ||
496 | |||
497 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_readback = | ||
498 | RFPGA0_XA_LSSIREADBACK; | ||
499 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_readback = | ||
500 | RFPGA0_XB_LSSIREADBACK; | ||
501 | rtlphy->phyreg_def[RF90_PATH_C].rflssi_readback = | ||
502 | RFPGA0_XC_LSSIREADBACK; | ||
503 | rtlphy->phyreg_def[RF90_PATH_D].rflssi_readback = | ||
504 | RFPGA0_XD_LSSIREADBACK; | ||
505 | |||
506 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_readbackpi = | ||
507 | TRANSCEIVEA_HSPI_READBACK; | ||
508 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_readbackpi = | ||
509 | TRANSCEIVEB_HSPI_READBACK; | ||
510 | |||
511 | } | ||
512 | EXPORT_SYMBOL(_rtl92c_phy_init_bb_rf_register_definition); | ||
513 | |||
514 | void rtl92c_phy_get_txpower_level(struct ieee80211_hw *hw, long *powerlevel) | ||
515 | { | ||
516 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
517 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
518 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
519 | u8 txpwr_level; | ||
520 | long txpwr_dbm; | ||
521 | |||
522 | txpwr_level = rtlphy->cur_cck_txpwridx; | ||
523 | txpwr_dbm = _rtl92c_phy_txpwr_idx_to_dbm(hw, | ||
524 | WIRELESS_MODE_B, txpwr_level); | ||
525 | txpwr_level = rtlphy->cur_ofdm24g_txpwridx + | ||
526 | rtlefuse->legacy_ht_txpowerdiff; | ||
527 | if (_rtl92c_phy_txpwr_idx_to_dbm(hw, | ||
528 | WIRELESS_MODE_G, | ||
529 | txpwr_level) > txpwr_dbm) | ||
530 | txpwr_dbm = | ||
531 | _rtl92c_phy_txpwr_idx_to_dbm(hw, WIRELESS_MODE_G, | ||
532 | txpwr_level); | ||
533 | txpwr_level = rtlphy->cur_ofdm24g_txpwridx; | ||
534 | if (_rtl92c_phy_txpwr_idx_to_dbm(hw, | ||
535 | WIRELESS_MODE_N_24G, | ||
536 | txpwr_level) > txpwr_dbm) | ||
537 | txpwr_dbm = | ||
538 | _rtl92c_phy_txpwr_idx_to_dbm(hw, WIRELESS_MODE_N_24G, | ||
539 | txpwr_level); | ||
540 | *powerlevel = txpwr_dbm; | ||
541 | } | ||
542 | |||
543 | static void _rtl92c_get_txpower_index(struct ieee80211_hw *hw, u8 channel, | ||
544 | u8 *cckpowerlevel, u8 *ofdmpowerlevel) | ||
545 | { | ||
546 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
547 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
548 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
549 | u8 index = (channel - 1); | ||
550 | |||
551 | cckpowerlevel[RF90_PATH_A] = | ||
552 | rtlefuse->txpwrlevel_cck[RF90_PATH_A][index]; | ||
553 | cckpowerlevel[RF90_PATH_B] = | ||
554 | rtlefuse->txpwrlevel_cck[RF90_PATH_B][index]; | ||
555 | if (get_rf_type(rtlphy) == RF_1T2R || get_rf_type(rtlphy) == RF_1T1R) { | ||
556 | ofdmpowerlevel[RF90_PATH_A] = | ||
557 | rtlefuse->txpwrlevel_ht40_1s[RF90_PATH_A][index]; | ||
558 | ofdmpowerlevel[RF90_PATH_B] = | ||
559 | rtlefuse->txpwrlevel_ht40_1s[RF90_PATH_B][index]; | ||
560 | } else if (get_rf_type(rtlphy) == RF_2T2R) { | ||
561 | ofdmpowerlevel[RF90_PATH_A] = | ||
562 | rtlefuse->txpwrlevel_ht40_2s[RF90_PATH_A][index]; | ||
563 | ofdmpowerlevel[RF90_PATH_B] = | ||
564 | rtlefuse->txpwrlevel_ht40_2s[RF90_PATH_B][index]; | ||
565 | } | ||
566 | } | ||
567 | |||
568 | static void _rtl92c_ccxpower_index_check(struct ieee80211_hw *hw, | ||
569 | u8 channel, u8 *cckpowerlevel, | ||
570 | u8 *ofdmpowerlevel) | ||
571 | { | ||
572 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
573 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
574 | |||
575 | rtlphy->cur_cck_txpwridx = cckpowerlevel[0]; | ||
576 | rtlphy->cur_ofdm24g_txpwridx = ofdmpowerlevel[0]; | ||
577 | |||
578 | } | ||
579 | |||
580 | void rtl92c_phy_set_txpower_level(struct ieee80211_hw *hw, u8 channel) | ||
581 | { | ||
582 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
583 | struct rtl_efuse *rtlefuse = rtl_efuse(rtlpriv); | ||
584 | u8 cckpowerlevel[2], ofdmpowerlevel[2]; | ||
585 | |||
586 | if (rtlefuse->txpwr_fromeprom == false) | ||
587 | return; | ||
588 | _rtl92c_get_txpower_index(hw, channel, | ||
589 | &cckpowerlevel[0], &ofdmpowerlevel[0]); | ||
590 | _rtl92c_ccxpower_index_check(hw, | ||
591 | channel, &cckpowerlevel[0], | ||
592 | &ofdmpowerlevel[0]); | ||
593 | rtlpriv->cfg->ops->phy_rf6052_set_cck_txpower(hw, &cckpowerlevel[0]); | ||
594 | rtlpriv->cfg->ops->phy_rf6052_set_ofdm_txpower(hw, &ofdmpowerlevel[0], | ||
595 | channel); | ||
596 | } | ||
597 | EXPORT_SYMBOL(rtl92c_phy_set_txpower_level); | ||
598 | |||
599 | bool rtl92c_phy_update_txpower_dbm(struct ieee80211_hw *hw, long power_indbm) | ||
600 | { | ||
601 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
602 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
603 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
604 | u8 idx; | ||
605 | u8 rf_path; | ||
606 | u8 ccktxpwridx = _rtl92c_phy_dbm_to_txpwr_Idx(hw, | ||
607 | WIRELESS_MODE_B, | ||
608 | power_indbm); | ||
609 | u8 ofdmtxpwridx = _rtl92c_phy_dbm_to_txpwr_Idx(hw, | ||
610 | WIRELESS_MODE_N_24G, | ||
611 | power_indbm); | ||
612 | if (ofdmtxpwridx - rtlefuse->legacy_ht_txpowerdiff > 0) | ||
613 | ofdmtxpwridx -= rtlefuse->legacy_ht_txpowerdiff; | ||
614 | else | ||
615 | ofdmtxpwridx = 0; | ||
616 | RT_TRACE(rtlpriv, COMP_TXAGC, DBG_TRACE, | ||
617 | ("%lx dBm, ccktxpwridx = %d, ofdmtxpwridx = %d\n", | ||
618 | power_indbm, ccktxpwridx, ofdmtxpwridx)); | ||
619 | for (idx = 0; idx < 14; idx++) { | ||
620 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
621 | rtlefuse->txpwrlevel_cck[rf_path][idx] = ccktxpwridx; | ||
622 | rtlefuse->txpwrlevel_ht40_1s[rf_path][idx] = | ||
623 | ofdmtxpwridx; | ||
624 | rtlefuse->txpwrlevel_ht40_2s[rf_path][idx] = | ||
625 | ofdmtxpwridx; | ||
626 | } | ||
627 | } | ||
628 | rtl92c_phy_set_txpower_level(hw, rtlphy->current_channel); | ||
629 | return true; | ||
630 | } | ||
631 | EXPORT_SYMBOL(rtl92c_phy_update_txpower_dbm); | ||
632 | |||
633 | u8 _rtl92c_phy_dbm_to_txpwr_Idx(struct ieee80211_hw *hw, | ||
634 | enum wireless_mode wirelessmode, | ||
635 | long power_indbm) | ||
636 | { | ||
637 | u8 txpwridx; | ||
638 | long offset; | ||
639 | |||
640 | switch (wirelessmode) { | ||
641 | case WIRELESS_MODE_B: | ||
642 | offset = -7; | ||
643 | break; | ||
644 | case WIRELESS_MODE_G: | ||
645 | case WIRELESS_MODE_N_24G: | ||
646 | offset = -8; | ||
647 | break; | ||
648 | default: | ||
649 | offset = -8; | ||
650 | break; | ||
651 | } | ||
652 | |||
653 | if ((power_indbm - offset) > 0) | ||
654 | txpwridx = (u8) ((power_indbm - offset) * 2); | ||
655 | else | ||
656 | txpwridx = 0; | ||
657 | |||
658 | if (txpwridx > MAX_TXPWR_IDX_NMODE_92S) | ||
659 | txpwridx = MAX_TXPWR_IDX_NMODE_92S; | ||
660 | |||
661 | return txpwridx; | ||
662 | } | ||
663 | EXPORT_SYMBOL(_rtl92c_phy_dbm_to_txpwr_Idx); | ||
664 | |||
665 | long _rtl92c_phy_txpwr_idx_to_dbm(struct ieee80211_hw *hw, | ||
666 | enum wireless_mode wirelessmode, | ||
667 | u8 txpwridx) | ||
668 | { | ||
669 | long offset; | ||
670 | long pwrout_dbm; | ||
671 | |||
672 | switch (wirelessmode) { | ||
673 | case WIRELESS_MODE_B: | ||
674 | offset = -7; | ||
675 | break; | ||
676 | case WIRELESS_MODE_G: | ||
677 | case WIRELESS_MODE_N_24G: | ||
678 | offset = -8; | ||
679 | break; | ||
680 | default: | ||
681 | offset = -8; | ||
682 | break; | ||
683 | } | ||
684 | pwrout_dbm = txpwridx / 2 + offset; | ||
685 | return pwrout_dbm; | ||
686 | } | ||
687 | EXPORT_SYMBOL(_rtl92c_phy_txpwr_idx_to_dbm); | ||
688 | |||
689 | void rtl92c_phy_scan_operation_backup(struct ieee80211_hw *hw, u8 operation) | ||
690 | { | ||
691 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
692 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
693 | enum io_type iotype; | ||
694 | |||
695 | if (!is_hal_stop(rtlhal)) { | ||
696 | switch (operation) { | ||
697 | case SCAN_OPT_BACKUP: | ||
698 | iotype = IO_CMD_PAUSE_DM_BY_SCAN; | ||
699 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
700 | HW_VAR_IO_CMD, | ||
701 | (u8 *)&iotype); | ||
702 | |||
703 | break; | ||
704 | case SCAN_OPT_RESTORE: | ||
705 | iotype = IO_CMD_RESUME_DM_BY_SCAN; | ||
706 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
707 | HW_VAR_IO_CMD, | ||
708 | (u8 *)&iotype); | ||
709 | break; | ||
710 | default: | ||
711 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
712 | ("Unknown Scan Backup operation.\n")); | ||
713 | break; | ||
714 | } | ||
715 | } | ||
716 | } | ||
717 | EXPORT_SYMBOL(rtl92c_phy_scan_operation_backup); | ||
718 | |||
719 | void rtl92c_phy_set_bw_mode(struct ieee80211_hw *hw, | ||
720 | enum nl80211_channel_type ch_type) | ||
721 | { | ||
722 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
723 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
724 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
725 | u8 tmp_bw = rtlphy->current_chan_bw; | ||
726 | |||
727 | if (rtlphy->set_bwmode_inprogress) | ||
728 | return; | ||
729 | rtlphy->set_bwmode_inprogress = true; | ||
730 | if ((!is_hal_stop(rtlhal)) && !(RT_CANNOT_IO(hw))) { | ||
731 | rtlpriv->cfg->ops->phy_set_bw_mode_callback(hw); | ||
732 | } else { | ||
733 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
734 | ("FALSE driver sleep or unload\n")); | ||
735 | rtlphy->set_bwmode_inprogress = false; | ||
736 | rtlphy->current_chan_bw = tmp_bw; | ||
737 | } | ||
738 | } | ||
739 | EXPORT_SYMBOL(rtl92c_phy_set_bw_mode); | ||
740 | |||
741 | void rtl92c_phy_sw_chnl_callback(struct ieee80211_hw *hw) | ||
742 | { | ||
743 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
744 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
745 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
746 | u32 delay; | ||
747 | |||
748 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, | ||
749 | ("switch to channel%d\n", rtlphy->current_channel)); | ||
750 | if (is_hal_stop(rtlhal)) | ||
751 | return; | ||
752 | do { | ||
753 | if (!rtlphy->sw_chnl_inprogress) | ||
754 | break; | ||
755 | if (!_rtl92c_phy_sw_chnl_step_by_step | ||
756 | (hw, rtlphy->current_channel, &rtlphy->sw_chnl_stage, | ||
757 | &rtlphy->sw_chnl_step, &delay)) { | ||
758 | if (delay > 0) | ||
759 | mdelay(delay); | ||
760 | else | ||
761 | continue; | ||
762 | } else { | ||
763 | rtlphy->sw_chnl_inprogress = false; | ||
764 | } | ||
765 | break; | ||
766 | } while (true); | ||
767 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n")); | ||
768 | } | ||
769 | EXPORT_SYMBOL(rtl92c_phy_sw_chnl_callback); | ||
770 | |||
771 | u8 rtl92c_phy_sw_chnl(struct ieee80211_hw *hw) | ||
772 | { | ||
773 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
774 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
775 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
776 | |||
777 | if (rtlphy->sw_chnl_inprogress) | ||
778 | return 0; | ||
779 | if (rtlphy->set_bwmode_inprogress) | ||
780 | return 0; | ||
781 | RT_ASSERT((rtlphy->current_channel <= 14), | ||
782 | ("WIRELESS_MODE_G but channel>14")); | ||
783 | rtlphy->sw_chnl_inprogress = true; | ||
784 | rtlphy->sw_chnl_stage = 0; | ||
785 | rtlphy->sw_chnl_step = 0; | ||
786 | if (!(is_hal_stop(rtlhal)) && !(RT_CANNOT_IO(hw))) { | ||
787 | rtl92c_phy_sw_chnl_callback(hw); | ||
788 | RT_TRACE(rtlpriv, COMP_CHAN, DBG_LOUD, | ||
789 | ("sw_chnl_inprogress false schdule workitem\n")); | ||
790 | rtlphy->sw_chnl_inprogress = false; | ||
791 | } else { | ||
792 | RT_TRACE(rtlpriv, COMP_CHAN, DBG_LOUD, | ||
793 | ("sw_chnl_inprogress false driver sleep or" | ||
794 | " unload\n")); | ||
795 | rtlphy->sw_chnl_inprogress = false; | ||
796 | } | ||
797 | return 1; | ||
798 | } | ||
799 | EXPORT_SYMBOL(rtl92c_phy_sw_chnl); | ||
800 | |||
801 | static bool _rtl92c_phy_set_sw_chnl_cmdarray(struct swchnlcmd *cmdtable, | ||
802 | u32 cmdtableidx, u32 cmdtablesz, | ||
803 | enum swchnlcmd_id cmdid, | ||
804 | u32 para1, u32 para2, u32 msdelay) | ||
805 | { | ||
806 | struct swchnlcmd *pcmd; | ||
807 | |||
808 | if (cmdtable == NULL) { | ||
809 | RT_ASSERT(false, ("cmdtable cannot be NULL.\n")); | ||
810 | return false; | ||
811 | } | ||
812 | |||
813 | if (cmdtableidx >= cmdtablesz) | ||
814 | return false; | ||
815 | |||
816 | pcmd = cmdtable + cmdtableidx; | ||
817 | pcmd->cmdid = cmdid; | ||
818 | pcmd->para1 = para1; | ||
819 | pcmd->para2 = para2; | ||
820 | pcmd->msdelay = msdelay; | ||
821 | return true; | ||
822 | } | ||
823 | |||
824 | bool _rtl92c_phy_sw_chnl_step_by_step(struct ieee80211_hw *hw, | ||
825 | u8 channel, u8 *stage, u8 *step, | ||
826 | u32 *delay) | ||
827 | { | ||
828 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
829 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
830 | struct swchnlcmd precommoncmd[MAX_PRECMD_CNT]; | ||
831 | u32 precommoncmdcnt; | ||
832 | struct swchnlcmd postcommoncmd[MAX_POSTCMD_CNT]; | ||
833 | u32 postcommoncmdcnt; | ||
834 | struct swchnlcmd rfdependcmd[MAX_RFDEPENDCMD_CNT]; | ||
835 | u32 rfdependcmdcnt; | ||
836 | struct swchnlcmd *currentcmd = NULL; | ||
837 | u8 rfpath; | ||
838 | u8 num_total_rfpath = rtlphy->num_total_rfpath; | ||
839 | |||
840 | precommoncmdcnt = 0; | ||
841 | _rtl92c_phy_set_sw_chnl_cmdarray(precommoncmd, precommoncmdcnt++, | ||
842 | MAX_PRECMD_CNT, | ||
843 | CMDID_SET_TXPOWEROWER_LEVEL, 0, 0, 0); | ||
844 | _rtl92c_phy_set_sw_chnl_cmdarray(precommoncmd, precommoncmdcnt++, | ||
845 | MAX_PRECMD_CNT, CMDID_END, 0, 0, 0); | ||
846 | |||
847 | postcommoncmdcnt = 0; | ||
848 | |||
849 | _rtl92c_phy_set_sw_chnl_cmdarray(postcommoncmd, postcommoncmdcnt++, | ||
850 | MAX_POSTCMD_CNT, CMDID_END, 0, 0, 0); | ||
851 | |||
852 | rfdependcmdcnt = 0; | ||
853 | |||
854 | RT_ASSERT((channel >= 1 && channel <= 14), | ||
855 | ("illegal channel for Zebra: %d\n", channel)); | ||
856 | |||
857 | _rtl92c_phy_set_sw_chnl_cmdarray(rfdependcmd, rfdependcmdcnt++, | ||
858 | MAX_RFDEPENDCMD_CNT, CMDID_RF_WRITEREG, | ||
859 | RF_CHNLBW, channel, 10); | ||
860 | |||
861 | _rtl92c_phy_set_sw_chnl_cmdarray(rfdependcmd, rfdependcmdcnt++, | ||
862 | MAX_RFDEPENDCMD_CNT, CMDID_END, 0, 0, | ||
863 | 0); | ||
864 | |||
865 | do { | ||
866 | switch (*stage) { | ||
867 | case 0: | ||
868 | currentcmd = &precommoncmd[*step]; | ||
869 | break; | ||
870 | case 1: | ||
871 | currentcmd = &rfdependcmd[*step]; | ||
872 | break; | ||
873 | case 2: | ||
874 | currentcmd = &postcommoncmd[*step]; | ||
875 | break; | ||
876 | } | ||
877 | |||
878 | if (currentcmd->cmdid == CMDID_END) { | ||
879 | if ((*stage) == 2) { | ||
880 | return true; | ||
881 | } else { | ||
882 | (*stage)++; | ||
883 | (*step) = 0; | ||
884 | continue; | ||
885 | } | ||
886 | } | ||
887 | |||
888 | switch (currentcmd->cmdid) { | ||
889 | case CMDID_SET_TXPOWEROWER_LEVEL: | ||
890 | rtl92c_phy_set_txpower_level(hw, channel); | ||
891 | break; | ||
892 | case CMDID_WRITEPORT_ULONG: | ||
893 | rtl_write_dword(rtlpriv, currentcmd->para1, | ||
894 | currentcmd->para2); | ||
895 | break; | ||
896 | case CMDID_WRITEPORT_USHORT: | ||
897 | rtl_write_word(rtlpriv, currentcmd->para1, | ||
898 | (u16) currentcmd->para2); | ||
899 | break; | ||
900 | case CMDID_WRITEPORT_UCHAR: | ||
901 | rtl_write_byte(rtlpriv, currentcmd->para1, | ||
902 | (u8) currentcmd->para2); | ||
903 | break; | ||
904 | case CMDID_RF_WRITEREG: | ||
905 | for (rfpath = 0; rfpath < num_total_rfpath; rfpath++) { | ||
906 | rtlphy->rfreg_chnlval[rfpath] = | ||
907 | ((rtlphy->rfreg_chnlval[rfpath] & | ||
908 | 0xfffffc00) | currentcmd->para2); | ||
909 | |||
910 | rtl_set_rfreg(hw, (enum radio_path)rfpath, | ||
911 | currentcmd->para1, | ||
912 | RFREG_OFFSET_MASK, | ||
913 | rtlphy->rfreg_chnlval[rfpath]); | ||
914 | } | ||
915 | break; | ||
916 | default: | ||
917 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
918 | ("switch case not process\n")); | ||
919 | break; | ||
920 | } | ||
921 | |||
922 | break; | ||
923 | } while (true); | ||
924 | |||
925 | (*delay) = currentcmd->msdelay; | ||
926 | (*step)++; | ||
927 | return false; | ||
928 | } | ||
929 | |||
930 | bool rtl8192_phy_check_is_legal_rfpath(struct ieee80211_hw *hw, u32 rfpath) | ||
931 | { | ||
932 | return true; | ||
933 | } | ||
934 | EXPORT_SYMBOL(rtl8192_phy_check_is_legal_rfpath); | ||
935 | |||
936 | static u8 _rtl92c_phy_path_a_iqk(struct ieee80211_hw *hw, bool config_pathb) | ||
937 | { | ||
938 | u32 reg_eac, reg_e94, reg_e9c, reg_ea4; | ||
939 | u8 result = 0x00; | ||
940 | |||
941 | rtl_set_bbreg(hw, 0xe30, MASKDWORD, 0x10008c1f); | ||
942 | rtl_set_bbreg(hw, 0xe34, MASKDWORD, 0x10008c1f); | ||
943 | rtl_set_bbreg(hw, 0xe38, MASKDWORD, 0x82140102); | ||
944 | rtl_set_bbreg(hw, 0xe3c, MASKDWORD, | ||
945 | config_pathb ? 0x28160202 : 0x28160502); | ||
946 | |||
947 | if (config_pathb) { | ||
948 | rtl_set_bbreg(hw, 0xe50, MASKDWORD, 0x10008c22); | ||
949 | rtl_set_bbreg(hw, 0xe54, MASKDWORD, 0x10008c22); | ||
950 | rtl_set_bbreg(hw, 0xe58, MASKDWORD, 0x82140102); | ||
951 | rtl_set_bbreg(hw, 0xe5c, MASKDWORD, 0x28160202); | ||
952 | } | ||
953 | |||
954 | rtl_set_bbreg(hw, 0xe4c, MASKDWORD, 0x001028d1); | ||
955 | rtl_set_bbreg(hw, 0xe48, MASKDWORD, 0xf9000000); | ||
956 | rtl_set_bbreg(hw, 0xe48, MASKDWORD, 0xf8000000); | ||
957 | |||
958 | mdelay(IQK_DELAY_TIME); | ||
959 | |||
960 | reg_eac = rtl_get_bbreg(hw, 0xeac, MASKDWORD); | ||
961 | reg_e94 = rtl_get_bbreg(hw, 0xe94, MASKDWORD); | ||
962 | reg_e9c = rtl_get_bbreg(hw, 0xe9c, MASKDWORD); | ||
963 | reg_ea4 = rtl_get_bbreg(hw, 0xea4, MASKDWORD); | ||
964 | |||
965 | if (!(reg_eac & BIT(28)) && | ||
966 | (((reg_e94 & 0x03FF0000) >> 16) != 0x142) && | ||
967 | (((reg_e9c & 0x03FF0000) >> 16) != 0x42)) | ||
968 | result |= 0x01; | ||
969 | else | ||
970 | return result; | ||
971 | |||
972 | if (!(reg_eac & BIT(27)) && | ||
973 | (((reg_ea4 & 0x03FF0000) >> 16) != 0x132) && | ||
974 | (((reg_eac & 0x03FF0000) >> 16) != 0x36)) | ||
975 | result |= 0x02; | ||
976 | return result; | ||
977 | } | ||
978 | |||
979 | static u8 _rtl92c_phy_path_b_iqk(struct ieee80211_hw *hw) | ||
980 | { | ||
981 | u32 reg_eac, reg_eb4, reg_ebc, reg_ec4, reg_ecc; | ||
982 | u8 result = 0x00; | ||
983 | |||
984 | rtl_set_bbreg(hw, 0xe60, MASKDWORD, 0x00000002); | ||
985 | rtl_set_bbreg(hw, 0xe60, MASKDWORD, 0x00000000); | ||
986 | mdelay(IQK_DELAY_TIME); | ||
987 | reg_eac = rtl_get_bbreg(hw, 0xeac, MASKDWORD); | ||
988 | reg_eb4 = rtl_get_bbreg(hw, 0xeb4, MASKDWORD); | ||
989 | reg_ebc = rtl_get_bbreg(hw, 0xebc, MASKDWORD); | ||
990 | reg_ec4 = rtl_get_bbreg(hw, 0xec4, MASKDWORD); | ||
991 | reg_ecc = rtl_get_bbreg(hw, 0xecc, MASKDWORD); | ||
992 | |||
993 | if (!(reg_eac & BIT(31)) && | ||
994 | (((reg_eb4 & 0x03FF0000) >> 16) != 0x142) && | ||
995 | (((reg_ebc & 0x03FF0000) >> 16) != 0x42)) | ||
996 | result |= 0x01; | ||
997 | else | ||
998 | return result; | ||
999 | if (!(reg_eac & BIT(30)) && | ||
1000 | (((reg_ec4 & 0x03FF0000) >> 16) != 0x132) && | ||
1001 | (((reg_ecc & 0x03FF0000) >> 16) != 0x36)) | ||
1002 | result |= 0x02; | ||
1003 | return result; | ||
1004 | } | ||
1005 | |||
1006 | static void _rtl92c_phy_path_a_fill_iqk_matrix(struct ieee80211_hw *hw, | ||
1007 | bool iqk_ok, long result[][8], | ||
1008 | u8 final_candidate, bool btxonly) | ||
1009 | { | ||
1010 | u32 oldval_0, x, tx0_a, reg; | ||
1011 | long y, tx0_c; | ||
1012 | |||
1013 | if (final_candidate == 0xFF) { | ||
1014 | return; | ||
1015 | } else if (iqk_ok) { | ||
1016 | oldval_0 = (rtl_get_bbreg(hw, ROFDM0_XATXIQIMBALANCE, | ||
1017 | MASKDWORD) >> 22) & 0x3FF; | ||
1018 | x = result[final_candidate][0]; | ||
1019 | if ((x & 0x00000200) != 0) | ||
1020 | x = x | 0xFFFFFC00; | ||
1021 | tx0_a = (x * oldval_0) >> 8; | ||
1022 | rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, 0x3FF, tx0_a); | ||
1023 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(31), | ||
1024 | ((x * oldval_0 >> 7) & 0x1)); | ||
1025 | y = result[final_candidate][1]; | ||
1026 | if ((y & 0x00000200) != 0) | ||
1027 | y = y | 0xFFFFFC00; | ||
1028 | tx0_c = (y * oldval_0) >> 8; | ||
1029 | rtl_set_bbreg(hw, ROFDM0_XCTXAFE, 0xF0000000, | ||
1030 | ((tx0_c & 0x3C0) >> 6)); | ||
1031 | rtl_set_bbreg(hw, ROFDM0_XATXIQIMBALANCE, 0x003F0000, | ||
1032 | (tx0_c & 0x3F)); | ||
1033 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(29), | ||
1034 | ((y * oldval_0 >> 7) & 0x1)); | ||
1035 | if (btxonly) | ||
1036 | return; | ||
1037 | reg = result[final_candidate][2]; | ||
1038 | rtl_set_bbreg(hw, ROFDM0_XARXIQIMBALANCE, 0x3FF, reg); | ||
1039 | reg = result[final_candidate][3] & 0x3F; | ||
1040 | rtl_set_bbreg(hw, ROFDM0_XARXIQIMBALANCE, 0xFC00, reg); | ||
1041 | reg = (result[final_candidate][3] >> 6) & 0xF; | ||
1042 | rtl_set_bbreg(hw, 0xca0, 0xF0000000, reg); | ||
1043 | } | ||
1044 | } | ||
1045 | |||
1046 | static void _rtl92c_phy_path_b_fill_iqk_matrix(struct ieee80211_hw *hw, | ||
1047 | bool iqk_ok, long result[][8], | ||
1048 | u8 final_candidate, bool btxonly) | ||
1049 | { | ||
1050 | u32 oldval_1, x, tx1_a, reg; | ||
1051 | long y, tx1_c; | ||
1052 | |||
1053 | if (final_candidate == 0xFF) { | ||
1054 | return; | ||
1055 | } else if (iqk_ok) { | ||
1056 | oldval_1 = (rtl_get_bbreg(hw, ROFDM0_XBTXIQIMBALANCE, | ||
1057 | MASKDWORD) >> 22) & 0x3FF; | ||
1058 | x = result[final_candidate][4]; | ||
1059 | if ((x & 0x00000200) != 0) | ||
1060 | x = x | 0xFFFFFC00; | ||
1061 | tx1_a = (x * oldval_1) >> 8; | ||
1062 | rtl_set_bbreg(hw, ROFDM0_XBTXIQIMBALANCE, 0x3FF, tx1_a); | ||
1063 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(27), | ||
1064 | ((x * oldval_1 >> 7) & 0x1)); | ||
1065 | y = result[final_candidate][5]; | ||
1066 | if ((y & 0x00000200) != 0) | ||
1067 | y = y | 0xFFFFFC00; | ||
1068 | tx1_c = (y * oldval_1) >> 8; | ||
1069 | rtl_set_bbreg(hw, ROFDM0_XDTXAFE, 0xF0000000, | ||
1070 | ((tx1_c & 0x3C0) >> 6)); | ||
1071 | rtl_set_bbreg(hw, ROFDM0_XBTXIQIMBALANCE, 0x003F0000, | ||
1072 | (tx1_c & 0x3F)); | ||
1073 | rtl_set_bbreg(hw, ROFDM0_ECCATHRESHOLD, BIT(25), | ||
1074 | ((y * oldval_1 >> 7) & 0x1)); | ||
1075 | if (btxonly) | ||
1076 | return; | ||
1077 | reg = result[final_candidate][6]; | ||
1078 | rtl_set_bbreg(hw, ROFDM0_XBRXIQIMBALANCE, 0x3FF, reg); | ||
1079 | reg = result[final_candidate][7] & 0x3F; | ||
1080 | rtl_set_bbreg(hw, ROFDM0_XBRXIQIMBALANCE, 0xFC00, reg); | ||
1081 | reg = (result[final_candidate][7] >> 6) & 0xF; | ||
1082 | rtl_set_bbreg(hw, ROFDM0_AGCRSSITABLE, 0x0000F000, reg); | ||
1083 | } | ||
1084 | } | ||
1085 | |||
1086 | static void _rtl92c_phy_save_adda_registers(struct ieee80211_hw *hw, | ||
1087 | u32 *addareg, u32 *addabackup, | ||
1088 | u32 registernum) | ||
1089 | { | ||
1090 | u32 i; | ||
1091 | |||
1092 | for (i = 0; i < registernum; i++) | ||
1093 | addabackup[i] = rtl_get_bbreg(hw, addareg[i], MASKDWORD); | ||
1094 | } | ||
1095 | |||
1096 | static void _rtl92c_phy_save_mac_registers(struct ieee80211_hw *hw, | ||
1097 | u32 *macreg, u32 *macbackup) | ||
1098 | { | ||
1099 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1100 | u32 i; | ||
1101 | |||
1102 | for (i = 0; i < (IQK_MAC_REG_NUM - 1); i++) | ||
1103 | macbackup[i] = rtl_read_byte(rtlpriv, macreg[i]); | ||
1104 | macbackup[i] = rtl_read_dword(rtlpriv, macreg[i]); | ||
1105 | } | ||
1106 | |||
1107 | static void _rtl92c_phy_reload_adda_registers(struct ieee80211_hw *hw, | ||
1108 | u32 *addareg, u32 *addabackup, | ||
1109 | u32 regiesternum) | ||
1110 | { | ||
1111 | u32 i; | ||
1112 | |||
1113 | for (i = 0; i < regiesternum; i++) | ||
1114 | rtl_set_bbreg(hw, addareg[i], MASKDWORD, addabackup[i]); | ||
1115 | } | ||
1116 | |||
1117 | static void _rtl92c_phy_reload_mac_registers(struct ieee80211_hw *hw, | ||
1118 | u32 *macreg, u32 *macbackup) | ||
1119 | { | ||
1120 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1121 | u32 i; | ||
1122 | |||
1123 | for (i = 0; i < (IQK_MAC_REG_NUM - 1); i++) | ||
1124 | rtl_write_byte(rtlpriv, macreg[i], (u8) macbackup[i]); | ||
1125 | rtl_write_dword(rtlpriv, macreg[i], macbackup[i]); | ||
1126 | } | ||
1127 | |||
1128 | static void _rtl92c_phy_path_adda_on(struct ieee80211_hw *hw, | ||
1129 | u32 *addareg, bool is_patha_on, bool is2t) | ||
1130 | { | ||
1131 | u32 pathOn; | ||
1132 | u32 i; | ||
1133 | |||
1134 | pathOn = is_patha_on ? 0x04db25a4 : 0x0b1b25a4; | ||
1135 | if (false == is2t) { | ||
1136 | pathOn = 0x0bdb25a0; | ||
1137 | rtl_set_bbreg(hw, addareg[0], MASKDWORD, 0x0b1b25a0); | ||
1138 | } else { | ||
1139 | rtl_set_bbreg(hw, addareg[0], MASKDWORD, pathOn); | ||
1140 | } | ||
1141 | |||
1142 | for (i = 1; i < IQK_ADDA_REG_NUM; i++) | ||
1143 | rtl_set_bbreg(hw, addareg[i], MASKDWORD, pathOn); | ||
1144 | } | ||
1145 | |||
1146 | static void _rtl92c_phy_mac_setting_calibration(struct ieee80211_hw *hw, | ||
1147 | u32 *macreg, u32 *macbackup) | ||
1148 | { | ||
1149 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1150 | u32 i; | ||
1151 | |||
1152 | rtl_write_byte(rtlpriv, macreg[0], 0x3F); | ||
1153 | |||
1154 | for (i = 1; i < (IQK_MAC_REG_NUM - 1); i++) | ||
1155 | rtl_write_byte(rtlpriv, macreg[i], | ||
1156 | (u8) (macbackup[i] & (~BIT(3)))); | ||
1157 | rtl_write_byte(rtlpriv, macreg[i], (u8) (macbackup[i] & (~BIT(5)))); | ||
1158 | } | ||
1159 | |||
1160 | static void _rtl92c_phy_path_a_standby(struct ieee80211_hw *hw) | ||
1161 | { | ||
1162 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x0); | ||
1163 | rtl_set_bbreg(hw, 0x840, MASKDWORD, 0x00010000); | ||
1164 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x80800000); | ||
1165 | } | ||
1166 | |||
1167 | static void _rtl92c_phy_pi_mode_switch(struct ieee80211_hw *hw, bool pi_mode) | ||
1168 | { | ||
1169 | u32 mode; | ||
1170 | |||
1171 | mode = pi_mode ? 0x01000100 : 0x01000000; | ||
1172 | rtl_set_bbreg(hw, 0x820, MASKDWORD, mode); | ||
1173 | rtl_set_bbreg(hw, 0x828, MASKDWORD, mode); | ||
1174 | } | ||
1175 | |||
1176 | static bool _rtl92c_phy_simularity_compare(struct ieee80211_hw *hw, | ||
1177 | long result[][8], u8 c1, u8 c2) | ||
1178 | { | ||
1179 | u32 i, j, diff, simularity_bitmap, bound; | ||
1180 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1181 | |||
1182 | u8 final_candidate[2] = { 0xFF, 0xFF }; | ||
1183 | bool bresult = true, is2t = IS_92C_SERIAL(rtlhal->version); | ||
1184 | |||
1185 | if (is2t) | ||
1186 | bound = 8; | ||
1187 | else | ||
1188 | bound = 4; | ||
1189 | |||
1190 | simularity_bitmap = 0; | ||
1191 | |||
1192 | for (i = 0; i < bound; i++) { | ||
1193 | diff = (result[c1][i] > result[c2][i]) ? | ||
1194 | (result[c1][i] - result[c2][i]) : | ||
1195 | (result[c2][i] - result[c1][i]); | ||
1196 | |||
1197 | if (diff > MAX_TOLERANCE) { | ||
1198 | if ((i == 2 || i == 6) && !simularity_bitmap) { | ||
1199 | if (result[c1][i] + result[c1][i + 1] == 0) | ||
1200 | final_candidate[(i / 4)] = c2; | ||
1201 | else if (result[c2][i] + result[c2][i + 1] == 0) | ||
1202 | final_candidate[(i / 4)] = c1; | ||
1203 | else | ||
1204 | simularity_bitmap = simularity_bitmap | | ||
1205 | (1 << i); | ||
1206 | } else | ||
1207 | simularity_bitmap = | ||
1208 | simularity_bitmap | (1 << i); | ||
1209 | } | ||
1210 | } | ||
1211 | |||
1212 | if (simularity_bitmap == 0) { | ||
1213 | for (i = 0; i < (bound / 4); i++) { | ||
1214 | if (final_candidate[i] != 0xFF) { | ||
1215 | for (j = i * 4; j < (i + 1) * 4 - 2; j++) | ||
1216 | result[3][j] = | ||
1217 | result[final_candidate[i]][j]; | ||
1218 | bresult = false; | ||
1219 | } | ||
1220 | } | ||
1221 | return bresult; | ||
1222 | } else if (!(simularity_bitmap & 0x0F)) { | ||
1223 | for (i = 0; i < 4; i++) | ||
1224 | result[3][i] = result[c1][i]; | ||
1225 | return false; | ||
1226 | } else if (!(simularity_bitmap & 0xF0) && is2t) { | ||
1227 | for (i = 4; i < 8; i++) | ||
1228 | result[3][i] = result[c1][i]; | ||
1229 | return false; | ||
1230 | } else { | ||
1231 | return false; | ||
1232 | } | ||
1233 | |||
1234 | } | ||
1235 | |||
1236 | static void _rtl92c_phy_iq_calibrate(struct ieee80211_hw *hw, | ||
1237 | long result[][8], u8 t, bool is2t) | ||
1238 | { | ||
1239 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1240 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1241 | u32 i; | ||
1242 | u8 patha_ok, pathb_ok; | ||
1243 | u32 adda_reg[IQK_ADDA_REG_NUM] = { | ||
1244 | 0x85c, 0xe6c, 0xe70, 0xe74, | ||
1245 | 0xe78, 0xe7c, 0xe80, 0xe84, | ||
1246 | 0xe88, 0xe8c, 0xed0, 0xed4, | ||
1247 | 0xed8, 0xedc, 0xee0, 0xeec | ||
1248 | }; | ||
1249 | |||
1250 | u32 iqk_mac_reg[IQK_MAC_REG_NUM] = { | ||
1251 | 0x522, 0x550, 0x551, 0x040 | ||
1252 | }; | ||
1253 | |||
1254 | const u32 retrycount = 2; | ||
1255 | |||
1256 | u32 bbvalue; | ||
1257 | |||
1258 | if (t == 0) { | ||
1259 | bbvalue = rtl_get_bbreg(hw, 0x800, MASKDWORD); | ||
1260 | |||
1261 | _rtl92c_phy_save_adda_registers(hw, adda_reg, | ||
1262 | rtlphy->adda_backup, 16); | ||
1263 | _rtl92c_phy_save_mac_registers(hw, iqk_mac_reg, | ||
1264 | rtlphy->iqk_mac_backup); | ||
1265 | } | ||
1266 | _rtl92c_phy_path_adda_on(hw, adda_reg, true, is2t); | ||
1267 | if (t == 0) { | ||
1268 | rtlphy->rfpi_enable = (u8) rtl_get_bbreg(hw, | ||
1269 | RFPGA0_XA_HSSIPARAMETER1, | ||
1270 | BIT(8)); | ||
1271 | } | ||
1272 | |||
1273 | if (!rtlphy->rfpi_enable) | ||
1274 | _rtl92c_phy_pi_mode_switch(hw, true); | ||
1275 | if (t == 0) { | ||
1276 | rtlphy->reg_c04 = rtl_get_bbreg(hw, 0xc04, MASKDWORD); | ||
1277 | rtlphy->reg_c08 = rtl_get_bbreg(hw, 0xc08, MASKDWORD); | ||
1278 | rtlphy->reg_874 = rtl_get_bbreg(hw, 0x874, MASKDWORD); | ||
1279 | } | ||
1280 | rtl_set_bbreg(hw, 0xc04, MASKDWORD, 0x03a05600); | ||
1281 | rtl_set_bbreg(hw, 0xc08, MASKDWORD, 0x000800e4); | ||
1282 | rtl_set_bbreg(hw, 0x874, MASKDWORD, 0x22204000); | ||
1283 | if (is2t) { | ||
1284 | rtl_set_bbreg(hw, 0x840, MASKDWORD, 0x00010000); | ||
1285 | rtl_set_bbreg(hw, 0x844, MASKDWORD, 0x00010000); | ||
1286 | } | ||
1287 | _rtl92c_phy_mac_setting_calibration(hw, iqk_mac_reg, | ||
1288 | rtlphy->iqk_mac_backup); | ||
1289 | rtl_set_bbreg(hw, 0xb68, MASKDWORD, 0x00080000); | ||
1290 | if (is2t) | ||
1291 | rtl_set_bbreg(hw, 0xb6c, MASKDWORD, 0x00080000); | ||
1292 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x80800000); | ||
1293 | rtl_set_bbreg(hw, 0xe40, MASKDWORD, 0x01007c00); | ||
1294 | rtl_set_bbreg(hw, 0xe44, MASKDWORD, 0x01004800); | ||
1295 | for (i = 0; i < retrycount; i++) { | ||
1296 | patha_ok = _rtl92c_phy_path_a_iqk(hw, is2t); | ||
1297 | if (patha_ok == 0x03) { | ||
1298 | result[t][0] = (rtl_get_bbreg(hw, 0xe94, MASKDWORD) & | ||
1299 | 0x3FF0000) >> 16; | ||
1300 | result[t][1] = (rtl_get_bbreg(hw, 0xe9c, MASKDWORD) & | ||
1301 | 0x3FF0000) >> 16; | ||
1302 | result[t][2] = (rtl_get_bbreg(hw, 0xea4, MASKDWORD) & | ||
1303 | 0x3FF0000) >> 16; | ||
1304 | result[t][3] = (rtl_get_bbreg(hw, 0xeac, MASKDWORD) & | ||
1305 | 0x3FF0000) >> 16; | ||
1306 | break; | ||
1307 | } else if (i == (retrycount - 1) && patha_ok == 0x01) | ||
1308 | |||
1309 | result[t][0] = (rtl_get_bbreg(hw, 0xe94, | ||
1310 | MASKDWORD) & 0x3FF0000) >> | ||
1311 | 16; | ||
1312 | result[t][1] = | ||
1313 | (rtl_get_bbreg(hw, 0xe9c, MASKDWORD) & 0x3FF0000) >> 16; | ||
1314 | |||
1315 | } | ||
1316 | |||
1317 | if (is2t) { | ||
1318 | _rtl92c_phy_path_a_standby(hw); | ||
1319 | _rtl92c_phy_path_adda_on(hw, adda_reg, false, is2t); | ||
1320 | for (i = 0; i < retrycount; i++) { | ||
1321 | pathb_ok = _rtl92c_phy_path_b_iqk(hw); | ||
1322 | if (pathb_ok == 0x03) { | ||
1323 | result[t][4] = (rtl_get_bbreg(hw, | ||
1324 | 0xeb4, | ||
1325 | MASKDWORD) & | ||
1326 | 0x3FF0000) >> 16; | ||
1327 | result[t][5] = | ||
1328 | (rtl_get_bbreg(hw, 0xebc, MASKDWORD) & | ||
1329 | 0x3FF0000) >> 16; | ||
1330 | result[t][6] = | ||
1331 | (rtl_get_bbreg(hw, 0xec4, MASKDWORD) & | ||
1332 | 0x3FF0000) >> 16; | ||
1333 | result[t][7] = | ||
1334 | (rtl_get_bbreg(hw, 0xecc, MASKDWORD) & | ||
1335 | 0x3FF0000) >> 16; | ||
1336 | break; | ||
1337 | } else if (i == (retrycount - 1) && pathb_ok == 0x01) { | ||
1338 | result[t][4] = (rtl_get_bbreg(hw, | ||
1339 | 0xeb4, | ||
1340 | MASKDWORD) & | ||
1341 | 0x3FF0000) >> 16; | ||
1342 | } | ||
1343 | result[t][5] = (rtl_get_bbreg(hw, 0xebc, MASKDWORD) & | ||
1344 | 0x3FF0000) >> 16; | ||
1345 | } | ||
1346 | } | ||
1347 | rtl_set_bbreg(hw, 0xc04, MASKDWORD, rtlphy->reg_c04); | ||
1348 | rtl_set_bbreg(hw, 0x874, MASKDWORD, rtlphy->reg_874); | ||
1349 | rtl_set_bbreg(hw, 0xc08, MASKDWORD, rtlphy->reg_c08); | ||
1350 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0); | ||
1351 | rtl_set_bbreg(hw, 0x840, MASKDWORD, 0x00032ed3); | ||
1352 | if (is2t) | ||
1353 | rtl_set_bbreg(hw, 0x844, MASKDWORD, 0x00032ed3); | ||
1354 | if (t != 0) { | ||
1355 | if (!rtlphy->rfpi_enable) | ||
1356 | _rtl92c_phy_pi_mode_switch(hw, false); | ||
1357 | _rtl92c_phy_reload_adda_registers(hw, adda_reg, | ||
1358 | rtlphy->adda_backup, 16); | ||
1359 | _rtl92c_phy_reload_mac_registers(hw, iqk_mac_reg, | ||
1360 | rtlphy->iqk_mac_backup); | ||
1361 | } | ||
1362 | } | ||
1363 | |||
1364 | static void _rtl92c_phy_ap_calibrate(struct ieee80211_hw *hw, | ||
1365 | char delta, bool is2t) | ||
1366 | { | ||
1367 | #if 0 /* This routine is deliberately dummied out for later fixes */ | ||
1368 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1369 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1370 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1371 | |||
1372 | u32 reg_d[PATH_NUM]; | ||
1373 | u32 tmpreg, index, offset, path, i, pathbound = PATH_NUM, apkbound; | ||
1374 | |||
1375 | u32 bb_backup[APK_BB_REG_NUM]; | ||
1376 | u32 bb_reg[APK_BB_REG_NUM] = { | ||
1377 | 0x904, 0xc04, 0x800, 0xc08, 0x874 | ||
1378 | }; | ||
1379 | u32 bb_ap_mode[APK_BB_REG_NUM] = { | ||
1380 | 0x00000020, 0x00a05430, 0x02040000, | ||
1381 | 0x000800e4, 0x00204000 | ||
1382 | }; | ||
1383 | u32 bb_normal_ap_mode[APK_BB_REG_NUM] = { | ||
1384 | 0x00000020, 0x00a05430, 0x02040000, | ||
1385 | 0x000800e4, 0x22204000 | ||
1386 | }; | ||
1387 | |||
1388 | u32 afe_backup[APK_AFE_REG_NUM]; | ||
1389 | u32 afe_reg[APK_AFE_REG_NUM] = { | ||
1390 | 0x85c, 0xe6c, 0xe70, 0xe74, 0xe78, | ||
1391 | 0xe7c, 0xe80, 0xe84, 0xe88, 0xe8c, | ||
1392 | 0xed0, 0xed4, 0xed8, 0xedc, 0xee0, | ||
1393 | 0xeec | ||
1394 | }; | ||
1395 | |||
1396 | u32 mac_backup[IQK_MAC_REG_NUM]; | ||
1397 | u32 mac_reg[IQK_MAC_REG_NUM] = { | ||
1398 | 0x522, 0x550, 0x551, 0x040 | ||
1399 | }; | ||
1400 | |||
1401 | u32 apk_rf_init_value[PATH_NUM][APK_BB_REG_NUM] = { | ||
1402 | {0x0852c, 0x1852c, 0x5852c, 0x1852c, 0x5852c}, | ||
1403 | {0x2852e, 0x0852e, 0x3852e, 0x0852e, 0x0852e} | ||
1404 | }; | ||
1405 | |||
1406 | u32 apk_normal_rf_init_value[PATH_NUM][APK_BB_REG_NUM] = { | ||
1407 | {0x0852c, 0x0a52c, 0x3a52c, 0x5a52c, 0x5a52c}, | ||
1408 | {0x0852c, 0x0a52c, 0x5a52c, 0x5a52c, 0x5a52c} | ||
1409 | }; | ||
1410 | |||
1411 | u32 apk_rf_value_0[PATH_NUM][APK_BB_REG_NUM] = { | ||
1412 | {0x52019, 0x52014, 0x52013, 0x5200f, 0x5208d}, | ||
1413 | {0x5201a, 0x52019, 0x52016, 0x52033, 0x52050} | ||
1414 | }; | ||
1415 | |||
1416 | u32 apk_normal_rf_value_0[PATH_NUM][APK_BB_REG_NUM] = { | ||
1417 | {0x52019, 0x52017, 0x52010, 0x5200d, 0x5206a}, | ||
1418 | {0x52019, 0x52017, 0x52010, 0x5200d, 0x5206a} | ||
1419 | }; | ||
1420 | |||
1421 | u32 afe_on_off[PATH_NUM] = { | ||
1422 | 0x04db25a4, 0x0b1b25a4 | ||
1423 | }; | ||
1424 | |||
1425 | const u32 apk_offset[PATH_NUM] = { 0xb68, 0xb6c }; | ||
1426 | |||
1427 | u32 apk_normal_offset[PATH_NUM] = { 0xb28, 0xb98 }; | ||
1428 | |||
1429 | u32 apk_value[PATH_NUM] = { 0x92fc0000, 0x12fc0000 }; | ||
1430 | |||
1431 | u32 apk_normal_value[PATH_NUM] = { 0x92680000, 0x12680000 }; | ||
1432 | |||
1433 | const char apk_delta_mapping[APK_BB_REG_NUM][13] = { | ||
1434 | {-4, -3, -2, -2, -1, -1, 0, 1, 2, 3, 4, 5, 6}, | ||
1435 | {-4, -3, -2, -2, -1, -1, 0, 1, 2, 3, 4, 5, 6}, | ||
1436 | {-6, -4, -2, -2, -1, -1, 0, 1, 2, 3, 4, 5, 6}, | ||
1437 | {-1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6}, | ||
1438 | {-11, -9, -7, -5, -3, -1, 0, 0, 0, 0, 0, 0, 0} | ||
1439 | }; | ||
1440 | |||
1441 | const u32 apk_normal_setting_value_1[13] = { | ||
1442 | 0x01017018, 0xf7ed8f84, 0x1b1a1816, 0x2522201e, 0x322e2b28, | ||
1443 | 0x433f3a36, 0x5b544e49, 0x7b726a62, 0xa69a8f84, 0xdfcfc0b3, | ||
1444 | 0x12680000, 0x00880000, 0x00880000 | ||
1445 | }; | ||
1446 | |||
1447 | const u32 apk_normal_setting_value_2[16] = { | ||
1448 | 0x01c7021d, 0x01670183, 0x01000123, 0x00bf00e2, 0x008d00a3, | ||
1449 | 0x0068007b, 0x004d0059, 0x003a0042, 0x002b0031, 0x001f0025, | ||
1450 | 0x0017001b, 0x00110014, 0x000c000f, 0x0009000b, 0x00070008, | ||
1451 | 0x00050006 | ||
1452 | }; | ||
1453 | |||
1454 | u32 apk_result[PATH_NUM][APK_BB_REG_NUM]; | ||
1455 | |||
1456 | long bb_offset, delta_v, delta_offset; | ||
1457 | |||
1458 | if (!is2t) | ||
1459 | pathbound = 1; | ||
1460 | |||
1461 | return; | ||
1462 | |||
1463 | for (index = 0; index < PATH_NUM; index++) { | ||
1464 | apk_offset[index] = apk_normal_offset[index]; | ||
1465 | apk_value[index] = apk_normal_value[index]; | ||
1466 | afe_on_off[index] = 0x6fdb25a4; | ||
1467 | } | ||
1468 | |||
1469 | for (index = 0; index < APK_BB_REG_NUM; index++) { | ||
1470 | for (path = 0; path < pathbound; path++) { | ||
1471 | apk_rf_init_value[path][index] = | ||
1472 | apk_normal_rf_init_value[path][index]; | ||
1473 | apk_rf_value_0[path][index] = | ||
1474 | apk_normal_rf_value_0[path][index]; | ||
1475 | } | ||
1476 | bb_ap_mode[index] = bb_normal_ap_mode[index]; | ||
1477 | |||
1478 | apkbound = 6; | ||
1479 | } | ||
1480 | |||
1481 | for (index = 0; index < APK_BB_REG_NUM; index++) { | ||
1482 | if (index == 0) | ||
1483 | continue; | ||
1484 | bb_backup[index] = rtl_get_bbreg(hw, bb_reg[index], MASKDWORD); | ||
1485 | } | ||
1486 | |||
1487 | _rtl92c_phy_save_mac_registers(hw, mac_reg, mac_backup); | ||
1488 | |||
1489 | _rtl92c_phy_save_adda_registers(hw, afe_reg, afe_backup, 16); | ||
1490 | |||
1491 | for (path = 0; path < pathbound; path++) { | ||
1492 | if (path == RF90_PATH_A) { | ||
1493 | offset = 0xb00; | ||
1494 | for (index = 0; index < 11; index++) { | ||
1495 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1496 | apk_normal_setting_value_1 | ||
1497 | [index]); | ||
1498 | |||
1499 | offset += 0x04; | ||
1500 | } | ||
1501 | |||
1502 | rtl_set_bbreg(hw, 0xb98, MASKDWORD, 0x12680000); | ||
1503 | |||
1504 | offset = 0xb68; | ||
1505 | for (; index < 13; index++) { | ||
1506 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1507 | apk_normal_setting_value_1 | ||
1508 | [index]); | ||
1509 | |||
1510 | offset += 0x04; | ||
1511 | } | ||
1512 | |||
1513 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x40000000); | ||
1514 | |||
1515 | offset = 0xb00; | ||
1516 | for (index = 0; index < 16; index++) { | ||
1517 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1518 | apk_normal_setting_value_2 | ||
1519 | [index]); | ||
1520 | |||
1521 | offset += 0x04; | ||
1522 | } | ||
1523 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x00000000); | ||
1524 | } else if (path == RF90_PATH_B) { | ||
1525 | offset = 0xb70; | ||
1526 | for (index = 0; index < 10; index++) { | ||
1527 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1528 | apk_normal_setting_value_1 | ||
1529 | [index]); | ||
1530 | |||
1531 | offset += 0x04; | ||
1532 | } | ||
1533 | rtl_set_bbreg(hw, 0xb28, MASKDWORD, 0x12680000); | ||
1534 | rtl_set_bbreg(hw, 0xb98, MASKDWORD, 0x12680000); | ||
1535 | |||
1536 | offset = 0xb68; | ||
1537 | index = 11; | ||
1538 | for (; index < 13; index++) { | ||
1539 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1540 | apk_normal_setting_value_1 | ||
1541 | [index]); | ||
1542 | |||
1543 | offset += 0x04; | ||
1544 | } | ||
1545 | |||
1546 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x40000000); | ||
1547 | |||
1548 | offset = 0xb60; | ||
1549 | for (index = 0; index < 16; index++) { | ||
1550 | rtl_set_bbreg(hw, offset, MASKDWORD, | ||
1551 | apk_normal_setting_value_2 | ||
1552 | [index]); | ||
1553 | |||
1554 | offset += 0x04; | ||
1555 | } | ||
1556 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x00000000); | ||
1557 | } | ||
1558 | |||
1559 | reg_d[path] = rtl_get_rfreg(hw, (enum radio_path)path, | ||
1560 | 0xd, MASKDWORD); | ||
1561 | |||
1562 | for (index = 0; index < APK_AFE_REG_NUM; index++) | ||
1563 | rtl_set_bbreg(hw, afe_reg[index], MASKDWORD, | ||
1564 | afe_on_off[path]); | ||
1565 | |||
1566 | if (path == RF90_PATH_A) { | ||
1567 | for (index = 0; index < APK_BB_REG_NUM; index++) { | ||
1568 | if (index == 0) | ||
1569 | continue; | ||
1570 | rtl_set_bbreg(hw, bb_reg[index], MASKDWORD, | ||
1571 | bb_ap_mode[index]); | ||
1572 | } | ||
1573 | } | ||
1574 | |||
1575 | _rtl92c_phy_mac_setting_calibration(hw, mac_reg, mac_backup); | ||
1576 | |||
1577 | if (path == 0) { | ||
1578 | rtl_set_rfreg(hw, RF90_PATH_B, 0x0, MASKDWORD, 0x10000); | ||
1579 | } else { | ||
1580 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASKDWORD, | ||
1581 | 0x10000); | ||
1582 | rtl_set_rfreg(hw, RF90_PATH_A, 0x10, MASKDWORD, | ||
1583 | 0x1000f); | ||
1584 | rtl_set_rfreg(hw, RF90_PATH_A, 0x11, MASKDWORD, | ||
1585 | 0x20103); | ||
1586 | } | ||
1587 | |||
1588 | delta_offset = ((delta + 14) / 2); | ||
1589 | if (delta_offset < 0) | ||
1590 | delta_offset = 0; | ||
1591 | else if (delta_offset > 12) | ||
1592 | delta_offset = 12; | ||
1593 | |||
1594 | for (index = 0; index < APK_BB_REG_NUM; index++) { | ||
1595 | if (index != 1) | ||
1596 | continue; | ||
1597 | |||
1598 | tmpreg = apk_rf_init_value[path][index]; | ||
1599 | |||
1600 | if (!rtlefuse->apk_thermalmeterignore) { | ||
1601 | bb_offset = (tmpreg & 0xF0000) >> 16; | ||
1602 | |||
1603 | if (!(tmpreg & BIT(15))) | ||
1604 | bb_offset = -bb_offset; | ||
1605 | |||
1606 | delta_v = | ||
1607 | apk_delta_mapping[index][delta_offset]; | ||
1608 | |||
1609 | bb_offset += delta_v; | ||
1610 | |||
1611 | if (bb_offset < 0) { | ||
1612 | tmpreg = tmpreg & (~BIT(15)); | ||
1613 | bb_offset = -bb_offset; | ||
1614 | } else { | ||
1615 | tmpreg = tmpreg | BIT(15); | ||
1616 | } | ||
1617 | |||
1618 | tmpreg = | ||
1619 | (tmpreg & 0xFFF0FFFF) | (bb_offset << 16); | ||
1620 | } | ||
1621 | |||
1622 | rtl_set_rfreg(hw, (enum radio_path)path, 0xc, | ||
1623 | MASKDWORD, 0x8992e); | ||
1624 | rtl_set_rfreg(hw, (enum radio_path)path, 0x0, | ||
1625 | MASKDWORD, apk_rf_value_0[path][index]); | ||
1626 | rtl_set_rfreg(hw, (enum radio_path)path, 0xd, | ||
1627 | MASKDWORD, tmpreg); | ||
1628 | |||
1629 | i = 0; | ||
1630 | do { | ||
1631 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x80000000); | ||
1632 | rtl_set_bbreg(hw, apk_offset[path], | ||
1633 | MASKDWORD, apk_value[0]); | ||
1634 | RTPRINT(rtlpriv, FINIT, INIT_IQK, | ||
1635 | ("PHY_APCalibrate() offset 0x%x " | ||
1636 | "value 0x%x\n", | ||
1637 | apk_offset[path], | ||
1638 | rtl_get_bbreg(hw, apk_offset[path], | ||
1639 | MASKDWORD))); | ||
1640 | |||
1641 | mdelay(3); | ||
1642 | |||
1643 | rtl_set_bbreg(hw, apk_offset[path], | ||
1644 | MASKDWORD, apk_value[1]); | ||
1645 | RTPRINT(rtlpriv, FINIT, INIT_IQK, | ||
1646 | ("PHY_APCalibrate() offset 0x%x " | ||
1647 | "value 0x%x\n", | ||
1648 | apk_offset[path], | ||
1649 | rtl_get_bbreg(hw, apk_offset[path], | ||
1650 | MASKDWORD))); | ||
1651 | |||
1652 | mdelay(20); | ||
1653 | |||
1654 | rtl_set_bbreg(hw, 0xe28, MASKDWORD, 0x00000000); | ||
1655 | |||
1656 | if (path == RF90_PATH_A) | ||
1657 | tmpreg = rtl_get_bbreg(hw, 0xbd8, | ||
1658 | 0x03E00000); | ||
1659 | else | ||
1660 | tmpreg = rtl_get_bbreg(hw, 0xbd8, | ||
1661 | 0xF8000000); | ||
1662 | |||
1663 | RTPRINT(rtlpriv, FINIT, INIT_IQK, | ||
1664 | ("PHY_APCalibrate() offset " | ||
1665 | "0xbd8[25:21] %x\n", tmpreg)); | ||
1666 | |||
1667 | i++; | ||
1668 | |||
1669 | } while (tmpreg > apkbound && i < 4); | ||
1670 | |||
1671 | apk_result[path][index] = tmpreg; | ||
1672 | } | ||
1673 | } | ||
1674 | |||
1675 | _rtl92c_phy_reload_mac_registers(hw, mac_reg, mac_backup); | ||
1676 | |||
1677 | for (index = 0; index < APK_BB_REG_NUM; index++) { | ||
1678 | if (index == 0) | ||
1679 | continue; | ||
1680 | rtl_set_bbreg(hw, bb_reg[index], MASKDWORD, bb_backup[index]); | ||
1681 | } | ||
1682 | |||
1683 | _rtl92c_phy_reload_adda_registers(hw, afe_reg, afe_backup, 16); | ||
1684 | |||
1685 | for (path = 0; path < pathbound; path++) { | ||
1686 | rtl_set_rfreg(hw, (enum radio_path)path, 0xd, | ||
1687 | MASKDWORD, reg_d[path]); | ||
1688 | |||
1689 | if (path == RF90_PATH_B) { | ||
1690 | rtl_set_rfreg(hw, RF90_PATH_A, 0x10, MASKDWORD, | ||
1691 | 0x1000f); | ||
1692 | rtl_set_rfreg(hw, RF90_PATH_A, 0x11, MASKDWORD, | ||
1693 | 0x20101); | ||
1694 | } | ||
1695 | |||
1696 | if (apk_result[path][1] > 6) | ||
1697 | apk_result[path][1] = 6; | ||
1698 | } | ||
1699 | |||
1700 | for (path = 0; path < pathbound; path++) { | ||
1701 | rtl_set_rfreg(hw, (enum radio_path)path, 0x3, MASKDWORD, | ||
1702 | ((apk_result[path][1] << 15) | | ||
1703 | (apk_result[path][1] << 10) | | ||
1704 | (apk_result[path][1] << 5) | | ||
1705 | apk_result[path][1])); | ||
1706 | |||
1707 | if (path == RF90_PATH_A) | ||
1708 | rtl_set_rfreg(hw, (enum radio_path)path, 0x4, MASKDWORD, | ||
1709 | ((apk_result[path][1] << 15) | | ||
1710 | (apk_result[path][1] << 10) | | ||
1711 | (0x00 << 5) | 0x05)); | ||
1712 | else | ||
1713 | rtl_set_rfreg(hw, (enum radio_path)path, 0x4, MASKDWORD, | ||
1714 | ((apk_result[path][1] << 15) | | ||
1715 | (apk_result[path][1] << 10) | | ||
1716 | (0x02 << 5) | 0x05)); | ||
1717 | |||
1718 | rtl_set_rfreg(hw, (enum radio_path)path, 0xe, MASKDWORD, | ||
1719 | ((0x08 << 15) | (0x08 << 10) | (0x08 << 5) | | ||
1720 | 0x08)); | ||
1721 | |||
1722 | } | ||
1723 | rtlphy->b_apk_done = true; | ||
1724 | #endif | ||
1725 | } | ||
1726 | |||
1727 | static void _rtl92c_phy_set_rfpath_switch(struct ieee80211_hw *hw, | ||
1728 | bool bmain, bool is2t) | ||
1729 | { | ||
1730 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1731 | |||
1732 | if (is_hal_stop(rtlhal)) { | ||
1733 | rtl_set_bbreg(hw, REG_LEDCFG0, BIT(23), 0x01); | ||
1734 | rtl_set_bbreg(hw, rFPGA0_XAB_RFPARAMETER, BIT(13), 0x01); | ||
1735 | } | ||
1736 | if (is2t) { | ||
1737 | if (bmain) | ||
1738 | rtl_set_bbreg(hw, RFPGA0_XB_RFINTERFACEOE, | ||
1739 | BIT(5) | BIT(6), 0x1); | ||
1740 | else | ||
1741 | rtl_set_bbreg(hw, RFPGA0_XB_RFINTERFACEOE, | ||
1742 | BIT(5) | BIT(6), 0x2); | ||
1743 | } else { | ||
1744 | if (bmain) | ||
1745 | rtl_set_bbreg(hw, RFPGA0_XA_RFINTERFACEOE, 0x300, 0x2); | ||
1746 | else | ||
1747 | rtl_set_bbreg(hw, RFPGA0_XA_RFINTERFACEOE, 0x300, 0x1); | ||
1748 | |||
1749 | } | ||
1750 | |||
1751 | } | ||
1752 | |||
1753 | #undef IQK_ADDA_REG_NUM | ||
1754 | #undef IQK_DELAY_TIME | ||
1755 | |||
1756 | void rtl92c_phy_iq_calibrate(struct ieee80211_hw *hw, bool recovery) | ||
1757 | { | ||
1758 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1759 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1760 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1761 | |||
1762 | long result[4][8]; | ||
1763 | u8 i, final_candidate; | ||
1764 | bool patha_ok, pathb_ok; | ||
1765 | long reg_e94, reg_e9c, reg_ea4, reg_eac, reg_eb4, reg_ebc, reg_ec4, | ||
1766 | reg_ecc, reg_tmp = 0; | ||
1767 | bool is12simular, is13simular, is23simular; | ||
1768 | bool start_conttx = false, singletone = false; | ||
1769 | u32 iqk_bb_reg[10] = { | ||
1770 | ROFDM0_XARXIQIMBALANCE, | ||
1771 | ROFDM0_XBRXIQIMBALANCE, | ||
1772 | ROFDM0_ECCATHRESHOLD, | ||
1773 | ROFDM0_AGCRSSITABLE, | ||
1774 | ROFDM0_XATXIQIMBALANCE, | ||
1775 | ROFDM0_XBTXIQIMBALANCE, | ||
1776 | ROFDM0_XCTXIQIMBALANCE, | ||
1777 | ROFDM0_XCTXAFE, | ||
1778 | ROFDM0_XDTXAFE, | ||
1779 | ROFDM0_RXIQEXTANTA | ||
1780 | }; | ||
1781 | |||
1782 | if (recovery) { | ||
1783 | _rtl92c_phy_reload_adda_registers(hw, | ||
1784 | iqk_bb_reg, | ||
1785 | rtlphy->iqk_bb_backup, 10); | ||
1786 | return; | ||
1787 | } | ||
1788 | if (start_conttx || singletone) | ||
1789 | return; | ||
1790 | for (i = 0; i < 8; i++) { | ||
1791 | result[0][i] = 0; | ||
1792 | result[1][i] = 0; | ||
1793 | result[2][i] = 0; | ||
1794 | result[3][i] = 0; | ||
1795 | } | ||
1796 | final_candidate = 0xff; | ||
1797 | patha_ok = false; | ||
1798 | pathb_ok = false; | ||
1799 | is12simular = false; | ||
1800 | is23simular = false; | ||
1801 | is13simular = false; | ||
1802 | for (i = 0; i < 3; i++) { | ||
1803 | if (IS_92C_SERIAL(rtlhal->version)) | ||
1804 | _rtl92c_phy_iq_calibrate(hw, result, i, true); | ||
1805 | else | ||
1806 | _rtl92c_phy_iq_calibrate(hw, result, i, false); | ||
1807 | if (i == 1) { | ||
1808 | is12simular = _rtl92c_phy_simularity_compare(hw, | ||
1809 | result, 0, | ||
1810 | 1); | ||
1811 | if (is12simular) { | ||
1812 | final_candidate = 0; | ||
1813 | break; | ||
1814 | } | ||
1815 | } | ||
1816 | if (i == 2) { | ||
1817 | is13simular = _rtl92c_phy_simularity_compare(hw, | ||
1818 | result, 0, | ||
1819 | 2); | ||
1820 | if (is13simular) { | ||
1821 | final_candidate = 0; | ||
1822 | break; | ||
1823 | } | ||
1824 | is23simular = _rtl92c_phy_simularity_compare(hw, | ||
1825 | result, 1, | ||
1826 | 2); | ||
1827 | if (is23simular) | ||
1828 | final_candidate = 1; | ||
1829 | else { | ||
1830 | for (i = 0; i < 8; i++) | ||
1831 | reg_tmp += result[3][i]; | ||
1832 | |||
1833 | if (reg_tmp != 0) | ||
1834 | final_candidate = 3; | ||
1835 | else | ||
1836 | final_candidate = 0xFF; | ||
1837 | } | ||
1838 | } | ||
1839 | } | ||
1840 | for (i = 0; i < 4; i++) { | ||
1841 | reg_e94 = result[i][0]; | ||
1842 | reg_e9c = result[i][1]; | ||
1843 | reg_ea4 = result[i][2]; | ||
1844 | reg_eac = result[i][3]; | ||
1845 | reg_eb4 = result[i][4]; | ||
1846 | reg_ebc = result[i][5]; | ||
1847 | reg_ec4 = result[i][6]; | ||
1848 | reg_ecc = result[i][7]; | ||
1849 | } | ||
1850 | if (final_candidate != 0xff) { | ||
1851 | rtlphy->reg_e94 = reg_e94 = result[final_candidate][0]; | ||
1852 | rtlphy->reg_e9c = reg_e9c = result[final_candidate][1]; | ||
1853 | reg_ea4 = result[final_candidate][2]; | ||
1854 | reg_eac = result[final_candidate][3]; | ||
1855 | rtlphy->reg_eb4 = reg_eb4 = result[final_candidate][4]; | ||
1856 | rtlphy->reg_ebc = reg_ebc = result[final_candidate][5]; | ||
1857 | reg_ec4 = result[final_candidate][6]; | ||
1858 | reg_ecc = result[final_candidate][7]; | ||
1859 | patha_ok = pathb_ok = true; | ||
1860 | } else { | ||
1861 | rtlphy->reg_e94 = rtlphy->reg_eb4 = 0x100; | ||
1862 | rtlphy->reg_e9c = rtlphy->reg_ebc = 0x0; | ||
1863 | } | ||
1864 | if (reg_e94 != 0) /*&&(reg_ea4 != 0) */ | ||
1865 | _rtl92c_phy_path_a_fill_iqk_matrix(hw, patha_ok, result, | ||
1866 | final_candidate, | ||
1867 | (reg_ea4 == 0)); | ||
1868 | if (IS_92C_SERIAL(rtlhal->version)) { | ||
1869 | if (reg_eb4 != 0) /*&&(reg_ec4 != 0) */ | ||
1870 | _rtl92c_phy_path_b_fill_iqk_matrix(hw, pathb_ok, | ||
1871 | result, | ||
1872 | final_candidate, | ||
1873 | (reg_ec4 == 0)); | ||
1874 | } | ||
1875 | _rtl92c_phy_save_adda_registers(hw, iqk_bb_reg, | ||
1876 | rtlphy->iqk_bb_backup, 10); | ||
1877 | } | ||
1878 | EXPORT_SYMBOL(rtl92c_phy_iq_calibrate); | ||
1879 | |||
1880 | void rtl92c_phy_lc_calibrate(struct ieee80211_hw *hw) | ||
1881 | { | ||
1882 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1883 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1884 | bool start_conttx = false, singletone = false; | ||
1885 | |||
1886 | if (start_conttx || singletone) | ||
1887 | return; | ||
1888 | if (IS_92C_SERIAL(rtlhal->version)) | ||
1889 | rtlpriv->cfg->ops->phy_lc_calibrate(hw, true); | ||
1890 | else | ||
1891 | rtlpriv->cfg->ops->phy_lc_calibrate(hw, false); | ||
1892 | } | ||
1893 | EXPORT_SYMBOL(rtl92c_phy_lc_calibrate); | ||
1894 | |||
1895 | void rtl92c_phy_ap_calibrate(struct ieee80211_hw *hw, char delta) | ||
1896 | { | ||
1897 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1898 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1899 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1900 | |||
1901 | if (rtlphy->apk_done) | ||
1902 | return; | ||
1903 | if (IS_92C_SERIAL(rtlhal->version)) | ||
1904 | _rtl92c_phy_ap_calibrate(hw, delta, true); | ||
1905 | else | ||
1906 | _rtl92c_phy_ap_calibrate(hw, delta, false); | ||
1907 | } | ||
1908 | EXPORT_SYMBOL(rtl92c_phy_ap_calibrate); | ||
1909 | |||
1910 | void rtl92c_phy_set_rfpath_switch(struct ieee80211_hw *hw, bool bmain) | ||
1911 | { | ||
1912 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1913 | |||
1914 | if (IS_92C_SERIAL(rtlhal->version)) | ||
1915 | _rtl92c_phy_set_rfpath_switch(hw, bmain, true); | ||
1916 | else | ||
1917 | _rtl92c_phy_set_rfpath_switch(hw, bmain, false); | ||
1918 | } | ||
1919 | EXPORT_SYMBOL(rtl92c_phy_set_rfpath_switch); | ||
1920 | |||
1921 | bool rtl92c_phy_set_io_cmd(struct ieee80211_hw *hw, enum io_type iotype) | ||
1922 | { | ||
1923 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1924 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1925 | bool postprocessing = false; | ||
1926 | |||
1927 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, | ||
1928 | ("-->IO Cmd(%#x), set_io_inprogress(%d)\n", | ||
1929 | iotype, rtlphy->set_io_inprogress)); | ||
1930 | do { | ||
1931 | switch (iotype) { | ||
1932 | case IO_CMD_RESUME_DM_BY_SCAN: | ||
1933 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, | ||
1934 | ("[IO CMD] Resume DM after scan.\n")); | ||
1935 | postprocessing = true; | ||
1936 | break; | ||
1937 | case IO_CMD_PAUSE_DM_BY_SCAN: | ||
1938 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, | ||
1939 | ("[IO CMD] Pause DM before scan.\n")); | ||
1940 | postprocessing = true; | ||
1941 | break; | ||
1942 | default: | ||
1943 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1944 | ("switch case not process\n")); | ||
1945 | break; | ||
1946 | } | ||
1947 | } while (false); | ||
1948 | if (postprocessing && !rtlphy->set_io_inprogress) { | ||
1949 | rtlphy->set_io_inprogress = true; | ||
1950 | rtlphy->current_io_type = iotype; | ||
1951 | } else { | ||
1952 | return false; | ||
1953 | } | ||
1954 | rtl92c_phy_set_io(hw); | ||
1955 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, ("<--IO Type(%#x)\n", iotype)); | ||
1956 | return true; | ||
1957 | } | ||
1958 | EXPORT_SYMBOL(rtl92c_phy_set_io_cmd); | ||
1959 | |||
1960 | void rtl92c_phy_set_io(struct ieee80211_hw *hw) | ||
1961 | { | ||
1962 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1963 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1964 | |||
1965 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, | ||
1966 | ("--->Cmd(%#x), set_io_inprogress(%d)\n", | ||
1967 | rtlphy->current_io_type, rtlphy->set_io_inprogress)); | ||
1968 | switch (rtlphy->current_io_type) { | ||
1969 | case IO_CMD_RESUME_DM_BY_SCAN: | ||
1970 | dm_digtable.cur_igvalue = rtlphy->initgain_backup.xaagccore1; | ||
1971 | rtl92c_dm_write_dig(hw); | ||
1972 | rtl92c_phy_set_txpower_level(hw, rtlphy->current_channel); | ||
1973 | break; | ||
1974 | case IO_CMD_PAUSE_DM_BY_SCAN: | ||
1975 | rtlphy->initgain_backup.xaagccore1 = dm_digtable.cur_igvalue; | ||
1976 | dm_digtable.cur_igvalue = 0x17; | ||
1977 | rtl92c_dm_write_dig(hw); | ||
1978 | break; | ||
1979 | default: | ||
1980 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1981 | ("switch case not process\n")); | ||
1982 | break; | ||
1983 | } | ||
1984 | rtlphy->set_io_inprogress = false; | ||
1985 | RT_TRACE(rtlpriv, COMP_CMD, DBG_TRACE, | ||
1986 | ("<---(%#x)\n", rtlphy->current_io_type)); | ||
1987 | } | ||
1988 | EXPORT_SYMBOL(rtl92c_phy_set_io); | ||
1989 | |||
1990 | void rtl92ce_phy_set_rf_on(struct ieee80211_hw *hw) | ||
1991 | { | ||
1992 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1993 | |||
1994 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x2b); | ||
1995 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE3); | ||
1996 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x00); | ||
1997 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
1998 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE3); | ||
1999 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00); | ||
2000 | } | ||
2001 | EXPORT_SYMBOL(rtl92ce_phy_set_rf_on); | ||
2002 | |||
2003 | void _rtl92c_phy_set_rf_sleep(struct ieee80211_hw *hw) | ||
2004 | { | ||
2005 | u32 u4b_tmp; | ||
2006 | u8 delay = 5; | ||
2007 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2008 | |||
2009 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
2010 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, RFREG_OFFSET_MASK, 0x00); | ||
2011 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x40); | ||
2012 | u4b_tmp = rtl_get_rfreg(hw, RF90_PATH_A, 0, RFREG_OFFSET_MASK); | ||
2013 | while (u4b_tmp != 0 && delay > 0) { | ||
2014 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x0); | ||
2015 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, RFREG_OFFSET_MASK, 0x00); | ||
2016 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x40); | ||
2017 | u4b_tmp = rtl_get_rfreg(hw, RF90_PATH_A, 0, RFREG_OFFSET_MASK); | ||
2018 | delay--; | ||
2019 | } | ||
2020 | if (delay == 0) { | ||
2021 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x00); | ||
2022 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
2023 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE3); | ||
2024 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00); | ||
2025 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
2026 | ("Switch RF timeout !!!.\n")); | ||
2027 | return; | ||
2028 | } | ||
2029 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
2030 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x22); | ||
2031 | } | ||
2032 | EXPORT_SYMBOL(_rtl92c_phy_set_rf_sleep); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.h b/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.h new file mode 100644 index 000000000000..9a264c0d6127 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192c/phy_common.h | |||
@@ -0,0 +1,258 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_PHY_COMMON_H__ | ||
31 | #define __RTL92C_PHY_COMMON_H__ | ||
32 | |||
33 | #define MAX_PRECMD_CNT 16 | ||
34 | #define MAX_RFDEPENDCMD_CNT 16 | ||
35 | #define MAX_POSTCMD_CNT 16 | ||
36 | |||
37 | #define MAX_DOZE_WAITING_TIMES_9x 64 | ||
38 | |||
39 | #define RT_CANNOT_IO(hw) false | ||
40 | #define HIGHPOWER_RADIOA_ARRAYLEN 22 | ||
41 | |||
42 | #define IQK_ADDA_REG_NUM 16 | ||
43 | #define MAX_TOLERANCE 5 | ||
44 | #define IQK_DELAY_TIME 1 | ||
45 | |||
46 | #define APK_BB_REG_NUM 5 | ||
47 | #define APK_AFE_REG_NUM 16 | ||
48 | #define APK_CURVE_REG_NUM 4 | ||
49 | #define PATH_NUM 2 | ||
50 | |||
51 | #define LOOP_LIMIT 5 | ||
52 | #define MAX_STALL_TIME 50 | ||
53 | #define AntennaDiversityValue 0x80 | ||
54 | #define MAX_TXPWR_IDX_NMODE_92S 63 | ||
55 | #define Reset_Cnt_Limit 3 | ||
56 | |||
57 | #define IQK_ADDA_REG_NUM 16 | ||
58 | #define IQK_MAC_REG_NUM 4 | ||
59 | |||
60 | #define IQK_DELAY_TIME 1 | ||
61 | #define RF90_PATH_MAX 2 | ||
62 | |||
63 | #define CT_OFFSET_MAC_ADDR 0X16 | ||
64 | |||
65 | #define CT_OFFSET_CCK_TX_PWR_IDX 0x5A | ||
66 | #define CT_OFFSET_HT401S_TX_PWR_IDX 0x60 | ||
67 | #define CT_OFFSET_HT402S_TX_PWR_IDX_DIF 0x66 | ||
68 | #define CT_OFFSET_HT20_TX_PWR_IDX_DIFF 0x69 | ||
69 | #define CT_OFFSET_OFDM_TX_PWR_IDX_DIFF 0x6C | ||
70 | |||
71 | #define CT_OFFSET_HT40_MAX_PWR_OFFSET 0x6F | ||
72 | #define CT_OFFSET_HT20_MAX_PWR_OFFSET 0x72 | ||
73 | |||
74 | #define CT_OFFSET_CHANNEL_PLAH 0x75 | ||
75 | #define CT_OFFSET_THERMAL_METER 0x78 | ||
76 | #define CT_OFFSET_RF_OPTION 0x79 | ||
77 | #define CT_OFFSET_VERSION 0x7E | ||
78 | #define CT_OFFSET_CUSTOMER_ID 0x7F | ||
79 | |||
80 | #define RTL92C_MAX_PATH_NUM 2 | ||
81 | #define LLT_LAST_ENTRY_OF_TX_PKT_BUFFER 255 | ||
82 | |||
83 | enum swchnlcmd_id { | ||
84 | CMDID_END, | ||
85 | CMDID_SET_TXPOWEROWER_LEVEL, | ||
86 | CMDID_BBREGWRITE10, | ||
87 | CMDID_WRITEPORT_ULONG, | ||
88 | CMDID_WRITEPORT_USHORT, | ||
89 | CMDID_WRITEPORT_UCHAR, | ||
90 | CMDID_RF_WRITEREG, | ||
91 | }; | ||
92 | |||
93 | struct swchnlcmd { | ||
94 | enum swchnlcmd_id cmdid; | ||
95 | u32 para1; | ||
96 | u32 para2; | ||
97 | u32 msdelay; | ||
98 | }; | ||
99 | |||
100 | enum hw90_block_e { | ||
101 | HW90_BLOCK_MAC = 0, | ||
102 | HW90_BLOCK_PHY0 = 1, | ||
103 | HW90_BLOCK_PHY1 = 2, | ||
104 | HW90_BLOCK_RF = 3, | ||
105 | HW90_BLOCK_MAXIMUM = 4, | ||
106 | }; | ||
107 | |||
108 | enum baseband_config_type { | ||
109 | BASEBAND_CONFIG_PHY_REG = 0, | ||
110 | BASEBAND_CONFIG_AGC_TAB = 1, | ||
111 | }; | ||
112 | |||
113 | enum ra_offset_area { | ||
114 | RA_OFFSET_LEGACY_OFDM1, | ||
115 | RA_OFFSET_LEGACY_OFDM2, | ||
116 | RA_OFFSET_HT_OFDM1, | ||
117 | RA_OFFSET_HT_OFDM2, | ||
118 | RA_OFFSET_HT_OFDM3, | ||
119 | RA_OFFSET_HT_OFDM4, | ||
120 | RA_OFFSET_HT_CCK, | ||
121 | }; | ||
122 | |||
123 | enum antenna_path { | ||
124 | ANTENNA_NONE, | ||
125 | ANTENNA_D, | ||
126 | ANTENNA_C, | ||
127 | ANTENNA_CD, | ||
128 | ANTENNA_B, | ||
129 | ANTENNA_BD, | ||
130 | ANTENNA_BC, | ||
131 | ANTENNA_BCD, | ||
132 | ANTENNA_A, | ||
133 | ANTENNA_AD, | ||
134 | ANTENNA_AC, | ||
135 | ANTENNA_ACD, | ||
136 | ANTENNA_AB, | ||
137 | ANTENNA_ABD, | ||
138 | ANTENNA_ABC, | ||
139 | ANTENNA_ABCD | ||
140 | }; | ||
141 | |||
142 | struct r_antenna_select_ofdm { | ||
143 | u32 r_tx_antenna:4; | ||
144 | u32 r_ant_l:4; | ||
145 | u32 r_ant_non_ht:4; | ||
146 | u32 r_ant_ht1:4; | ||
147 | u32 r_ant_ht2:4; | ||
148 | u32 r_ant_ht_s1:4; | ||
149 | u32 r_ant_non_ht_s1:4; | ||
150 | u32 ofdm_txsc:2; | ||
151 | u32 reserved:2; | ||
152 | }; | ||
153 | |||
154 | struct r_antenna_select_cck { | ||
155 | u8 r_cckrx_enable_2:2; | ||
156 | u8 r_cckrx_enable:2; | ||
157 | u8 r_ccktx_enable:4; | ||
158 | }; | ||
159 | |||
160 | struct efuse_contents { | ||
161 | u8 mac_addr[ETH_ALEN]; | ||
162 | u8 cck_tx_power_idx[6]; | ||
163 | u8 ht40_1s_tx_power_idx[6]; | ||
164 | u8 ht40_2s_tx_power_idx_diff[3]; | ||
165 | u8 ht20_tx_power_idx_diff[3]; | ||
166 | u8 ofdm_tx_power_idx_diff[3]; | ||
167 | u8 ht40_max_power_offset[3]; | ||
168 | u8 ht20_max_power_offset[3]; | ||
169 | u8 channel_plan; | ||
170 | u8 thermal_meter; | ||
171 | u8 rf_option[5]; | ||
172 | u8 version; | ||
173 | u8 oem_id; | ||
174 | u8 regulatory; | ||
175 | }; | ||
176 | |||
177 | struct tx_power_struct { | ||
178 | u8 cck[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
179 | u8 ht40_1s[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
180 | u8 ht40_2s[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
181 | u8 ht20_diff[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
182 | u8 legacy_ht_diff[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
183 | u8 legacy_ht_txpowerdiff; | ||
184 | u8 groupht20[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
185 | u8 groupht40[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
186 | u8 pwrgroup_cnt; | ||
187 | u32 mcs_original_offset[4][16]; | ||
188 | }; | ||
189 | |||
190 | u32 rtl92c_phy_query_bb_reg(struct ieee80211_hw *hw, | ||
191 | u32 regaddr, u32 bitmask); | ||
192 | void rtl92c_phy_set_bb_reg(struct ieee80211_hw *hw, | ||
193 | u32 regaddr, u32 bitmask, u32 data); | ||
194 | u32 rtl92c_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
195 | enum radio_path rfpath, u32 regaddr, | ||
196 | u32 bitmask); | ||
197 | bool rtl92c_phy_mac_config(struct ieee80211_hw *hw); | ||
198 | bool rtl92c_phy_bb_config(struct ieee80211_hw *hw); | ||
199 | bool rtl92c_phy_rf_config(struct ieee80211_hw *hw); | ||
200 | bool rtl92c_phy_config_rf_with_feaderfile(struct ieee80211_hw *hw, | ||
201 | enum radio_path rfpath); | ||
202 | void rtl92c_phy_get_hw_reg_originalvalue(struct ieee80211_hw *hw); | ||
203 | void rtl92c_phy_get_txpower_level(struct ieee80211_hw *hw, | ||
204 | long *powerlevel); | ||
205 | void rtl92c_phy_set_txpower_level(struct ieee80211_hw *hw, u8 channel); | ||
206 | bool rtl92c_phy_update_txpower_dbm(struct ieee80211_hw *hw, | ||
207 | long power_indbm); | ||
208 | void rtl92c_phy_scan_operation_backup(struct ieee80211_hw *hw, | ||
209 | u8 operation); | ||
210 | void rtl92c_phy_set_bw_mode(struct ieee80211_hw *hw, | ||
211 | enum nl80211_channel_type ch_type); | ||
212 | void rtl92c_phy_sw_chnl_callback(struct ieee80211_hw *hw); | ||
213 | u8 rtl92c_phy_sw_chnl(struct ieee80211_hw *hw); | ||
214 | void rtl92c_phy_iq_calibrate(struct ieee80211_hw *hw, bool b_recovery); | ||
215 | void rtl92c_phy_set_beacon_hw_reg(struct ieee80211_hw *hw, | ||
216 | u16 beaconinterval); | ||
217 | void rtl92c_phy_ap_calibrate(struct ieee80211_hw *hw, char delta); | ||
218 | void rtl92c_phy_lc_calibrate(struct ieee80211_hw *hw); | ||
219 | void rtl92c_phy_set_rfpath_switch(struct ieee80211_hw *hw, bool bmain); | ||
220 | bool rtl92c_phy_config_rf_with_headerfile(struct ieee80211_hw *hw, | ||
221 | enum radio_path rfpath); | ||
222 | bool rtl8192_phy_check_is_legal_rfpath(struct ieee80211_hw *hw, | ||
223 | u32 rfpath); | ||
224 | bool rtl92c_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
225 | enum rf_pwrstate rfpwr_state); | ||
226 | void rtl92ce_phy_set_rf_on(struct ieee80211_hw *hw); | ||
227 | void rtl92c_phy_set_io(struct ieee80211_hw *hw); | ||
228 | void rtl92c_bb_block_on(struct ieee80211_hw *hw); | ||
229 | u32 _rtl92c_phy_calculate_bit_shift(u32 bitmask); | ||
230 | long _rtl92c_phy_txpwr_idx_to_dbm(struct ieee80211_hw *hw, | ||
231 | enum wireless_mode wirelessmode, | ||
232 | u8 txpwridx); | ||
233 | u8 _rtl92c_phy_dbm_to_txpwr_Idx(struct ieee80211_hw *hw, | ||
234 | enum wireless_mode wirelessmode, | ||
235 | long power_indbm); | ||
236 | void _rtl92c_phy_init_bb_rf_register_definition(struct ieee80211_hw *hw); | ||
237 | void _rtl92c_phy_set_rf_sleep(struct ieee80211_hw *hw); | ||
238 | bool _rtl92c_phy_sw_chnl_step_by_step(struct ieee80211_hw *hw, | ||
239 | u8 channel, u8 *stage, u8 *step, | ||
240 | u32 *delay); | ||
241 | u8 rtl92c_bt_rssi_state_change(struct ieee80211_hw *hw); | ||
242 | u32 _rtl92c_phy_fw_rf_serial_read(struct ieee80211_hw *hw, | ||
243 | enum radio_path rfpath, u32 offset); | ||
244 | void _rtl92c_phy_fw_rf_serial_write(struct ieee80211_hw *hw, | ||
245 | enum radio_path rfpath, u32 offset, | ||
246 | u32 data); | ||
247 | u32 _rtl92c_phy_rf_serial_read(struct ieee80211_hw *hw, | ||
248 | enum radio_path rfpath, u32 offset); | ||
249 | void _rtl92c_phy_rf_serial_write(struct ieee80211_hw *hw, | ||
250 | enum radio_path rfpath, u32 offset, | ||
251 | u32 data); | ||
252 | bool _rtl92c_phy_bb8192c_config_parafile(struct ieee80211_hw *hw); | ||
253 | void _rtl92c_store_pwrIndex_diffrate_offset(struct ieee80211_hw *hw, | ||
254 | u32 regaddr, u32 bitmask, | ||
255 | u32 data); | ||
256 | bool rtl92c_phy_set_io_cmd(struct ieee80211_hw *hw, enum io_type iotype); | ||
257 | |||
258 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/Makefile b/drivers/net/wireless/rtlwifi/rtl8192ce/Makefile new file mode 100644 index 000000000000..c0cb0cfe7d37 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/Makefile | |||
@@ -0,0 +1,13 @@ | |||
1 | rtl8192ce-objs := \ | ||
2 | dm.o \ | ||
3 | hw.o \ | ||
4 | led.o \ | ||
5 | phy.o \ | ||
6 | rf.o \ | ||
7 | sw.o \ | ||
8 | table.o \ | ||
9 | trx.o | ||
10 | |||
11 | obj-$(CONFIG_RTL8192CE) += rtl8192ce.o | ||
12 | |||
13 | ccflags-y += -D__CHECK_ENDIAN__ | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/def.h b/drivers/net/wireless/rtlwifi/rtl8192ce/def.h new file mode 100644 index 000000000000..35ff7df41a1d --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/def.h | |||
@@ -0,0 +1,374 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_DEF_H__ | ||
31 | #define __RTL92C_DEF_H__ | ||
32 | |||
33 | #define HAL_RETRY_LIMIT_INFRA 48 | ||
34 | #define HAL_RETRY_LIMIT_AP_ADHOC 7 | ||
35 | |||
36 | #define PHY_RSSI_SLID_WIN_MAX 100 | ||
37 | #define PHY_LINKQUALITY_SLID_WIN_MAX 20 | ||
38 | #define PHY_BEACON_RSSI_SLID_WIN_MAX 10 | ||
39 | |||
40 | #define RESET_DELAY_8185 20 | ||
41 | |||
42 | #define RT_IBSS_INT_MASKS (IMR_BCNINT | IMR_TBDOK | IMR_TBDER) | ||
43 | #define RT_AC_INT_MASKS (IMR_VIDOK | IMR_VODOK | IMR_BEDOK|IMR_BKDOK) | ||
44 | |||
45 | #define NUM_OF_FIRMWARE_QUEUE 10 | ||
46 | #define NUM_OF_PAGES_IN_FW 0x100 | ||
47 | #define NUM_OF_PAGE_IN_FW_QUEUE_BK 0x07 | ||
48 | #define NUM_OF_PAGE_IN_FW_QUEUE_BE 0x07 | ||
49 | #define NUM_OF_PAGE_IN_FW_QUEUE_VI 0x07 | ||
50 | #define NUM_OF_PAGE_IN_FW_QUEUE_VO 0x07 | ||
51 | #define NUM_OF_PAGE_IN_FW_QUEUE_HCCA 0x0 | ||
52 | #define NUM_OF_PAGE_IN_FW_QUEUE_CMD 0x0 | ||
53 | #define NUM_OF_PAGE_IN_FW_QUEUE_MGNT 0x02 | ||
54 | #define NUM_OF_PAGE_IN_FW_QUEUE_HIGH 0x02 | ||
55 | #define NUM_OF_PAGE_IN_FW_QUEUE_BCN 0x2 | ||
56 | #define NUM_OF_PAGE_IN_FW_QUEUE_PUB 0xA1 | ||
57 | |||
58 | #define NUM_OF_PAGE_IN_FW_QUEUE_BK_DTM 0x026 | ||
59 | #define NUM_OF_PAGE_IN_FW_QUEUE_BE_DTM 0x048 | ||
60 | #define NUM_OF_PAGE_IN_FW_QUEUE_VI_DTM 0x048 | ||
61 | #define NUM_OF_PAGE_IN_FW_QUEUE_VO_DTM 0x026 | ||
62 | #define NUM_OF_PAGE_IN_FW_QUEUE_PUB_DTM 0x00 | ||
63 | |||
64 | #define MAX_LINES_HWCONFIG_TXT 1000 | ||
65 | #define MAX_BYTES_LINE_HWCONFIG_TXT 256 | ||
66 | |||
67 | #define SW_THREE_WIRE 0 | ||
68 | #define HW_THREE_WIRE 2 | ||
69 | |||
70 | #define BT_DEMO_BOARD 0 | ||
71 | #define BT_QA_BOARD 1 | ||
72 | #define BT_FPGA 2 | ||
73 | |||
74 | #define RX_SMOOTH_FACTOR 20 | ||
75 | |||
76 | #define HAL_PRIME_CHNL_OFFSET_DONT_CARE 0 | ||
77 | #define HAL_PRIME_CHNL_OFFSET_LOWER 1 | ||
78 | #define HAL_PRIME_CHNL_OFFSET_UPPER 2 | ||
79 | |||
80 | #define MAX_H2C_QUEUE_NUM 10 | ||
81 | |||
82 | #define RX_MPDU_QUEUE 0 | ||
83 | #define RX_CMD_QUEUE 1 | ||
84 | #define RX_MAX_QUEUE 2 | ||
85 | #define AC2QUEUEID(_AC) (_AC) | ||
86 | |||
87 | #define C2H_RX_CMD_HDR_LEN 8 | ||
88 | #define GET_C2H_CMD_CMD_LEN(__prxhdr) \ | ||
89 | LE_BITS_TO_4BYTE((__prxhdr), 0, 16) | ||
90 | #define GET_C2H_CMD_ELEMENT_ID(__prxhdr) \ | ||
91 | LE_BITS_TO_4BYTE((__prxhdr), 16, 8) | ||
92 | #define GET_C2H_CMD_CMD_SEQ(__prxhdr) \ | ||
93 | LE_BITS_TO_4BYTE((__prxhdr), 24, 7) | ||
94 | #define GET_C2H_CMD_CONTINUE(__prxhdr) \ | ||
95 | LE_BITS_TO_4BYTE((__prxhdr), 31, 1) | ||
96 | #define GET_C2H_CMD_CONTENT(__prxhdr) \ | ||
97 | ((u8 *)(__prxhdr) + C2H_RX_CMD_HDR_LEN) | ||
98 | |||
99 | #define GET_C2H_CMD_FEEDBACK_ELEMENT_ID(__pcmdfbhdr) \ | ||
100 | LE_BITS_TO_4BYTE((__pcmdfbhdr), 0, 8) | ||
101 | #define GET_C2H_CMD_FEEDBACK_CCX_LEN(__pcmdfbhdr) \ | ||
102 | LE_BITS_TO_4BYTE((__pcmdfbhdr), 8, 8) | ||
103 | #define GET_C2H_CMD_FEEDBACK_CCX_CMD_CNT(__pcmdfbhdr) \ | ||
104 | LE_BITS_TO_4BYTE((__pcmdfbhdr), 16, 16) | ||
105 | #define GET_C2H_CMD_FEEDBACK_CCX_MAC_ID(__pcmdfbhdr) \ | ||
106 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 0, 5) | ||
107 | #define GET_C2H_CMD_FEEDBACK_CCX_VALID(__pcmdfbhdr) \ | ||
108 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 7, 1) | ||
109 | #define GET_C2H_CMD_FEEDBACK_CCX_RETRY_CNT(__pcmdfbhdr) \ | ||
110 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 8, 5) | ||
111 | #define GET_C2H_CMD_FEEDBACK_CCX_TOK(__pcmdfbhdr) \ | ||
112 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 15, 1) | ||
113 | #define GET_C2H_CMD_FEEDBACK_CCX_QSEL(__pcmdfbhdr) \ | ||
114 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 16, 4) | ||
115 | #define GET_C2H_CMD_FEEDBACK_CCX_SEQ(__pcmdfbhdr) \ | ||
116 | LE_BITS_TO_4BYTE(((__pcmdfbhdr) + 4), 20, 12) | ||
117 | |||
118 | #define CHIP_VER_B BIT(4) | ||
119 | #define CHIP_92C_BITMASK BIT(0) | ||
120 | #define CHIP_92C_1T2R 0x03 | ||
121 | #define CHIP_92C 0x01 | ||
122 | #define CHIP_88C 0x00 | ||
123 | |||
124 | enum version_8192c { | ||
125 | VERSION_A_CHIP_92C = 0x01, | ||
126 | VERSION_A_CHIP_88C = 0x00, | ||
127 | VERSION_B_CHIP_92C = 0x11, | ||
128 | VERSION_B_CHIP_88C = 0x10, | ||
129 | VERSION_TEST_CHIP_88C = 0x00, | ||
130 | VERSION_TEST_CHIP_92C = 0x01, | ||
131 | VERSION_NORMAL_TSMC_CHIP_88C = 0x10, | ||
132 | VERSION_NORMAL_TSMC_CHIP_92C = 0x11, | ||
133 | VERSION_NORMAL_TSMC_CHIP_92C_1T2R = 0x13, | ||
134 | VERSION_NORMAL_UMC_CHIP_88C_A_CUT = 0x30, | ||
135 | VERSION_NORMAL_UMC_CHIP_92C_A_CUT = 0x31, | ||
136 | VERSION_NORMAL_UMC_CHIP_92C_1T2R_A_CUT = 0x33, | ||
137 | VERSION_NORMA_UMC_CHIP_8723_1T1R_A_CUT = 0x34, | ||
138 | VERSION_NORMA_UMC_CHIP_8723_1T1R_B_CUT = 0x3c, | ||
139 | VERSION_NORMAL_UMC_CHIP_88C_B_CUT = 0x70, | ||
140 | VERSION_NORMAL_UMC_CHIP_92C_B_CUT = 0x71, | ||
141 | VERSION_NORMAL_UMC_CHIP_92C_1T2R_B_CUT = 0x73, | ||
142 | VERSION_UNKNOWN = 0x88, | ||
143 | }; | ||
144 | |||
145 | #define IS_CHIP_VER_B(version) ((version & CHIP_VER_B) ? true : false) | ||
146 | #define IS_92C_SERIAL(version) ((version & CHIP_92C_BITMASK) ? true : false) | ||
147 | |||
148 | enum rtl819x_loopback_e { | ||
149 | RTL819X_NO_LOOPBACK = 0, | ||
150 | RTL819X_MAC_LOOPBACK = 1, | ||
151 | RTL819X_DMA_LOOPBACK = 2, | ||
152 | RTL819X_CCK_LOOPBACK = 3, | ||
153 | }; | ||
154 | |||
155 | enum rf_optype { | ||
156 | RF_OP_BY_SW_3WIRE = 0, | ||
157 | RF_OP_BY_FW, | ||
158 | RF_OP_MAX | ||
159 | }; | ||
160 | |||
161 | enum rf_power_state { | ||
162 | RF_ON, | ||
163 | RF_OFF, | ||
164 | RF_SLEEP, | ||
165 | RF_SHUT_DOWN, | ||
166 | }; | ||
167 | |||
168 | enum power_save_mode { | ||
169 | POWER_SAVE_MODE_ACTIVE, | ||
170 | POWER_SAVE_MODE_SAVE, | ||
171 | }; | ||
172 | |||
173 | enum power_polocy_config { | ||
174 | POWERCFG_MAX_POWER_SAVINGS, | ||
175 | POWERCFG_GLOBAL_POWER_SAVINGS, | ||
176 | POWERCFG_LOCAL_POWER_SAVINGS, | ||
177 | POWERCFG_LENOVO, | ||
178 | }; | ||
179 | |||
180 | enum interface_select_pci { | ||
181 | INTF_SEL1_MINICARD = 0, | ||
182 | INTF_SEL0_PCIE = 1, | ||
183 | INTF_SEL2_RSV = 2, | ||
184 | INTF_SEL3_RSV = 3, | ||
185 | }; | ||
186 | |||
187 | enum hal_fw_c2h_cmd_id { | ||
188 | HAL_FW_C2H_CMD_Read_MACREG = 0, | ||
189 | HAL_FW_C2H_CMD_Read_BBREG = 1, | ||
190 | HAL_FW_C2H_CMD_Read_RFREG = 2, | ||
191 | HAL_FW_C2H_CMD_Read_EEPROM = 3, | ||
192 | HAL_FW_C2H_CMD_Read_EFUSE = 4, | ||
193 | HAL_FW_C2H_CMD_Read_CAM = 5, | ||
194 | HAL_FW_C2H_CMD_Get_BasicRate = 6, | ||
195 | HAL_FW_C2H_CMD_Get_DataRate = 7, | ||
196 | HAL_FW_C2H_CMD_Survey = 8, | ||
197 | HAL_FW_C2H_CMD_SurveyDone = 9, | ||
198 | HAL_FW_C2H_CMD_JoinBss = 10, | ||
199 | HAL_FW_C2H_CMD_AddSTA = 11, | ||
200 | HAL_FW_C2H_CMD_DelSTA = 12, | ||
201 | HAL_FW_C2H_CMD_AtimDone = 13, | ||
202 | HAL_FW_C2H_CMD_TX_Report = 14, | ||
203 | HAL_FW_C2H_CMD_CCX_Report = 15, | ||
204 | HAL_FW_C2H_CMD_DTM_Report = 16, | ||
205 | HAL_FW_C2H_CMD_TX_Rate_Statistics = 17, | ||
206 | HAL_FW_C2H_CMD_C2HLBK = 18, | ||
207 | HAL_FW_C2H_CMD_C2HDBG = 19, | ||
208 | HAL_FW_C2H_CMD_C2HFEEDBACK = 20, | ||
209 | HAL_FW_C2H_CMD_MAX | ||
210 | }; | ||
211 | |||
212 | enum rtl_desc_qsel { | ||
213 | QSLT_BK = 0x2, | ||
214 | QSLT_BE = 0x0, | ||
215 | QSLT_VI = 0x5, | ||
216 | QSLT_VO = 0x7, | ||
217 | QSLT_BEACON = 0x10, | ||
218 | QSLT_HIGH = 0x11, | ||
219 | QSLT_MGNT = 0x12, | ||
220 | QSLT_CMD = 0x13, | ||
221 | }; | ||
222 | |||
223 | enum rtl_desc92c_rate { | ||
224 | DESC92C_RATE1M = 0x00, | ||
225 | DESC92C_RATE2M = 0x01, | ||
226 | DESC92C_RATE5_5M = 0x02, | ||
227 | DESC92C_RATE11M = 0x03, | ||
228 | |||
229 | DESC92C_RATE6M = 0x04, | ||
230 | DESC92C_RATE9M = 0x05, | ||
231 | DESC92C_RATE12M = 0x06, | ||
232 | DESC92C_RATE18M = 0x07, | ||
233 | DESC92C_RATE24M = 0x08, | ||
234 | DESC92C_RATE36M = 0x09, | ||
235 | DESC92C_RATE48M = 0x0a, | ||
236 | DESC92C_RATE54M = 0x0b, | ||
237 | |||
238 | DESC92C_RATEMCS0 = 0x0c, | ||
239 | DESC92C_RATEMCS1 = 0x0d, | ||
240 | DESC92C_RATEMCS2 = 0x0e, | ||
241 | DESC92C_RATEMCS3 = 0x0f, | ||
242 | DESC92C_RATEMCS4 = 0x10, | ||
243 | DESC92C_RATEMCS5 = 0x11, | ||
244 | DESC92C_RATEMCS6 = 0x12, | ||
245 | DESC92C_RATEMCS7 = 0x13, | ||
246 | DESC92C_RATEMCS8 = 0x14, | ||
247 | DESC92C_RATEMCS9 = 0x15, | ||
248 | DESC92C_RATEMCS10 = 0x16, | ||
249 | DESC92C_RATEMCS11 = 0x17, | ||
250 | DESC92C_RATEMCS12 = 0x18, | ||
251 | DESC92C_RATEMCS13 = 0x19, | ||
252 | DESC92C_RATEMCS14 = 0x1a, | ||
253 | DESC92C_RATEMCS15 = 0x1b, | ||
254 | DESC92C_RATEMCS15_SG = 0x1c, | ||
255 | DESC92C_RATEMCS32 = 0x20, | ||
256 | }; | ||
257 | |||
258 | struct phy_sts_cck_8192s_t { | ||
259 | u8 adc_pwdb_X[4]; | ||
260 | u8 sq_rpt; | ||
261 | u8 cck_agc_rpt; | ||
262 | }; | ||
263 | |||
264 | struct h2c_cmd_8192c { | ||
265 | u8 element_id; | ||
266 | u32 cmd_len; | ||
267 | u8 *p_cmdbuffer; | ||
268 | }; | ||
269 | |||
270 | /* NOTE: reference to rtl8192c_rates struct */ | ||
271 | static inline int _rtl92c_rate_mapping(struct ieee80211_hw *hw, bool isHT, | ||
272 | u8 desc_rate, bool first_ampdu) | ||
273 | { | ||
274 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
275 | int rate_idx = 0; | ||
276 | |||
277 | if (first_ampdu) { | ||
278 | if (false == isHT) { | ||
279 | switch (desc_rate) { | ||
280 | case DESC92C_RATE1M: | ||
281 | rate_idx = 0; | ||
282 | break; | ||
283 | case DESC92C_RATE2M: | ||
284 | rate_idx = 1; | ||
285 | break; | ||
286 | case DESC92C_RATE5_5M: | ||
287 | rate_idx = 2; | ||
288 | break; | ||
289 | case DESC92C_RATE11M: | ||
290 | rate_idx = 3; | ||
291 | break; | ||
292 | case DESC92C_RATE6M: | ||
293 | rate_idx = 4; | ||
294 | break; | ||
295 | case DESC92C_RATE9M: | ||
296 | rate_idx = 5; | ||
297 | break; | ||
298 | case DESC92C_RATE12M: | ||
299 | rate_idx = 6; | ||
300 | break; | ||
301 | case DESC92C_RATE18M: | ||
302 | rate_idx = 7; | ||
303 | break; | ||
304 | case DESC92C_RATE24M: | ||
305 | rate_idx = 8; | ||
306 | break; | ||
307 | case DESC92C_RATE36M: | ||
308 | rate_idx = 9; | ||
309 | break; | ||
310 | case DESC92C_RATE48M: | ||
311 | rate_idx = 10; | ||
312 | break; | ||
313 | case DESC92C_RATE54M: | ||
314 | rate_idx = 11; | ||
315 | break; | ||
316 | default: | ||
317 | RT_TRACE(rtlpriv, COMP_ERR, DBG_DMESG, | ||
318 | ("Rate %d is not support, set to " | ||
319 | "1M rate.\n", desc_rate)); | ||
320 | rate_idx = 0; | ||
321 | break; | ||
322 | } | ||
323 | } else { | ||
324 | rate_idx = 11; | ||
325 | } | ||
326 | return rate_idx; | ||
327 | } | ||
328 | switch (desc_rate) { | ||
329 | case DESC92C_RATE1M: | ||
330 | rate_idx = 0; | ||
331 | break; | ||
332 | case DESC92C_RATE2M: | ||
333 | rate_idx = 1; | ||
334 | break; | ||
335 | case DESC92C_RATE5_5M: | ||
336 | rate_idx = 2; | ||
337 | break; | ||
338 | case DESC92C_RATE11M: | ||
339 | rate_idx = 3; | ||
340 | break; | ||
341 | case DESC92C_RATE6M: | ||
342 | rate_idx = 4; | ||
343 | break; | ||
344 | case DESC92C_RATE9M: | ||
345 | rate_idx = 5; | ||
346 | break; | ||
347 | case DESC92C_RATE12M: | ||
348 | rate_idx = 6; | ||
349 | break; | ||
350 | case DESC92C_RATE18M: | ||
351 | rate_idx = 7; | ||
352 | break; | ||
353 | case DESC92C_RATE24M: | ||
354 | rate_idx = 8; | ||
355 | break; | ||
356 | case DESC92C_RATE36M: | ||
357 | rate_idx = 9; | ||
358 | break; | ||
359 | case DESC92C_RATE48M: | ||
360 | rate_idx = 10; | ||
361 | break; | ||
362 | case DESC92C_RATE54M: | ||
363 | rate_idx = 11; | ||
364 | break; | ||
365 | /* TODO: How to mapping MCS rate? */ | ||
366 | /* NOTE: referenc to __ieee80211_rx */ | ||
367 | default: | ||
368 | rate_idx = 11; | ||
369 | break; | ||
370 | } | ||
371 | return rate_idx; | ||
372 | } | ||
373 | |||
374 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/dm.c b/drivers/net/wireless/rtlwifi/rtl8192ce/dm.c new file mode 100644 index 000000000000..2df33e53e15a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/dm.c | |||
@@ -0,0 +1,115 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../base.h" | ||
32 | #include "../pci.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "phy.h" | ||
36 | #include "dm.h" | ||
37 | #include "../rtl8192c/fw_common.h" | ||
38 | |||
39 | void rtl92ce_dm_dynamic_txpower(struct ieee80211_hw *hw) | ||
40 | { | ||
41 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
42 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
43 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
44 | long undecorated_smoothed_pwdb; | ||
45 | |||
46 | if (!rtlpriv->dm.dynamic_txpower_enable) | ||
47 | return; | ||
48 | |||
49 | if (rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) { | ||
50 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
51 | return; | ||
52 | } | ||
53 | |||
54 | if ((mac->link_state < MAC80211_LINKED) && | ||
55 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0)) { | ||
56 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
57 | ("Not connected to any\n")); | ||
58 | |||
59 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
60 | |||
61 | rtlpriv->dm.last_dtp_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
62 | return; | ||
63 | } | ||
64 | |||
65 | if (mac->link_state >= MAC80211_LINKED) { | ||
66 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
67 | undecorated_smoothed_pwdb = | ||
68 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
69 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
70 | ("AP Client PWDB = 0x%lx\n", | ||
71 | undecorated_smoothed_pwdb)); | ||
72 | } else { | ||
73 | undecorated_smoothed_pwdb = | ||
74 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
75 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
76 | ("STA Default Port PWDB = 0x%lx\n", | ||
77 | undecorated_smoothed_pwdb)); | ||
78 | } | ||
79 | } else { | ||
80 | undecorated_smoothed_pwdb = | ||
81 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
82 | |||
83 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
84 | ("AP Ext Port PWDB = 0x%lx\n", | ||
85 | undecorated_smoothed_pwdb)); | ||
86 | } | ||
87 | |||
88 | if (undecorated_smoothed_pwdb >= TX_POWER_NEAR_FIELD_THRESH_LVL2) { | ||
89 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
90 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
91 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x0)\n")); | ||
92 | } else if ((undecorated_smoothed_pwdb < | ||
93 | (TX_POWER_NEAR_FIELD_THRESH_LVL2 - 3)) && | ||
94 | (undecorated_smoothed_pwdb >= | ||
95 | TX_POWER_NEAR_FIELD_THRESH_LVL1)) { | ||
96 | |||
97 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
98 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
99 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x10)\n")); | ||
100 | } else if (undecorated_smoothed_pwdb < | ||
101 | (TX_POWER_NEAR_FIELD_THRESH_LVL1 - 5)) { | ||
102 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
103 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
104 | ("TXHIGHPWRLEVEL_NORMAL\n")); | ||
105 | } | ||
106 | |||
107 | if ((rtlpriv->dm.dynamic_txhighpower_lvl != rtlpriv->dm.last_dtp_lvl)) { | ||
108 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
109 | ("PHY_SetTxPowerLevel8192S() Channel = %d\n", | ||
110 | rtlphy->current_channel)); | ||
111 | rtl92c_phy_set_txpower_level(hw, rtlphy->current_channel); | ||
112 | } | ||
113 | |||
114 | rtlpriv->dm.last_dtp_lvl = rtlpriv->dm.dynamic_txhighpower_lvl; | ||
115 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/dm.h b/drivers/net/wireless/rtlwifi/rtl8192ce/dm.h new file mode 100644 index 000000000000..07dd9552e82f --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/dm.h | |||
@@ -0,0 +1,198 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_DM_H__ | ||
31 | #define __RTL92C_DM_H__ | ||
32 | |||
33 | #define HAL_DM_DIG_DISABLE BIT(0) | ||
34 | #define HAL_DM_HIPWR_DISABLE BIT(1) | ||
35 | |||
36 | #define OFDM_TABLE_LENGTH 37 | ||
37 | #define CCK_TABLE_LENGTH 33 | ||
38 | |||
39 | #define OFDM_TABLE_SIZE 37 | ||
40 | #define CCK_TABLE_SIZE 33 | ||
41 | |||
42 | #define BW_AUTO_SWITCH_HIGH_LOW 25 | ||
43 | #define BW_AUTO_SWITCH_LOW_HIGH 30 | ||
44 | |||
45 | #define DM_DIG_THRESH_HIGH 40 | ||
46 | #define DM_DIG_THRESH_LOW 35 | ||
47 | |||
48 | #define DM_FALSEALARM_THRESH_LOW 400 | ||
49 | #define DM_FALSEALARM_THRESH_HIGH 1000 | ||
50 | |||
51 | #define DM_DIG_MAX 0x3e | ||
52 | #define DM_DIG_MIN 0x1e | ||
53 | |||
54 | #define DM_DIG_FA_UPPER 0x32 | ||
55 | #define DM_DIG_FA_LOWER 0x20 | ||
56 | #define DM_DIG_FA_TH0 0x20 | ||
57 | #define DM_DIG_FA_TH1 0x100 | ||
58 | #define DM_DIG_FA_TH2 0x200 | ||
59 | |||
60 | #define DM_DIG_BACKOFF_MAX 12 | ||
61 | #define DM_DIG_BACKOFF_MIN -4 | ||
62 | #define DM_DIG_BACKOFF_DEFAULT 10 | ||
63 | |||
64 | #define RXPATHSELECTION_SS_TH_lOW 30 | ||
65 | #define RXPATHSELECTION_DIFF_TH 18 | ||
66 | |||
67 | #define DM_RATR_STA_INIT 0 | ||
68 | #define DM_RATR_STA_HIGH 1 | ||
69 | #define DM_RATR_STA_MIDDLE 2 | ||
70 | #define DM_RATR_STA_LOW 3 | ||
71 | |||
72 | #define CTS2SELF_THVAL 30 | ||
73 | #define REGC38_TH 20 | ||
74 | |||
75 | #define WAIOTTHVal 25 | ||
76 | |||
77 | #define TXHIGHPWRLEVEL_NORMAL 0 | ||
78 | #define TXHIGHPWRLEVEL_LEVEL1 1 | ||
79 | #define TXHIGHPWRLEVEL_LEVEL2 2 | ||
80 | #define TXHIGHPWRLEVEL_BT1 3 | ||
81 | #define TXHIGHPWRLEVEL_BT2 4 | ||
82 | |||
83 | #define DM_TYPE_BYFW 0 | ||
84 | #define DM_TYPE_BYDRIVER 1 | ||
85 | |||
86 | #define TX_POWER_NEAR_FIELD_THRESH_LVL2 74 | ||
87 | #define TX_POWER_NEAR_FIELD_THRESH_LVL1 67 | ||
88 | |||
89 | struct ps_t { | ||
90 | u8 pre_ccastate; | ||
91 | u8 cur_ccasate; | ||
92 | u8 pre_rfstate; | ||
93 | u8 cur_rfstate; | ||
94 | long rssi_val_min; | ||
95 | }; | ||
96 | |||
97 | struct dig_t { | ||
98 | u8 dig_enable_flag; | ||
99 | u8 dig_ext_port_stage; | ||
100 | u32 rssi_lowthresh; | ||
101 | u32 rssi_highthresh; | ||
102 | u32 fa_lowthresh; | ||
103 | u32 fa_highthresh; | ||
104 | u8 cursta_connectctate; | ||
105 | u8 presta_connectstate; | ||
106 | u8 curmultista_connectstate; | ||
107 | u8 pre_igvalue; | ||
108 | u8 cur_igvalue; | ||
109 | char backoff_val; | ||
110 | char backoff_val_range_max; | ||
111 | char backoff_val_range_min; | ||
112 | u8 rx_gain_range_max; | ||
113 | u8 rx_gain_range_min; | ||
114 | u8 rssi_val_min; | ||
115 | u8 pre_cck_pd_state; | ||
116 | u8 cur_cck_pd_state; | ||
117 | u8 pre_cck_fa_state; | ||
118 | u8 cur_cck_fa_state; | ||
119 | u8 pre_ccastate; | ||
120 | u8 cur_ccasate; | ||
121 | }; | ||
122 | |||
123 | struct swat_t { | ||
124 | u8 failure_cnt; | ||
125 | u8 try_flag; | ||
126 | u8 stop_trying; | ||
127 | long pre_rssi; | ||
128 | long trying_threshold; | ||
129 | u8 cur_antenna; | ||
130 | u8 pre_antenna; | ||
131 | }; | ||
132 | |||
133 | enum tag_dynamic_init_gain_operation_type_definition { | ||
134 | DIG_TYPE_THRESH_HIGH = 0, | ||
135 | DIG_TYPE_THRESH_LOW = 1, | ||
136 | DIG_TYPE_BACKOFF = 2, | ||
137 | DIG_TYPE_RX_GAIN_MIN = 3, | ||
138 | DIG_TYPE_RX_GAIN_MAX = 4, | ||
139 | DIG_TYPE_ENABLE = 5, | ||
140 | DIG_TYPE_DISABLE = 6, | ||
141 | DIG_OP_TYPE_MAX | ||
142 | }; | ||
143 | |||
144 | enum tag_cck_packet_detection_threshold_type_definition { | ||
145 | CCK_PD_STAGE_LowRssi = 0, | ||
146 | CCK_PD_STAGE_HighRssi = 1, | ||
147 | CCK_FA_STAGE_Low = 2, | ||
148 | CCK_FA_STAGE_High = 3, | ||
149 | CCK_PD_STAGE_MAX = 4, | ||
150 | }; | ||
151 | |||
152 | enum dm_1r_cca_e { | ||
153 | CCA_1R = 0, | ||
154 | CCA_2R = 1, | ||
155 | CCA_MAX = 2, | ||
156 | }; | ||
157 | |||
158 | enum dm_rf_e { | ||
159 | RF_SAVE = 0, | ||
160 | RF_NORMAL = 1, | ||
161 | RF_MAX = 2, | ||
162 | }; | ||
163 | |||
164 | enum dm_sw_ant_switch_e { | ||
165 | ANS_ANTENNA_B = 1, | ||
166 | ANS_ANTENNA_A = 2, | ||
167 | ANS_ANTENNA_MAX = 3, | ||
168 | }; | ||
169 | |||
170 | enum dm_dig_ext_port_alg_e { | ||
171 | DIG_EXT_PORT_STAGE_0 = 0, | ||
172 | DIG_EXT_PORT_STAGE_1 = 1, | ||
173 | DIG_EXT_PORT_STAGE_2 = 2, | ||
174 | DIG_EXT_PORT_STAGE_3 = 3, | ||
175 | DIG_EXT_PORT_STAGE_MAX = 4, | ||
176 | }; | ||
177 | |||
178 | enum dm_dig_connect_e { | ||
179 | DIG_STA_DISCONNECT = 0, | ||
180 | DIG_STA_CONNECT = 1, | ||
181 | DIG_STA_BEFORE_CONNECT = 2, | ||
182 | DIG_MULTISTA_DISCONNECT = 3, | ||
183 | DIG_MULTISTA_CONNECT = 4, | ||
184 | DIG_CONNECT_MAX | ||
185 | }; | ||
186 | |||
187 | extern struct dig_t dm_digtable; | ||
188 | void rtl92c_dm_init(struct ieee80211_hw *hw); | ||
189 | void rtl92c_dm_watchdog(struct ieee80211_hw *hw); | ||
190 | void rtl92c_dm_write_dig(struct ieee80211_hw *hw); | ||
191 | void rtl92c_dm_init_edca_turbo(struct ieee80211_hw *hw); | ||
192 | void rtl92c_dm_check_txpower_tracking(struct ieee80211_hw *hw); | ||
193 | void rtl92c_dm_init_rate_adaptive_mask(struct ieee80211_hw *hw); | ||
194 | void rtl92c_dm_rf_saving(struct ieee80211_hw *hw, u8 bforce_in_normal); | ||
195 | void rtl92c_dm_bt_coexist(struct ieee80211_hw *hw); | ||
196 | void rtl92ce_dm_dynamic_txpower(struct ieee80211_hw *hw); | ||
197 | |||
198 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/hw.c b/drivers/net/wireless/rtlwifi/rtl8192ce/hw.c new file mode 100644 index 000000000000..defb4370cf74 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/hw.c | |||
@@ -0,0 +1,2307 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../efuse.h" | ||
32 | #include "../base.h" | ||
33 | #include "../regd.h" | ||
34 | #include "../cam.h" | ||
35 | #include "../ps.h" | ||
36 | #include "../pci.h" | ||
37 | #include "reg.h" | ||
38 | #include "def.h" | ||
39 | #include "phy.h" | ||
40 | #include "../rtl8192c/fw_common.h" | ||
41 | #include "dm.h" | ||
42 | #include "led.h" | ||
43 | #include "hw.h" | ||
44 | |||
45 | #define LLT_CONFIG 5 | ||
46 | |||
47 | static void _rtl92ce_set_bcn_ctrl_reg(struct ieee80211_hw *hw, | ||
48 | u8 set_bits, u8 clear_bits) | ||
49 | { | ||
50 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
51 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
52 | |||
53 | rtlpci->reg_bcn_ctrl_val |= set_bits; | ||
54 | rtlpci->reg_bcn_ctrl_val &= ~clear_bits; | ||
55 | |||
56 | rtl_write_byte(rtlpriv, REG_BCN_CTRL, (u8) rtlpci->reg_bcn_ctrl_val); | ||
57 | } | ||
58 | |||
59 | static void _rtl92ce_stop_tx_beacon(struct ieee80211_hw *hw) | ||
60 | { | ||
61 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
62 | u8 tmp1byte; | ||
63 | |||
64 | tmp1byte = rtl_read_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2); | ||
65 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, tmp1byte & (~BIT(6))); | ||
66 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0x64); | ||
67 | tmp1byte = rtl_read_byte(rtlpriv, REG_TBTT_PROHIBIT + 2); | ||
68 | tmp1byte &= ~(BIT(0)); | ||
69 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 2, tmp1byte); | ||
70 | } | ||
71 | |||
72 | static void _rtl92ce_resume_tx_beacon(struct ieee80211_hw *hw) | ||
73 | { | ||
74 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
75 | u8 tmp1byte; | ||
76 | |||
77 | tmp1byte = rtl_read_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2); | ||
78 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, tmp1byte | BIT(6)); | ||
79 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0xff); | ||
80 | tmp1byte = rtl_read_byte(rtlpriv, REG_TBTT_PROHIBIT + 2); | ||
81 | tmp1byte |= BIT(0); | ||
82 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 2, tmp1byte); | ||
83 | } | ||
84 | |||
85 | static void _rtl92ce_enable_bcn_sub_func(struct ieee80211_hw *hw) | ||
86 | { | ||
87 | _rtl92ce_set_bcn_ctrl_reg(hw, 0, BIT(1)); | ||
88 | } | ||
89 | |||
90 | static void _rtl92ce_disable_bcn_sub_func(struct ieee80211_hw *hw) | ||
91 | { | ||
92 | _rtl92ce_set_bcn_ctrl_reg(hw, BIT(1), 0); | ||
93 | } | ||
94 | |||
95 | void rtl92ce_get_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
96 | { | ||
97 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
98 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
99 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
100 | |||
101 | switch (variable) { | ||
102 | case HW_VAR_RCR: | ||
103 | *((u32 *) (val)) = rtlpci->receive_config; | ||
104 | break; | ||
105 | case HW_VAR_RF_STATE: | ||
106 | *((enum rf_pwrstate *)(val)) = ppsc->rfpwr_state; | ||
107 | break; | ||
108 | case HW_VAR_FWLPS_RF_ON:{ | ||
109 | enum rf_pwrstate rfState; | ||
110 | u32 val_rcr; | ||
111 | |||
112 | rtlpriv->cfg->ops->get_hw_reg(hw, | ||
113 | HW_VAR_RF_STATE, | ||
114 | (u8 *) (&rfState)); | ||
115 | if (rfState == ERFOFF) { | ||
116 | *((bool *) (val)) = true; | ||
117 | } else { | ||
118 | val_rcr = rtl_read_dword(rtlpriv, REG_RCR); | ||
119 | val_rcr &= 0x00070000; | ||
120 | if (val_rcr) | ||
121 | *((bool *) (val)) = false; | ||
122 | else | ||
123 | *((bool *) (val)) = true; | ||
124 | } | ||
125 | break; | ||
126 | } | ||
127 | case HW_VAR_FW_PSMODE_STATUS: | ||
128 | *((bool *) (val)) = ppsc->fw_current_inpsmode; | ||
129 | break; | ||
130 | case HW_VAR_CORRECT_TSF:{ | ||
131 | u64 tsf; | ||
132 | u32 *ptsf_low = (u32 *)&tsf; | ||
133 | u32 *ptsf_high = ((u32 *)&tsf) + 1; | ||
134 | |||
135 | *ptsf_high = rtl_read_dword(rtlpriv, (REG_TSFTR + 4)); | ||
136 | *ptsf_low = rtl_read_dword(rtlpriv, REG_TSFTR); | ||
137 | |||
138 | *((u64 *) (val)) = tsf; | ||
139 | |||
140 | break; | ||
141 | } | ||
142 | default: | ||
143 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
144 | ("switch case not process\n")); | ||
145 | break; | ||
146 | } | ||
147 | } | ||
148 | |||
149 | void rtl92ce_set_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
150 | { | ||
151 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
152 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
153 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
154 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
155 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
156 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
157 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
158 | u8 idx; | ||
159 | |||
160 | switch (variable) { | ||
161 | case HW_VAR_ETHER_ADDR:{ | ||
162 | for (idx = 0; idx < ETH_ALEN; idx++) { | ||
163 | rtl_write_byte(rtlpriv, (REG_MACID + idx), | ||
164 | val[idx]); | ||
165 | } | ||
166 | break; | ||
167 | } | ||
168 | case HW_VAR_BASIC_RATE:{ | ||
169 | u16 rate_cfg = ((u16 *) val)[0]; | ||
170 | u8 rate_index = 0; | ||
171 | rate_cfg &= 0x15f; | ||
172 | rate_cfg |= 0x01; | ||
173 | rtl_write_byte(rtlpriv, REG_RRSR, rate_cfg & 0xff); | ||
174 | rtl_write_byte(rtlpriv, REG_RRSR + 1, | ||
175 | (rate_cfg >> 8) & 0xff); | ||
176 | while (rate_cfg > 0x1) { | ||
177 | rate_cfg = (rate_cfg >> 1); | ||
178 | rate_index++; | ||
179 | } | ||
180 | rtl_write_byte(rtlpriv, REG_INIRTS_RATE_SEL, | ||
181 | rate_index); | ||
182 | break; | ||
183 | } | ||
184 | case HW_VAR_BSSID:{ | ||
185 | for (idx = 0; idx < ETH_ALEN; idx++) { | ||
186 | rtl_write_byte(rtlpriv, (REG_BSSID + idx), | ||
187 | val[idx]); | ||
188 | } | ||
189 | break; | ||
190 | } | ||
191 | case HW_VAR_SIFS:{ | ||
192 | rtl_write_byte(rtlpriv, REG_SIFS_CTX + 1, val[0]); | ||
193 | rtl_write_byte(rtlpriv, REG_SIFS_TRX + 1, val[1]); | ||
194 | |||
195 | rtl_write_byte(rtlpriv, REG_SPEC_SIFS + 1, val[0]); | ||
196 | rtl_write_byte(rtlpriv, REG_MAC_SPEC_SIFS + 1, val[0]); | ||
197 | |||
198 | if (!mac->ht_enable) | ||
199 | rtl_write_word(rtlpriv, REG_RESP_SIFS_OFDM, | ||
200 | 0x0e0e); | ||
201 | else | ||
202 | rtl_write_word(rtlpriv, REG_RESP_SIFS_OFDM, | ||
203 | *((u16 *) val)); | ||
204 | break; | ||
205 | } | ||
206 | case HW_VAR_SLOT_TIME:{ | ||
207 | u8 e_aci; | ||
208 | |||
209 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
210 | ("HW_VAR_SLOT_TIME %x\n", val[0])); | ||
211 | |||
212 | rtl_write_byte(rtlpriv, REG_SLOT, val[0]); | ||
213 | |||
214 | for (e_aci = 0; e_aci < AC_MAX; e_aci++) { | ||
215 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
216 | HW_VAR_AC_PARAM, | ||
217 | (u8 *) (&e_aci)); | ||
218 | } | ||
219 | break; | ||
220 | } | ||
221 | case HW_VAR_ACK_PREAMBLE:{ | ||
222 | u8 reg_tmp; | ||
223 | u8 short_preamble = (bool) (*(u8 *) val); | ||
224 | reg_tmp = (mac->cur_40_prime_sc) << 5; | ||
225 | if (short_preamble) | ||
226 | reg_tmp |= 0x80; | ||
227 | |||
228 | rtl_write_byte(rtlpriv, REG_RRSR + 2, reg_tmp); | ||
229 | break; | ||
230 | } | ||
231 | case HW_VAR_AMPDU_MIN_SPACE:{ | ||
232 | u8 min_spacing_to_set; | ||
233 | u8 sec_min_space; | ||
234 | |||
235 | min_spacing_to_set = *((u8 *) val); | ||
236 | if (min_spacing_to_set <= 7) { | ||
237 | sec_min_space = 0; | ||
238 | |||
239 | if (min_spacing_to_set < sec_min_space) | ||
240 | min_spacing_to_set = sec_min_space; | ||
241 | |||
242 | mac->min_space_cfg = ((mac->min_space_cfg & | ||
243 | 0xf8) | | ||
244 | min_spacing_to_set); | ||
245 | |||
246 | *val = min_spacing_to_set; | ||
247 | |||
248 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
249 | ("Set HW_VAR_AMPDU_MIN_SPACE: %#x\n", | ||
250 | mac->min_space_cfg)); | ||
251 | |||
252 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, | ||
253 | mac->min_space_cfg); | ||
254 | } | ||
255 | break; | ||
256 | } | ||
257 | case HW_VAR_SHORTGI_DENSITY:{ | ||
258 | u8 density_to_set; | ||
259 | |||
260 | density_to_set = *((u8 *) val); | ||
261 | mac->min_space_cfg |= (density_to_set << 3); | ||
262 | |||
263 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
264 | ("Set HW_VAR_SHORTGI_DENSITY: %#x\n", | ||
265 | mac->min_space_cfg)); | ||
266 | |||
267 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, | ||
268 | mac->min_space_cfg); | ||
269 | |||
270 | break; | ||
271 | } | ||
272 | case HW_VAR_AMPDU_FACTOR:{ | ||
273 | u8 regtoset_normal[4] = {0x41, 0xa8, 0x72, 0xb9}; | ||
274 | u8 regtoset_bt[4] = {0x31, 0x74, 0x42, 0x97}; | ||
275 | |||
276 | u8 factor_toset; | ||
277 | u8 *p_regtoset = NULL; | ||
278 | u8 index = 0; | ||
279 | |||
280 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
281 | (rtlpcipriv->bt_coexist.bt_coexist_type == | ||
282 | BT_CSR_BC4)) | ||
283 | p_regtoset = regtoset_bt; | ||
284 | else | ||
285 | p_regtoset = regtoset_normal; | ||
286 | |||
287 | factor_toset = *((u8 *) val); | ||
288 | if (factor_toset <= 3) { | ||
289 | factor_toset = (1 << (factor_toset + 2)); | ||
290 | if (factor_toset > 0xf) | ||
291 | factor_toset = 0xf; | ||
292 | |||
293 | for (index = 0; index < 4; index++) { | ||
294 | if ((p_regtoset[index] & 0xf0) > | ||
295 | (factor_toset << 4)) | ||
296 | p_regtoset[index] = | ||
297 | (p_regtoset[index] & 0x0f) | | ||
298 | (factor_toset << 4); | ||
299 | |||
300 | if ((p_regtoset[index] & 0x0f) > | ||
301 | factor_toset) | ||
302 | p_regtoset[index] = | ||
303 | (p_regtoset[index] & 0xf0) | | ||
304 | (factor_toset); | ||
305 | |||
306 | rtl_write_byte(rtlpriv, | ||
307 | (REG_AGGLEN_LMT + index), | ||
308 | p_regtoset[index]); | ||
309 | |||
310 | } | ||
311 | |||
312 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
313 | ("Set HW_VAR_AMPDU_FACTOR: %#x\n", | ||
314 | factor_toset)); | ||
315 | } | ||
316 | break; | ||
317 | } | ||
318 | case HW_VAR_AC_PARAM:{ | ||
319 | u8 e_aci = *((u8 *) val); | ||
320 | rtl92c_dm_init_edca_turbo(hw); | ||
321 | |||
322 | if (rtlpci->acm_method != eAcmWay2_SW) | ||
323 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
324 | HW_VAR_ACM_CTRL, | ||
325 | (u8 *) (&e_aci)); | ||
326 | break; | ||
327 | } | ||
328 | case HW_VAR_ACM_CTRL:{ | ||
329 | u8 e_aci = *((u8 *) val); | ||
330 | union aci_aifsn *p_aci_aifsn = | ||
331 | (union aci_aifsn *)(&(mac->ac[0].aifs)); | ||
332 | u8 acm = p_aci_aifsn->f.acm; | ||
333 | u8 acm_ctrl = rtl_read_byte(rtlpriv, REG_ACMHWCTRL); | ||
334 | |||
335 | acm_ctrl = | ||
336 | acm_ctrl | ((rtlpci->acm_method == 2) ? 0x0 : 0x1); | ||
337 | |||
338 | if (acm) { | ||
339 | switch (e_aci) { | ||
340 | case AC0_BE: | ||
341 | acm_ctrl |= AcmHw_BeqEn; | ||
342 | break; | ||
343 | case AC2_VI: | ||
344 | acm_ctrl |= AcmHw_ViqEn; | ||
345 | break; | ||
346 | case AC3_VO: | ||
347 | acm_ctrl |= AcmHw_VoqEn; | ||
348 | break; | ||
349 | default: | ||
350 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
351 | ("HW_VAR_ACM_CTRL acm set " | ||
352 | "failed: eACI is %d\n", acm)); | ||
353 | break; | ||
354 | } | ||
355 | } else { | ||
356 | switch (e_aci) { | ||
357 | case AC0_BE: | ||
358 | acm_ctrl &= (~AcmHw_BeqEn); | ||
359 | break; | ||
360 | case AC2_VI: | ||
361 | acm_ctrl &= (~AcmHw_ViqEn); | ||
362 | break; | ||
363 | case AC3_VO: | ||
364 | acm_ctrl &= (~AcmHw_BeqEn); | ||
365 | break; | ||
366 | default: | ||
367 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
368 | ("switch case not process\n")); | ||
369 | break; | ||
370 | } | ||
371 | } | ||
372 | |||
373 | RT_TRACE(rtlpriv, COMP_QOS, DBG_TRACE, | ||
374 | ("SetHwReg8190pci(): [HW_VAR_ACM_CTRL] " | ||
375 | "Write 0x%X\n", acm_ctrl)); | ||
376 | rtl_write_byte(rtlpriv, REG_ACMHWCTRL, acm_ctrl); | ||
377 | break; | ||
378 | } | ||
379 | case HW_VAR_RCR:{ | ||
380 | rtl_write_dword(rtlpriv, REG_RCR, ((u32 *) (val))[0]); | ||
381 | rtlpci->receive_config = ((u32 *) (val))[0]; | ||
382 | break; | ||
383 | } | ||
384 | case HW_VAR_RETRY_LIMIT:{ | ||
385 | u8 retry_limit = ((u8 *) (val))[0]; | ||
386 | |||
387 | rtl_write_word(rtlpriv, REG_RL, | ||
388 | retry_limit << RETRY_LIMIT_SHORT_SHIFT | | ||
389 | retry_limit << RETRY_LIMIT_LONG_SHIFT); | ||
390 | break; | ||
391 | } | ||
392 | case HW_VAR_DUAL_TSF_RST: | ||
393 | rtl_write_byte(rtlpriv, REG_DUAL_TSF_RST, (BIT(0) | BIT(1))); | ||
394 | break; | ||
395 | case HW_VAR_EFUSE_BYTES: | ||
396 | rtlefuse->efuse_usedbytes = *((u16 *) val); | ||
397 | break; | ||
398 | case HW_VAR_EFUSE_USAGE: | ||
399 | rtlefuse->efuse_usedpercentage = *((u8 *) val); | ||
400 | break; | ||
401 | case HW_VAR_IO_CMD: | ||
402 | rtl92c_phy_set_io_cmd(hw, (*(enum io_type *)val)); | ||
403 | break; | ||
404 | case HW_VAR_WPA_CONFIG: | ||
405 | rtl_write_byte(rtlpriv, REG_SECCFG, *((u8 *) val)); | ||
406 | break; | ||
407 | case HW_VAR_SET_RPWM:{ | ||
408 | u8 rpwm_val; | ||
409 | |||
410 | rpwm_val = rtl_read_byte(rtlpriv, REG_PCIE_HRPWM); | ||
411 | udelay(1); | ||
412 | |||
413 | if (rpwm_val & BIT(7)) { | ||
414 | rtl_write_byte(rtlpriv, REG_PCIE_HRPWM, | ||
415 | (*(u8 *) val)); | ||
416 | } else { | ||
417 | rtl_write_byte(rtlpriv, REG_PCIE_HRPWM, | ||
418 | ((*(u8 *) val) | BIT(7))); | ||
419 | } | ||
420 | |||
421 | break; | ||
422 | } | ||
423 | case HW_VAR_H2C_FW_PWRMODE:{ | ||
424 | u8 psmode = (*(u8 *) val); | ||
425 | |||
426 | if ((psmode != FW_PS_ACTIVE_MODE) && | ||
427 | (!IS_92C_SERIAL(rtlhal->version))) { | ||
428 | rtl92c_dm_rf_saving(hw, true); | ||
429 | } | ||
430 | |||
431 | rtl92c_set_fw_pwrmode_cmd(hw, (*(u8 *) val)); | ||
432 | break; | ||
433 | } | ||
434 | case HW_VAR_FW_PSMODE_STATUS: | ||
435 | ppsc->fw_current_inpsmode = *((bool *) val); | ||
436 | break; | ||
437 | case HW_VAR_H2C_FW_JOINBSSRPT:{ | ||
438 | u8 mstatus = (*(u8 *) val); | ||
439 | u8 tmp_regcr, tmp_reg422; | ||
440 | bool recover = false; | ||
441 | |||
442 | if (mstatus == RT_MEDIA_CONNECT) { | ||
443 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_AID, | ||
444 | NULL); | ||
445 | |||
446 | tmp_regcr = rtl_read_byte(rtlpriv, REG_CR + 1); | ||
447 | rtl_write_byte(rtlpriv, REG_CR + 1, | ||
448 | (tmp_regcr | BIT(0))); | ||
449 | |||
450 | _rtl92ce_set_bcn_ctrl_reg(hw, 0, BIT(3)); | ||
451 | _rtl92ce_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
452 | |||
453 | tmp_reg422 = | ||
454 | rtl_read_byte(rtlpriv, | ||
455 | REG_FWHW_TXQ_CTRL + 2); | ||
456 | if (tmp_reg422 & BIT(6)) | ||
457 | recover = true; | ||
458 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, | ||
459 | tmp_reg422 & (~BIT(6))); | ||
460 | |||
461 | rtl92c_set_fw_rsvdpagepkt(hw, 0); | ||
462 | |||
463 | _rtl92ce_set_bcn_ctrl_reg(hw, BIT(3), 0); | ||
464 | _rtl92ce_set_bcn_ctrl_reg(hw, 0, BIT(4)); | ||
465 | |||
466 | if (recover) { | ||
467 | rtl_write_byte(rtlpriv, | ||
468 | REG_FWHW_TXQ_CTRL + 2, | ||
469 | tmp_reg422); | ||
470 | } | ||
471 | |||
472 | rtl_write_byte(rtlpriv, REG_CR + 1, | ||
473 | (tmp_regcr & ~(BIT(0)))); | ||
474 | } | ||
475 | rtl92c_set_fw_joinbss_report_cmd(hw, (*(u8 *) val)); | ||
476 | |||
477 | break; | ||
478 | } | ||
479 | case HW_VAR_AID:{ | ||
480 | u16 u2btmp; | ||
481 | u2btmp = rtl_read_word(rtlpriv, REG_BCN_PSR_RPT); | ||
482 | u2btmp &= 0xC000; | ||
483 | rtl_write_word(rtlpriv, REG_BCN_PSR_RPT, (u2btmp | | ||
484 | mac->assoc_id)); | ||
485 | |||
486 | break; | ||
487 | } | ||
488 | case HW_VAR_CORRECT_TSF:{ | ||
489 | u8 btype_ibss = ((u8 *) (val))[0]; | ||
490 | |||
491 | if (btype_ibss == true) | ||
492 | _rtl92ce_stop_tx_beacon(hw); | ||
493 | |||
494 | _rtl92ce_set_bcn_ctrl_reg(hw, 0, BIT(3)); | ||
495 | |||
496 | rtl_write_dword(rtlpriv, REG_TSFTR, | ||
497 | (u32) (mac->tsf & 0xffffffff)); | ||
498 | rtl_write_dword(rtlpriv, REG_TSFTR + 4, | ||
499 | (u32) ((mac->tsf >> 32) & 0xffffffff)); | ||
500 | |||
501 | _rtl92ce_set_bcn_ctrl_reg(hw, BIT(3), 0); | ||
502 | |||
503 | if (btype_ibss == true) | ||
504 | _rtl92ce_resume_tx_beacon(hw); | ||
505 | |||
506 | break; | ||
507 | |||
508 | } | ||
509 | default: | ||
510 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("switch case " | ||
511 | "not process\n")); | ||
512 | break; | ||
513 | } | ||
514 | } | ||
515 | |||
516 | static bool _rtl92ce_llt_write(struct ieee80211_hw *hw, u32 address, u32 data) | ||
517 | { | ||
518 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
519 | bool status = true; | ||
520 | long count = 0; | ||
521 | u32 value = _LLT_INIT_ADDR(address) | | ||
522 | _LLT_INIT_DATA(data) | _LLT_OP(_LLT_WRITE_ACCESS); | ||
523 | |||
524 | rtl_write_dword(rtlpriv, REG_LLT_INIT, value); | ||
525 | |||
526 | do { | ||
527 | value = rtl_read_dword(rtlpriv, REG_LLT_INIT); | ||
528 | if (_LLT_NO_ACTIVE == _LLT_OP_VALUE(value)) | ||
529 | break; | ||
530 | |||
531 | if (count > POLLING_LLT_THRESHOLD) { | ||
532 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
533 | ("Failed to polling write LLT done at " | ||
534 | "address %d!\n", address)); | ||
535 | status = false; | ||
536 | break; | ||
537 | } | ||
538 | } while (++count); | ||
539 | |||
540 | return status; | ||
541 | } | ||
542 | |||
543 | static bool _rtl92ce_llt_table_init(struct ieee80211_hw *hw) | ||
544 | { | ||
545 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
546 | unsigned short i; | ||
547 | u8 txpktbuf_bndy; | ||
548 | u8 maxPage; | ||
549 | bool status; | ||
550 | |||
551 | #if LLT_CONFIG == 1 | ||
552 | maxPage = 255; | ||
553 | txpktbuf_bndy = 252; | ||
554 | #elif LLT_CONFIG == 2 | ||
555 | maxPage = 127; | ||
556 | txpktbuf_bndy = 124; | ||
557 | #elif LLT_CONFIG == 3 | ||
558 | maxPage = 255; | ||
559 | txpktbuf_bndy = 174; | ||
560 | #elif LLT_CONFIG == 4 | ||
561 | maxPage = 255; | ||
562 | txpktbuf_bndy = 246; | ||
563 | #elif LLT_CONFIG == 5 | ||
564 | maxPage = 255; | ||
565 | txpktbuf_bndy = 246; | ||
566 | #endif | ||
567 | |||
568 | #if LLT_CONFIG == 1 | ||
569 | rtl_write_byte(rtlpriv, REG_RQPN_NPQ, 0x1c); | ||
570 | rtl_write_dword(rtlpriv, REG_RQPN, 0x80a71c1c); | ||
571 | #elif LLT_CONFIG == 2 | ||
572 | rtl_write_dword(rtlpriv, REG_RQPN, 0x845B1010); | ||
573 | #elif LLT_CONFIG == 3 | ||
574 | rtl_write_dword(rtlpriv, REG_RQPN, 0x84838484); | ||
575 | #elif LLT_CONFIG == 4 | ||
576 | rtl_write_dword(rtlpriv, REG_RQPN, 0x80bd1c1c); | ||
577 | #elif LLT_CONFIG == 5 | ||
578 | rtl_write_word(rtlpriv, REG_RQPN_NPQ, 0x0000); | ||
579 | |||
580 | rtl_write_dword(rtlpriv, REG_RQPN, 0x80b01c29); | ||
581 | #endif | ||
582 | |||
583 | rtl_write_dword(rtlpriv, REG_TRXFF_BNDY, (0x27FF0000 | txpktbuf_bndy)); | ||
584 | rtl_write_byte(rtlpriv, REG_TDECTRL + 1, txpktbuf_bndy); | ||
585 | |||
586 | rtl_write_byte(rtlpriv, REG_TXPKTBUF_BCNQ_BDNY, txpktbuf_bndy); | ||
587 | rtl_write_byte(rtlpriv, REG_TXPKTBUF_MGQ_BDNY, txpktbuf_bndy); | ||
588 | |||
589 | rtl_write_byte(rtlpriv, 0x45D, txpktbuf_bndy); | ||
590 | rtl_write_byte(rtlpriv, REG_PBP, 0x11); | ||
591 | rtl_write_byte(rtlpriv, REG_RX_DRVINFO_SZ, 0x4); | ||
592 | |||
593 | for (i = 0; i < (txpktbuf_bndy - 1); i++) { | ||
594 | status = _rtl92ce_llt_write(hw, i, i + 1); | ||
595 | if (true != status) | ||
596 | return status; | ||
597 | } | ||
598 | |||
599 | status = _rtl92ce_llt_write(hw, (txpktbuf_bndy - 1), 0xFF); | ||
600 | if (true != status) | ||
601 | return status; | ||
602 | |||
603 | for (i = txpktbuf_bndy; i < maxPage; i++) { | ||
604 | status = _rtl92ce_llt_write(hw, i, (i + 1)); | ||
605 | if (true != status) | ||
606 | return status; | ||
607 | } | ||
608 | |||
609 | status = _rtl92ce_llt_write(hw, maxPage, txpktbuf_bndy); | ||
610 | if (true != status) | ||
611 | return status; | ||
612 | |||
613 | return true; | ||
614 | } | ||
615 | |||
616 | static void _rtl92ce_gen_refresh_led_state(struct ieee80211_hw *hw) | ||
617 | { | ||
618 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
619 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
620 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
621 | struct rtl_led *pLed0 = &(pcipriv->ledctl.sw_led0); | ||
622 | |||
623 | if (rtlpci->up_first_time) | ||
624 | return; | ||
625 | |||
626 | if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS) | ||
627 | rtl92ce_sw_led_on(hw, pLed0); | ||
628 | else if (ppsc->rfoff_reason == RF_CHANGE_BY_INIT) | ||
629 | rtl92ce_sw_led_on(hw, pLed0); | ||
630 | else | ||
631 | rtl92ce_sw_led_off(hw, pLed0); | ||
632 | } | ||
633 | |||
634 | static bool _rtl92ce_init_mac(struct ieee80211_hw *hw) | ||
635 | { | ||
636 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
637 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
638 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
639 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
640 | |||
641 | unsigned char bytetmp; | ||
642 | unsigned short wordtmp; | ||
643 | u16 retry; | ||
644 | |||
645 | rtl_write_byte(rtlpriv, REG_RSV_CTRL, 0x00); | ||
646 | if (rtlpcipriv->bt_coexist.bt_coexistence) { | ||
647 | u32 value32; | ||
648 | value32 = rtl_read_dword(rtlpriv, REG_APS_FSMCO); | ||
649 | value32 |= (SOP_ABG | SOP_AMB | XOP_BTCK); | ||
650 | rtl_write_dword(rtlpriv, REG_APS_FSMCO, value32); | ||
651 | } | ||
652 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x2b); | ||
653 | rtl_write_byte(rtlpriv, REG_AFE_XTAL_CTRL, 0x0F); | ||
654 | |||
655 | if (rtlpcipriv->bt_coexist.bt_coexistence) { | ||
656 | u32 u4b_tmp = rtl_read_dword(rtlpriv, REG_AFE_XTAL_CTRL); | ||
657 | |||
658 | u4b_tmp &= (~0x00024800); | ||
659 | rtl_write_dword(rtlpriv, REG_AFE_XTAL_CTRL, u4b_tmp); | ||
660 | } | ||
661 | |||
662 | bytetmp = rtl_read_byte(rtlpriv, REG_APS_FSMCO + 1) | BIT(0); | ||
663 | udelay(2); | ||
664 | |||
665 | rtl_write_byte(rtlpriv, REG_APS_FSMCO + 1, bytetmp); | ||
666 | udelay(2); | ||
667 | |||
668 | bytetmp = rtl_read_byte(rtlpriv, REG_APS_FSMCO + 1); | ||
669 | udelay(2); | ||
670 | |||
671 | retry = 0; | ||
672 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("reg0xec:%x:%x\n", | ||
673 | rtl_read_dword(rtlpriv, 0xEC), | ||
674 | bytetmp)); | ||
675 | |||
676 | while ((bytetmp & BIT(0)) && retry < 1000) { | ||
677 | retry++; | ||
678 | udelay(50); | ||
679 | bytetmp = rtl_read_byte(rtlpriv, REG_APS_FSMCO + 1); | ||
680 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("reg0xec:%x:%x\n", | ||
681 | rtl_read_dword(rtlpriv, | ||
682 | 0xEC), | ||
683 | bytetmp)); | ||
684 | udelay(50); | ||
685 | } | ||
686 | |||
687 | rtl_write_word(rtlpriv, REG_APS_FSMCO, 0x1012); | ||
688 | |||
689 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL + 1, 0x82); | ||
690 | udelay(2); | ||
691 | |||
692 | if (rtlpcipriv->bt_coexist.bt_coexistence) { | ||
693 | bytetmp = rtl_read_byte(rtlpriv, REG_AFE_XTAL_CTRL+2) & 0xfd; | ||
694 | rtl_write_byte(rtlpriv, REG_AFE_XTAL_CTRL+2, bytetmp); | ||
695 | } | ||
696 | |||
697 | rtl_write_word(rtlpriv, REG_CR, 0x2ff); | ||
698 | |||
699 | if (_rtl92ce_llt_table_init(hw) == false) | ||
700 | return false; | ||
701 | |||
702 | rtl_write_dword(rtlpriv, REG_HISR, 0xffffffff); | ||
703 | rtl_write_byte(rtlpriv, REG_HISRE, 0xff); | ||
704 | |||
705 | rtl_write_word(rtlpriv, REG_TRXFF_BNDY + 2, 0x27ff); | ||
706 | |||
707 | wordtmp = rtl_read_word(rtlpriv, REG_TRXDMA_CTRL); | ||
708 | wordtmp &= 0xf; | ||
709 | wordtmp |= 0xF771; | ||
710 | rtl_write_word(rtlpriv, REG_TRXDMA_CTRL, wordtmp); | ||
711 | |||
712 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 1, 0x1F); | ||
713 | rtl_write_dword(rtlpriv, REG_RCR, rtlpci->receive_config); | ||
714 | rtl_write_dword(rtlpriv, REG_TCR, rtlpci->transmit_config); | ||
715 | |||
716 | rtl_write_byte(rtlpriv, 0x4d0, 0x0); | ||
717 | |||
718 | rtl_write_dword(rtlpriv, REG_BCNQ_DESA, | ||
719 | ((u64) rtlpci->tx_ring[BEACON_QUEUE].dma) & | ||
720 | DMA_BIT_MASK(32)); | ||
721 | rtl_write_dword(rtlpriv, REG_MGQ_DESA, | ||
722 | (u64) rtlpci->tx_ring[MGNT_QUEUE].dma & | ||
723 | DMA_BIT_MASK(32)); | ||
724 | rtl_write_dword(rtlpriv, REG_VOQ_DESA, | ||
725 | (u64) rtlpci->tx_ring[VO_QUEUE].dma & DMA_BIT_MASK(32)); | ||
726 | rtl_write_dword(rtlpriv, REG_VIQ_DESA, | ||
727 | (u64) rtlpci->tx_ring[VI_QUEUE].dma & DMA_BIT_MASK(32)); | ||
728 | rtl_write_dword(rtlpriv, REG_BEQ_DESA, | ||
729 | (u64) rtlpci->tx_ring[BE_QUEUE].dma & DMA_BIT_MASK(32)); | ||
730 | rtl_write_dword(rtlpriv, REG_BKQ_DESA, | ||
731 | (u64) rtlpci->tx_ring[BK_QUEUE].dma & DMA_BIT_MASK(32)); | ||
732 | rtl_write_dword(rtlpriv, REG_HQ_DESA, | ||
733 | (u64) rtlpci->tx_ring[HIGH_QUEUE].dma & | ||
734 | DMA_BIT_MASK(32)); | ||
735 | rtl_write_dword(rtlpriv, REG_RX_DESA, | ||
736 | (u64) rtlpci->rx_ring[RX_MPDU_QUEUE].dma & | ||
737 | DMA_BIT_MASK(32)); | ||
738 | |||
739 | if (IS_92C_SERIAL(rtlhal->version)) | ||
740 | rtl_write_byte(rtlpriv, REG_PCIE_CTRL_REG + 3, 0x77); | ||
741 | else | ||
742 | rtl_write_byte(rtlpriv, REG_PCIE_CTRL_REG + 3, 0x22); | ||
743 | |||
744 | rtl_write_dword(rtlpriv, REG_INT_MIG, 0); | ||
745 | |||
746 | bytetmp = rtl_read_byte(rtlpriv, REG_APSD_CTRL); | ||
747 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, bytetmp & ~BIT(6)); | ||
748 | do { | ||
749 | retry++; | ||
750 | bytetmp = rtl_read_byte(rtlpriv, REG_APSD_CTRL); | ||
751 | } while ((retry < 200) && (bytetmp & BIT(7))); | ||
752 | |||
753 | _rtl92ce_gen_refresh_led_state(hw); | ||
754 | |||
755 | rtl_write_dword(rtlpriv, REG_MCUTST_1, 0x0); | ||
756 | |||
757 | return true; | ||
758 | } | ||
759 | |||
760 | static void _rtl92ce_hw_configure(struct ieee80211_hw *hw) | ||
761 | { | ||
762 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
763 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
764 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
765 | u8 reg_bw_opmode; | ||
766 | u32 reg_ratr, reg_prsr; | ||
767 | |||
768 | reg_bw_opmode = BW_OPMODE_20MHZ; | ||
769 | reg_ratr = RATE_ALL_CCK | RATE_ALL_OFDM_AG | | ||
770 | RATE_ALL_OFDM_1SS | RATE_ALL_OFDM_2SS; | ||
771 | reg_prsr = RATE_ALL_CCK | RATE_ALL_OFDM_AG; | ||
772 | |||
773 | rtl_write_byte(rtlpriv, REG_INIRTS_RATE_SEL, 0x8); | ||
774 | |||
775 | rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode); | ||
776 | |||
777 | rtl_write_dword(rtlpriv, REG_RRSR, reg_prsr); | ||
778 | |||
779 | rtl_write_byte(rtlpriv, REG_SLOT, 0x09); | ||
780 | |||
781 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, 0x0); | ||
782 | |||
783 | rtl_write_word(rtlpriv, REG_FWHW_TXQ_CTRL, 0x1F80); | ||
784 | |||
785 | rtl_write_word(rtlpriv, REG_RL, 0x0707); | ||
786 | |||
787 | rtl_write_dword(rtlpriv, REG_BAR_MODE_CTRL, 0x02012802); | ||
788 | |||
789 | rtl_write_byte(rtlpriv, REG_HWSEQ_CTRL, 0xFF); | ||
790 | |||
791 | rtl_write_dword(rtlpriv, REG_DARFRC, 0x01000000); | ||
792 | rtl_write_dword(rtlpriv, REG_DARFRC + 4, 0x07060504); | ||
793 | rtl_write_dword(rtlpriv, REG_RARFRC, 0x01000000); | ||
794 | rtl_write_dword(rtlpriv, REG_RARFRC + 4, 0x07060504); | ||
795 | |||
796 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
797 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4)) | ||
798 | rtl_write_dword(rtlpriv, REG_AGGLEN_LMT, 0x97427431); | ||
799 | else | ||
800 | rtl_write_dword(rtlpriv, REG_AGGLEN_LMT, 0xb972a841); | ||
801 | |||
802 | rtl_write_byte(rtlpriv, REG_ATIMWND, 0x2); | ||
803 | |||
804 | rtl_write_byte(rtlpriv, REG_BCN_MAX_ERR, 0xff); | ||
805 | |||
806 | rtlpci->reg_bcn_ctrl_val = 0x1f; | ||
807 | rtl_write_byte(rtlpriv, REG_BCN_CTRL, rtlpci->reg_bcn_ctrl_val); | ||
808 | |||
809 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0xff); | ||
810 | |||
811 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0xff); | ||
812 | |||
813 | rtl_write_byte(rtlpriv, REG_PIFS, 0x1C); | ||
814 | rtl_write_byte(rtlpriv, REG_AGGR_BREAK_TIME, 0x16); | ||
815 | |||
816 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
817 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4)) { | ||
818 | rtl_write_word(rtlpriv, REG_NAV_PROT_LEN, 0x0020); | ||
819 | rtl_write_word(rtlpriv, REG_PROT_MODE_CTRL, 0x0402); | ||
820 | } else { | ||
821 | rtl_write_word(rtlpriv, REG_NAV_PROT_LEN, 0x0020); | ||
822 | rtl_write_word(rtlpriv, REG_NAV_PROT_LEN, 0x0020); | ||
823 | } | ||
824 | |||
825 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
826 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4)) | ||
827 | rtl_write_dword(rtlpriv, REG_FAST_EDCA_CTRL, 0x03086666); | ||
828 | else | ||
829 | rtl_write_dword(rtlpriv, REG_FAST_EDCA_CTRL, 0x086666); | ||
830 | |||
831 | rtl_write_byte(rtlpriv, REG_ACKTO, 0x40); | ||
832 | |||
833 | rtl_write_word(rtlpriv, REG_SPEC_SIFS, 0x1010); | ||
834 | rtl_write_word(rtlpriv, REG_MAC_SPEC_SIFS, 0x1010); | ||
835 | |||
836 | rtl_write_word(rtlpriv, REG_SIFS_CTX, 0x1010); | ||
837 | |||
838 | rtl_write_word(rtlpriv, REG_SIFS_TRX, 0x1010); | ||
839 | |||
840 | rtl_write_dword(rtlpriv, REG_MAR, 0xffffffff); | ||
841 | rtl_write_dword(rtlpriv, REG_MAR + 4, 0xffffffff); | ||
842 | |||
843 | } | ||
844 | |||
845 | static void _rtl92ce_enable_aspm_back_door(struct ieee80211_hw *hw) | ||
846 | { | ||
847 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
848 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
849 | |||
850 | rtl_write_byte(rtlpriv, 0x34b, 0x93); | ||
851 | rtl_write_word(rtlpriv, 0x350, 0x870c); | ||
852 | rtl_write_byte(rtlpriv, 0x352, 0x1); | ||
853 | |||
854 | if (ppsc->support_backdoor) | ||
855 | rtl_write_byte(rtlpriv, 0x349, 0x1b); | ||
856 | else | ||
857 | rtl_write_byte(rtlpriv, 0x349, 0x03); | ||
858 | |||
859 | rtl_write_word(rtlpriv, 0x350, 0x2718); | ||
860 | rtl_write_byte(rtlpriv, 0x352, 0x1); | ||
861 | } | ||
862 | |||
863 | void rtl92ce_enable_hw_security_config(struct ieee80211_hw *hw) | ||
864 | { | ||
865 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
866 | u8 sec_reg_value; | ||
867 | |||
868 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
869 | ("PairwiseEncAlgorithm = %d GroupEncAlgorithm = %d\n", | ||
870 | rtlpriv->sec.pairwise_enc_algorithm, | ||
871 | rtlpriv->sec.group_enc_algorithm)); | ||
872 | |||
873 | if (rtlpriv->cfg->mod_params->sw_crypto || rtlpriv->sec.use_sw_sec) { | ||
874 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("not open " | ||
875 | "hw encryption\n")); | ||
876 | return; | ||
877 | } | ||
878 | |||
879 | sec_reg_value = SCR_TxEncEnable | SCR_RxDecEnable; | ||
880 | |||
881 | if (rtlpriv->sec.use_defaultkey) { | ||
882 | sec_reg_value |= SCR_TxUseDK; | ||
883 | sec_reg_value |= SCR_RxUseDK; | ||
884 | } | ||
885 | |||
886 | sec_reg_value |= (SCR_RXBCUSEDK | SCR_TXBCUSEDK); | ||
887 | |||
888 | rtl_write_byte(rtlpriv, REG_CR + 1, 0x02); | ||
889 | |||
890 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
891 | ("The SECR-value %x\n", sec_reg_value)); | ||
892 | |||
893 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_WPA_CONFIG, &sec_reg_value); | ||
894 | |||
895 | } | ||
896 | |||
897 | int rtl92ce_hw_init(struct ieee80211_hw *hw) | ||
898 | { | ||
899 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
900 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
901 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
902 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
903 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
904 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
905 | static bool iqk_initialized; /* initialized to false */ | ||
906 | bool rtstatus = true; | ||
907 | bool is92c; | ||
908 | int err; | ||
909 | u8 tmp_u1b; | ||
910 | |||
911 | rtlpci->being_init_adapter = true; | ||
912 | rtlpriv->intf_ops->disable_aspm(hw); | ||
913 | rtstatus = _rtl92ce_init_mac(hw); | ||
914 | if (rtstatus != true) { | ||
915 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("Init MAC failed\n")); | ||
916 | err = 1; | ||
917 | return err; | ||
918 | } | ||
919 | |||
920 | err = rtl92c_download_fw(hw); | ||
921 | if (err) { | ||
922 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
923 | ("Failed to download FW. Init HW " | ||
924 | "without FW now..\n")); | ||
925 | err = 1; | ||
926 | rtlhal->fw_ready = false; | ||
927 | return err; | ||
928 | } else { | ||
929 | rtlhal->fw_ready = true; | ||
930 | } | ||
931 | |||
932 | rtlhal->last_hmeboxnum = 0; | ||
933 | rtl92c_phy_mac_config(hw); | ||
934 | rtl92c_phy_bb_config(hw); | ||
935 | rtlphy->rf_mode = RF_OP_BY_SW_3WIRE; | ||
936 | rtl92c_phy_rf_config(hw); | ||
937 | rtlphy->rfreg_chnlval[0] = rtl_get_rfreg(hw, (enum radio_path)0, | ||
938 | RF_CHNLBW, RFREG_OFFSET_MASK); | ||
939 | rtlphy->rfreg_chnlval[1] = rtl_get_rfreg(hw, (enum radio_path)1, | ||
940 | RF_CHNLBW, RFREG_OFFSET_MASK); | ||
941 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BCCKEN, 0x1); | ||
942 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BOFDMEN, 0x1); | ||
943 | rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 1); | ||
944 | _rtl92ce_hw_configure(hw); | ||
945 | rtl_cam_reset_all_entry(hw); | ||
946 | rtl92ce_enable_hw_security_config(hw); | ||
947 | |||
948 | ppsc->rfpwr_state = ERFON; | ||
949 | |||
950 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_ETHER_ADDR, mac->mac_addr); | ||
951 | _rtl92ce_enable_aspm_back_door(hw); | ||
952 | rtlpriv->intf_ops->enable_aspm(hw); | ||
953 | |||
954 | rtl8192ce_bt_hw_init(hw); | ||
955 | |||
956 | if (ppsc->rfpwr_state == ERFON) { | ||
957 | rtl92c_phy_set_rfpath_switch(hw, 1); | ||
958 | if (iqk_initialized) { | ||
959 | rtl92c_phy_iq_calibrate(hw, true); | ||
960 | } else { | ||
961 | rtl92c_phy_iq_calibrate(hw, false); | ||
962 | iqk_initialized = true; | ||
963 | } | ||
964 | |||
965 | rtl92c_dm_check_txpower_tracking(hw); | ||
966 | rtl92c_phy_lc_calibrate(hw); | ||
967 | } | ||
968 | |||
969 | is92c = IS_92C_SERIAL(rtlhal->version); | ||
970 | tmp_u1b = efuse_read_1byte(hw, 0x1FA); | ||
971 | if (!(tmp_u1b & BIT(0))) { | ||
972 | rtl_set_rfreg(hw, RF90_PATH_A, 0x15, 0x0F, 0x05); | ||
973 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("PA BIAS path A\n")); | ||
974 | } | ||
975 | |||
976 | if (!(tmp_u1b & BIT(1)) && is92c) { | ||
977 | rtl_set_rfreg(hw, RF90_PATH_B, 0x15, 0x0F, 0x05); | ||
978 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("PA BIAS path B\n")); | ||
979 | } | ||
980 | |||
981 | if (!(tmp_u1b & BIT(4))) { | ||
982 | tmp_u1b = rtl_read_byte(rtlpriv, 0x16); | ||
983 | tmp_u1b &= 0x0F; | ||
984 | rtl_write_byte(rtlpriv, 0x16, tmp_u1b | 0x80); | ||
985 | udelay(10); | ||
986 | rtl_write_byte(rtlpriv, 0x16, tmp_u1b | 0x90); | ||
987 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("under 1.5V\n")); | ||
988 | } | ||
989 | rtl92c_dm_init(hw); | ||
990 | rtlpci->being_init_adapter = false; | ||
991 | return err; | ||
992 | } | ||
993 | |||
994 | static enum version_8192c _rtl92ce_read_chip_version(struct ieee80211_hw *hw) | ||
995 | { | ||
996 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
997 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
998 | enum version_8192c version = VERSION_UNKNOWN; | ||
999 | u32 value32; | ||
1000 | |||
1001 | value32 = rtl_read_dword(rtlpriv, REG_SYS_CFG); | ||
1002 | if (value32 & TRP_VAUX_EN) { | ||
1003 | version = (value32 & TYPE_ID) ? VERSION_A_CHIP_92C : | ||
1004 | VERSION_A_CHIP_88C; | ||
1005 | } else { | ||
1006 | version = (value32 & TYPE_ID) ? VERSION_B_CHIP_92C : | ||
1007 | VERSION_B_CHIP_88C; | ||
1008 | } | ||
1009 | |||
1010 | switch (version) { | ||
1011 | case VERSION_B_CHIP_92C: | ||
1012 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1013 | ("Chip Version ID: VERSION_B_CHIP_92C.\n")); | ||
1014 | break; | ||
1015 | case VERSION_B_CHIP_88C: | ||
1016 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1017 | ("Chip Version ID: VERSION_B_CHIP_88C.\n")); | ||
1018 | break; | ||
1019 | case VERSION_A_CHIP_92C: | ||
1020 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1021 | ("Chip Version ID: VERSION_A_CHIP_92C.\n")); | ||
1022 | break; | ||
1023 | case VERSION_A_CHIP_88C: | ||
1024 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1025 | ("Chip Version ID: VERSION_A_CHIP_88C.\n")); | ||
1026 | break; | ||
1027 | default: | ||
1028 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1029 | ("Chip Version ID: Unknown. Bug?\n")); | ||
1030 | break; | ||
1031 | } | ||
1032 | |||
1033 | switch (version & 0x3) { | ||
1034 | case CHIP_88C: | ||
1035 | rtlphy->rf_type = RF_1T1R; | ||
1036 | break; | ||
1037 | case CHIP_92C: | ||
1038 | rtlphy->rf_type = RF_2T2R; | ||
1039 | break; | ||
1040 | case CHIP_92C_1T2R: | ||
1041 | rtlphy->rf_type = RF_1T2R; | ||
1042 | break; | ||
1043 | default: | ||
1044 | rtlphy->rf_type = RF_1T1R; | ||
1045 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1046 | ("ERROR RF_Type is set!!")); | ||
1047 | break; | ||
1048 | } | ||
1049 | |||
1050 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1051 | ("Chip RF Type: %s\n", (rtlphy->rf_type == RF_2T2R) ? | ||
1052 | "RF_2T2R" : "RF_1T1R")); | ||
1053 | |||
1054 | return version; | ||
1055 | } | ||
1056 | |||
1057 | static int _rtl92ce_set_media_status(struct ieee80211_hw *hw, | ||
1058 | enum nl80211_iftype type) | ||
1059 | { | ||
1060 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1061 | u8 bt_msr = rtl_read_byte(rtlpriv, MSR); | ||
1062 | enum led_ctl_mode ledaction = LED_CTL_NO_LINK; | ||
1063 | bt_msr &= 0xfc; | ||
1064 | |||
1065 | if (type == NL80211_IFTYPE_UNSPECIFIED || | ||
1066 | type == NL80211_IFTYPE_STATION) { | ||
1067 | _rtl92ce_stop_tx_beacon(hw); | ||
1068 | _rtl92ce_enable_bcn_sub_func(hw); | ||
1069 | } else if (type == NL80211_IFTYPE_ADHOC || type == NL80211_IFTYPE_AP) { | ||
1070 | _rtl92ce_resume_tx_beacon(hw); | ||
1071 | _rtl92ce_disable_bcn_sub_func(hw); | ||
1072 | } else { | ||
1073 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1074 | ("Set HW_VAR_MEDIA_STATUS: " | ||
1075 | "No such media status(%x).\n", type)); | ||
1076 | } | ||
1077 | |||
1078 | switch (type) { | ||
1079 | case NL80211_IFTYPE_UNSPECIFIED: | ||
1080 | bt_msr |= MSR_NOLINK; | ||
1081 | ledaction = LED_CTL_LINK; | ||
1082 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1083 | ("Set Network type to NO LINK!\n")); | ||
1084 | break; | ||
1085 | case NL80211_IFTYPE_ADHOC: | ||
1086 | bt_msr |= MSR_ADHOC; | ||
1087 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1088 | ("Set Network type to Ad Hoc!\n")); | ||
1089 | break; | ||
1090 | case NL80211_IFTYPE_STATION: | ||
1091 | bt_msr |= MSR_INFRA; | ||
1092 | ledaction = LED_CTL_LINK; | ||
1093 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1094 | ("Set Network type to STA!\n")); | ||
1095 | break; | ||
1096 | case NL80211_IFTYPE_AP: | ||
1097 | bt_msr |= MSR_AP; | ||
1098 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1099 | ("Set Network type to AP!\n")); | ||
1100 | break; | ||
1101 | default: | ||
1102 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1103 | ("Network type %d not support!\n", type)); | ||
1104 | return 1; | ||
1105 | break; | ||
1106 | |||
1107 | } | ||
1108 | |||
1109 | rtl_write_byte(rtlpriv, (MSR), bt_msr); | ||
1110 | rtlpriv->cfg->ops->led_control(hw, ledaction); | ||
1111 | if ((bt_msr & 0xfc) == MSR_AP) | ||
1112 | rtl_write_byte(rtlpriv, REG_BCNTCFG + 1, 0x00); | ||
1113 | else | ||
1114 | rtl_write_byte(rtlpriv, REG_BCNTCFG + 1, 0x66); | ||
1115 | return 0; | ||
1116 | } | ||
1117 | |||
1118 | void rtl92ce_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid) | ||
1119 | { | ||
1120 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1121 | u32 reg_rcr = rtl_read_dword(rtlpriv, REG_RCR); | ||
1122 | |||
1123 | if (rtlpriv->psc.rfpwr_state != ERFON) | ||
1124 | return; | ||
1125 | |||
1126 | if (check_bssid == true) { | ||
1127 | reg_rcr |= (RCR_CBSSID_DATA | RCR_CBSSID_BCN); | ||
1128 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, | ||
1129 | (u8 *) (®_rcr)); | ||
1130 | _rtl92ce_set_bcn_ctrl_reg(hw, 0, BIT(4)); | ||
1131 | } else if (check_bssid == false) { | ||
1132 | reg_rcr &= (~(RCR_CBSSID_DATA | RCR_CBSSID_BCN)); | ||
1133 | _rtl92ce_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
1134 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
1135 | HW_VAR_RCR, (u8 *) (®_rcr)); | ||
1136 | } | ||
1137 | |||
1138 | } | ||
1139 | |||
1140 | int rtl92ce_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type) | ||
1141 | { | ||
1142 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1143 | |||
1144 | if (_rtl92ce_set_media_status(hw, type)) | ||
1145 | return -EOPNOTSUPP; | ||
1146 | |||
1147 | if (rtlpriv->mac80211.link_state == MAC80211_LINKED) { | ||
1148 | if (type != NL80211_IFTYPE_AP) | ||
1149 | rtl92ce_set_check_bssid(hw, true); | ||
1150 | } else { | ||
1151 | rtl92ce_set_check_bssid(hw, false); | ||
1152 | } | ||
1153 | |||
1154 | return 0; | ||
1155 | } | ||
1156 | |||
1157 | /* don't set REG_EDCA_BE_PARAM here because mac80211 will send pkt when scan */ | ||
1158 | void rtl92ce_set_qos(struct ieee80211_hw *hw, int aci) | ||
1159 | { | ||
1160 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1161 | rtl92c_dm_init_edca_turbo(hw); | ||
1162 | switch (aci) { | ||
1163 | case AC1_BK: | ||
1164 | rtl_write_dword(rtlpriv, REG_EDCA_BK_PARAM, 0xa44f); | ||
1165 | break; | ||
1166 | case AC0_BE: | ||
1167 | /* rtl_write_dword(rtlpriv, REG_EDCA_BE_PARAM, u4b_ac_param); */ | ||
1168 | break; | ||
1169 | case AC2_VI: | ||
1170 | rtl_write_dword(rtlpriv, REG_EDCA_VI_PARAM, 0x5e4322); | ||
1171 | break; | ||
1172 | case AC3_VO: | ||
1173 | rtl_write_dword(rtlpriv, REG_EDCA_VO_PARAM, 0x2f3222); | ||
1174 | break; | ||
1175 | default: | ||
1176 | RT_ASSERT(false, ("invalid aci: %d !\n", aci)); | ||
1177 | break; | ||
1178 | } | ||
1179 | } | ||
1180 | |||
1181 | void rtl92ce_enable_interrupt(struct ieee80211_hw *hw) | ||
1182 | { | ||
1183 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1184 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1185 | |||
1186 | rtl_write_dword(rtlpriv, REG_HIMR, rtlpci->irq_mask[0] & 0xFFFFFFFF); | ||
1187 | rtl_write_dword(rtlpriv, REG_HIMRE, rtlpci->irq_mask[1] & 0xFFFFFFFF); | ||
1188 | rtlpci->irq_enabled = true; | ||
1189 | } | ||
1190 | |||
1191 | void rtl92ce_disable_interrupt(struct ieee80211_hw *hw) | ||
1192 | { | ||
1193 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1194 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1195 | |||
1196 | rtl_write_dword(rtlpriv, REG_HIMR, IMR8190_DISABLED); | ||
1197 | rtl_write_dword(rtlpriv, REG_HIMRE, IMR8190_DISABLED); | ||
1198 | rtlpci->irq_enabled = false; | ||
1199 | } | ||
1200 | |||
1201 | static void _rtl92ce_poweroff_adapter(struct ieee80211_hw *hw) | ||
1202 | { | ||
1203 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1204 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1205 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1206 | u8 u1b_tmp; | ||
1207 | u32 u4b_tmp; | ||
1208 | |||
1209 | rtlpriv->intf_ops->enable_aspm(hw); | ||
1210 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
1211 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, RFREG_OFFSET_MASK, 0x00); | ||
1212 | rtl_write_byte(rtlpriv, REG_RF_CTRL, 0x00); | ||
1213 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x40); | ||
1214 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
1215 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE0); | ||
1216 | if ((rtl_read_byte(rtlpriv, REG_MCUFWDL) & BIT(7)) && rtlhal->fw_ready) | ||
1217 | rtl92c_firmware_selfreset(hw); | ||
1218 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, 0x51); | ||
1219 | rtl_write_byte(rtlpriv, REG_MCUFWDL, 0x00); | ||
1220 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, 0x00000000); | ||
1221 | u1b_tmp = rtl_read_byte(rtlpriv, REG_GPIO_PIN_CTRL); | ||
1222 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
1223 | ((rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4) || | ||
1224 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC8))) { | ||
1225 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, 0x00F30000 | | ||
1226 | (u1b_tmp << 8)); | ||
1227 | } else { | ||
1228 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, 0x00FF0000 | | ||
1229 | (u1b_tmp << 8)); | ||
1230 | } | ||
1231 | rtl_write_word(rtlpriv, REG_GPIO_IO_SEL, 0x0790); | ||
1232 | rtl_write_word(rtlpriv, REG_LEDCFG0, 0x8080); | ||
1233 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL, 0x80); | ||
1234 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x23); | ||
1235 | if (rtlpcipriv->bt_coexist.bt_coexistence) { | ||
1236 | u4b_tmp = rtl_read_dword(rtlpriv, REG_AFE_XTAL_CTRL); | ||
1237 | u4b_tmp |= 0x03824800; | ||
1238 | rtl_write_dword(rtlpriv, REG_AFE_XTAL_CTRL, u4b_tmp); | ||
1239 | } else { | ||
1240 | rtl_write_dword(rtlpriv, REG_AFE_XTAL_CTRL, 0x0e); | ||
1241 | } | ||
1242 | |||
1243 | rtl_write_byte(rtlpriv, REG_RSV_CTRL, 0x0e); | ||
1244 | rtl_write_byte(rtlpriv, REG_APS_FSMCO + 1, 0x10); | ||
1245 | } | ||
1246 | |||
1247 | void rtl92ce_card_disable(struct ieee80211_hw *hw) | ||
1248 | { | ||
1249 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1250 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1251 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1252 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1253 | enum nl80211_iftype opmode; | ||
1254 | |||
1255 | mac->link_state = MAC80211_NOLINK; | ||
1256 | opmode = NL80211_IFTYPE_UNSPECIFIED; | ||
1257 | _rtl92ce_set_media_status(hw, opmode); | ||
1258 | if (rtlpci->driver_is_goingto_unload || | ||
1259 | ppsc->rfoff_reason > RF_CHANGE_BY_PS) | ||
1260 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_POWER_OFF); | ||
1261 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
1262 | _rtl92ce_poweroff_adapter(hw); | ||
1263 | } | ||
1264 | |||
1265 | void rtl92ce_interrupt_recognized(struct ieee80211_hw *hw, | ||
1266 | u32 *p_inta, u32 *p_intb) | ||
1267 | { | ||
1268 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1269 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1270 | |||
1271 | *p_inta = rtl_read_dword(rtlpriv, ISR) & rtlpci->irq_mask[0]; | ||
1272 | rtl_write_dword(rtlpriv, ISR, *p_inta); | ||
1273 | |||
1274 | /* | ||
1275 | * *p_intb = rtl_read_dword(rtlpriv, REG_HISRE) & rtlpci->irq_mask[1]; | ||
1276 | * rtl_write_dword(rtlpriv, ISR + 4, *p_intb); | ||
1277 | */ | ||
1278 | } | ||
1279 | |||
1280 | void rtl92ce_set_beacon_related_registers(struct ieee80211_hw *hw) | ||
1281 | { | ||
1282 | |||
1283 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1284 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1285 | u16 bcn_interval, atim_window; | ||
1286 | |||
1287 | bcn_interval = mac->beacon_interval; | ||
1288 | atim_window = 2; /*FIX MERGE */ | ||
1289 | rtl92ce_disable_interrupt(hw); | ||
1290 | rtl_write_word(rtlpriv, REG_ATIMWND, atim_window); | ||
1291 | rtl_write_word(rtlpriv, REG_BCN_INTERVAL, bcn_interval); | ||
1292 | rtl_write_word(rtlpriv, REG_BCNTCFG, 0x660f); | ||
1293 | rtl_write_byte(rtlpriv, REG_RXTSF_OFFSET_CCK, 0x18); | ||
1294 | rtl_write_byte(rtlpriv, REG_RXTSF_OFFSET_OFDM, 0x18); | ||
1295 | rtl_write_byte(rtlpriv, 0x606, 0x30); | ||
1296 | rtl92ce_enable_interrupt(hw); | ||
1297 | } | ||
1298 | |||
1299 | void rtl92ce_set_beacon_interval(struct ieee80211_hw *hw) | ||
1300 | { | ||
1301 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1302 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1303 | u16 bcn_interval = mac->beacon_interval; | ||
1304 | |||
1305 | RT_TRACE(rtlpriv, COMP_BEACON, DBG_DMESG, | ||
1306 | ("beacon_interval:%d\n", bcn_interval)); | ||
1307 | rtl92ce_disable_interrupt(hw); | ||
1308 | rtl_write_word(rtlpriv, REG_BCN_INTERVAL, bcn_interval); | ||
1309 | rtl92ce_enable_interrupt(hw); | ||
1310 | } | ||
1311 | |||
1312 | void rtl92ce_update_interrupt_mask(struct ieee80211_hw *hw, | ||
1313 | u32 add_msr, u32 rm_msr) | ||
1314 | { | ||
1315 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1316 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1317 | |||
1318 | RT_TRACE(rtlpriv, COMP_INTR, DBG_LOUD, | ||
1319 | ("add_msr:%x, rm_msr:%x\n", add_msr, rm_msr)); | ||
1320 | |||
1321 | if (add_msr) | ||
1322 | rtlpci->irq_mask[0] |= add_msr; | ||
1323 | if (rm_msr) | ||
1324 | rtlpci->irq_mask[0] &= (~rm_msr); | ||
1325 | rtl92ce_disable_interrupt(hw); | ||
1326 | rtl92ce_enable_interrupt(hw); | ||
1327 | } | ||
1328 | |||
1329 | static void _rtl92ce_read_txpower_info_from_hwpg(struct ieee80211_hw *hw, | ||
1330 | bool autoload_fail, | ||
1331 | u8 *hwinfo) | ||
1332 | { | ||
1333 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1334 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1335 | u8 rf_path, index, tempval; | ||
1336 | u16 i; | ||
1337 | |||
1338 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1339 | for (i = 0; i < 3; i++) { | ||
1340 | if (!autoload_fail) { | ||
1341 | rtlefuse-> | ||
1342 | eeprom_chnlarea_txpwr_cck[rf_path][i] = | ||
1343 | hwinfo[EEPROM_TXPOWERCCK + rf_path * 3 + i]; | ||
1344 | rtlefuse-> | ||
1345 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i] = | ||
1346 | hwinfo[EEPROM_TXPOWERHT40_1S + rf_path * 3 + | ||
1347 | i]; | ||
1348 | } else { | ||
1349 | rtlefuse-> | ||
1350 | eeprom_chnlarea_txpwr_cck[rf_path][i] = | ||
1351 | EEPROM_DEFAULT_TXPOWERLEVEL; | ||
1352 | rtlefuse-> | ||
1353 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i] = | ||
1354 | EEPROM_DEFAULT_TXPOWERLEVEL; | ||
1355 | } | ||
1356 | } | ||
1357 | } | ||
1358 | |||
1359 | for (i = 0; i < 3; i++) { | ||
1360 | if (!autoload_fail) | ||
1361 | tempval = hwinfo[EEPROM_TXPOWERHT40_2SDIFF + i]; | ||
1362 | else | ||
1363 | tempval = EEPROM_DEFAULT_HT40_2SDIFF; | ||
1364 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif[RF90_PATH_A][i] = | ||
1365 | (tempval & 0xf); | ||
1366 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif[RF90_PATH_B][i] = | ||
1367 | ((tempval & 0xf0) >> 4); | ||
1368 | } | ||
1369 | |||
1370 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1371 | for (i = 0; i < 3; i++) | ||
1372 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1373 | ("RF(%d) EEPROM CCK Area(%d) = 0x%x\n", rf_path, | ||
1374 | i, | ||
1375 | rtlefuse-> | ||
1376 | eeprom_chnlarea_txpwr_cck[rf_path][i])); | ||
1377 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1378 | for (i = 0; i < 3; i++) | ||
1379 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1380 | ("RF(%d) EEPROM HT40 1S Area(%d) = 0x%x\n", | ||
1381 | rf_path, i, | ||
1382 | rtlefuse-> | ||
1383 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i])); | ||
1384 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1385 | for (i = 0; i < 3; i++) | ||
1386 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1387 | ("RF(%d) EEPROM HT40 2S Diff Area(%d) = 0x%x\n", | ||
1388 | rf_path, i, | ||
1389 | rtlefuse-> | ||
1390 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path] | ||
1391 | [i])); | ||
1392 | |||
1393 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1394 | for (i = 0; i < 14; i++) { | ||
1395 | index = _rtl92c_get_chnl_group((u8) i); | ||
1396 | |||
1397 | rtlefuse->txpwrlevel_cck[rf_path][i] = | ||
1398 | rtlefuse->eeprom_chnlarea_txpwr_cck[rf_path][index]; | ||
1399 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i] = | ||
1400 | rtlefuse-> | ||
1401 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][index]; | ||
1402 | |||
1403 | if ((rtlefuse-> | ||
1404 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][index] - | ||
1405 | rtlefuse-> | ||
1406 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path][index]) | ||
1407 | > 0) { | ||
1408 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i] = | ||
1409 | rtlefuse-> | ||
1410 | eeprom_chnlarea_txpwr_ht40_1s[rf_path] | ||
1411 | [index] - | ||
1412 | rtlefuse-> | ||
1413 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path] | ||
1414 | [index]; | ||
1415 | } else { | ||
1416 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i] = 0; | ||
1417 | } | ||
1418 | } | ||
1419 | |||
1420 | for (i = 0; i < 14; i++) { | ||
1421 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1422 | ("RF(%d)-Ch(%d) [CCK / HT40_1S / HT40_2S] = " | ||
1423 | "[0x%x / 0x%x / 0x%x]\n", rf_path, i, | ||
1424 | rtlefuse->txpwrlevel_cck[rf_path][i], | ||
1425 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i], | ||
1426 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i])); | ||
1427 | } | ||
1428 | } | ||
1429 | |||
1430 | for (i = 0; i < 3; i++) { | ||
1431 | if (!autoload_fail) { | ||
1432 | rtlefuse->eeprom_pwrlimit_ht40[i] = | ||
1433 | hwinfo[EEPROM_TXPWR_GROUP + i]; | ||
1434 | rtlefuse->eeprom_pwrlimit_ht20[i] = | ||
1435 | hwinfo[EEPROM_TXPWR_GROUP + 3 + i]; | ||
1436 | } else { | ||
1437 | rtlefuse->eeprom_pwrlimit_ht40[i] = 0; | ||
1438 | rtlefuse->eeprom_pwrlimit_ht20[i] = 0; | ||
1439 | } | ||
1440 | } | ||
1441 | |||
1442 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1443 | for (i = 0; i < 14; i++) { | ||
1444 | index = _rtl92c_get_chnl_group((u8) i); | ||
1445 | |||
1446 | if (rf_path == RF90_PATH_A) { | ||
1447 | rtlefuse->pwrgroup_ht20[rf_path][i] = | ||
1448 | (rtlefuse->eeprom_pwrlimit_ht20[index] | ||
1449 | & 0xf); | ||
1450 | rtlefuse->pwrgroup_ht40[rf_path][i] = | ||
1451 | (rtlefuse->eeprom_pwrlimit_ht40[index] | ||
1452 | & 0xf); | ||
1453 | } else if (rf_path == RF90_PATH_B) { | ||
1454 | rtlefuse->pwrgroup_ht20[rf_path][i] = | ||
1455 | ((rtlefuse->eeprom_pwrlimit_ht20[index] | ||
1456 | & 0xf0) >> 4); | ||
1457 | rtlefuse->pwrgroup_ht40[rf_path][i] = | ||
1458 | ((rtlefuse->eeprom_pwrlimit_ht40[index] | ||
1459 | & 0xf0) >> 4); | ||
1460 | } | ||
1461 | |||
1462 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1463 | ("RF-%d pwrgroup_ht20[%d] = 0x%x\n", | ||
1464 | rf_path, i, | ||
1465 | rtlefuse->pwrgroup_ht20[rf_path][i])); | ||
1466 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1467 | ("RF-%d pwrgroup_ht40[%d] = 0x%x\n", | ||
1468 | rf_path, i, | ||
1469 | rtlefuse->pwrgroup_ht40[rf_path][i])); | ||
1470 | } | ||
1471 | } | ||
1472 | |||
1473 | for (i = 0; i < 14; i++) { | ||
1474 | index = _rtl92c_get_chnl_group((u8) i); | ||
1475 | |||
1476 | if (!autoload_fail) | ||
1477 | tempval = hwinfo[EEPROM_TXPOWERHT20DIFF + index]; | ||
1478 | else | ||
1479 | tempval = EEPROM_DEFAULT_HT20_DIFF; | ||
1480 | |||
1481 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] = (tempval & 0xF); | ||
1482 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] = | ||
1483 | ((tempval >> 4) & 0xF); | ||
1484 | |||
1485 | if (rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] & BIT(3)) | ||
1486 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] |= 0xF0; | ||
1487 | |||
1488 | if (rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] & BIT(3)) | ||
1489 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] |= 0xF0; | ||
1490 | |||
1491 | index = _rtl92c_get_chnl_group((u8) i); | ||
1492 | |||
1493 | if (!autoload_fail) | ||
1494 | tempval = hwinfo[EEPROM_TXPOWER_OFDMDIFF + index]; | ||
1495 | else | ||
1496 | tempval = EEPROM_DEFAULT_LEGACYHTTXPOWERDIFF; | ||
1497 | |||
1498 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i] = (tempval & 0xF); | ||
1499 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i] = | ||
1500 | ((tempval >> 4) & 0xF); | ||
1501 | } | ||
1502 | |||
1503 | rtlefuse->legacy_ht_txpowerdiff = | ||
1504 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][7]; | ||
1505 | |||
1506 | for (i = 0; i < 14; i++) | ||
1507 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1508 | ("RF-A Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
1509 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i])); | ||
1510 | for (i = 0; i < 14; i++) | ||
1511 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1512 | ("RF-A Legacy to Ht40 Diff[%d] = 0x%x\n", i, | ||
1513 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i])); | ||
1514 | for (i = 0; i < 14; i++) | ||
1515 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1516 | ("RF-B Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
1517 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i])); | ||
1518 | for (i = 0; i < 14; i++) | ||
1519 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1520 | ("RF-B Legacy to HT40 Diff[%d] = 0x%x\n", i, | ||
1521 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i])); | ||
1522 | |||
1523 | if (!autoload_fail) | ||
1524 | rtlefuse->eeprom_regulatory = (hwinfo[RF_OPTION1] & 0x7); | ||
1525 | else | ||
1526 | rtlefuse->eeprom_regulatory = 0; | ||
1527 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1528 | ("eeprom_regulatory = 0x%x\n", rtlefuse->eeprom_regulatory)); | ||
1529 | |||
1530 | if (!autoload_fail) { | ||
1531 | rtlefuse->eeprom_tssi[RF90_PATH_A] = hwinfo[EEPROM_TSSI_A]; | ||
1532 | rtlefuse->eeprom_tssi[RF90_PATH_B] = hwinfo[EEPROM_TSSI_B]; | ||
1533 | } else { | ||
1534 | rtlefuse->eeprom_tssi[RF90_PATH_A] = EEPROM_DEFAULT_TSSI; | ||
1535 | rtlefuse->eeprom_tssi[RF90_PATH_B] = EEPROM_DEFAULT_TSSI; | ||
1536 | } | ||
1537 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1538 | ("TSSI_A = 0x%x, TSSI_B = 0x%x\n", | ||
1539 | rtlefuse->eeprom_tssi[RF90_PATH_A], | ||
1540 | rtlefuse->eeprom_tssi[RF90_PATH_B])); | ||
1541 | |||
1542 | if (!autoload_fail) | ||
1543 | tempval = hwinfo[EEPROM_THERMAL_METER]; | ||
1544 | else | ||
1545 | tempval = EEPROM_DEFAULT_THERMALMETER; | ||
1546 | rtlefuse->eeprom_thermalmeter = (tempval & 0x1f); | ||
1547 | |||
1548 | if (rtlefuse->eeprom_thermalmeter == 0x1f || autoload_fail) | ||
1549 | rtlefuse->apk_thermalmeterignore = true; | ||
1550 | |||
1551 | rtlefuse->thermalmeter[0] = rtlefuse->eeprom_thermalmeter; | ||
1552 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1553 | ("thermalmeter = 0x%x\n", rtlefuse->eeprom_thermalmeter)); | ||
1554 | } | ||
1555 | |||
1556 | static void _rtl92ce_read_adapter_info(struct ieee80211_hw *hw) | ||
1557 | { | ||
1558 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1559 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1560 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1561 | u16 i, usvalue; | ||
1562 | u8 hwinfo[HWSET_MAX_SIZE]; | ||
1563 | u16 eeprom_id; | ||
1564 | |||
1565 | if (rtlefuse->epromtype == EEPROM_BOOT_EFUSE) { | ||
1566 | rtl_efuse_shadow_map_update(hw); | ||
1567 | |||
1568 | memcpy((void *)hwinfo, | ||
1569 | (void *)&rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
1570 | HWSET_MAX_SIZE); | ||
1571 | } else if (rtlefuse->epromtype == EEPROM_93C46) { | ||
1572 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1573 | ("RTL819X Not boot from eeprom, check it !!")); | ||
1574 | } | ||
1575 | |||
1576 | RT_PRINT_DATA(rtlpriv, COMP_INIT, DBG_DMESG, ("MAP\n"), | ||
1577 | hwinfo, HWSET_MAX_SIZE); | ||
1578 | |||
1579 | eeprom_id = *((u16 *)&hwinfo[0]); | ||
1580 | if (eeprom_id != RTL8190_EEPROM_ID) { | ||
1581 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1582 | ("EEPROM ID(%#x) is invalid!!\n", eeprom_id)); | ||
1583 | rtlefuse->autoload_failflag = true; | ||
1584 | } else { | ||
1585 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload OK\n")); | ||
1586 | rtlefuse->autoload_failflag = false; | ||
1587 | } | ||
1588 | |||
1589 | if (rtlefuse->autoload_failflag == true) | ||
1590 | return; | ||
1591 | |||
1592 | for (i = 0; i < 6; i += 2) { | ||
1593 | usvalue = *(u16 *)&hwinfo[EEPROM_MAC_ADDR + i]; | ||
1594 | *((u16 *) (&rtlefuse->dev_addr[i])) = usvalue; | ||
1595 | } | ||
1596 | |||
1597 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1598 | (MAC_FMT "\n", MAC_ARG(rtlefuse->dev_addr))); | ||
1599 | |||
1600 | _rtl92ce_read_txpower_info_from_hwpg(hw, | ||
1601 | rtlefuse->autoload_failflag, | ||
1602 | hwinfo); | ||
1603 | |||
1604 | rtl8192ce_read_bt_coexist_info_from_hwpg(hw, | ||
1605 | rtlefuse->autoload_failflag, | ||
1606 | hwinfo); | ||
1607 | |||
1608 | rtlefuse->eeprom_channelplan = *(u8 *)&hwinfo[EEPROM_CHANNELPLAN]; | ||
1609 | rtlefuse->eeprom_version = *(u16 *)&hwinfo[EEPROM_VERSION]; | ||
1610 | rtlefuse->txpwr_fromeprom = true; | ||
1611 | rtlefuse->eeprom_oemid = *(u8 *)&hwinfo[EEPROM_CUSTOMER_ID]; | ||
1612 | |||
1613 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1614 | ("EEPROM Customer ID: 0x%2x\n", rtlefuse->eeprom_oemid)); | ||
1615 | |||
1616 | /* set channel paln to world wide 13 */ | ||
1617 | rtlefuse->channel_plan = COUNTRY_CODE_WORLD_WIDE_13; | ||
1618 | |||
1619 | if (rtlhal->oem_id == RT_CID_DEFAULT) { | ||
1620 | switch (rtlefuse->eeprom_oemid) { | ||
1621 | case EEPROM_CID_DEFAULT: | ||
1622 | if (rtlefuse->eeprom_did == 0x8176) { | ||
1623 | if ((rtlefuse->eeprom_svid == 0x103C && | ||
1624 | rtlefuse->eeprom_smid == 0x1629)) | ||
1625 | rtlhal->oem_id = RT_CID_819x_HP; | ||
1626 | else | ||
1627 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
1628 | } else { | ||
1629 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
1630 | } | ||
1631 | break; | ||
1632 | case EEPROM_CID_TOSHIBA: | ||
1633 | rtlhal->oem_id = RT_CID_TOSHIBA; | ||
1634 | break; | ||
1635 | case EEPROM_CID_QMI: | ||
1636 | rtlhal->oem_id = RT_CID_819x_QMI; | ||
1637 | break; | ||
1638 | case EEPROM_CID_WHQL: | ||
1639 | default: | ||
1640 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
1641 | break; | ||
1642 | |||
1643 | } | ||
1644 | } | ||
1645 | |||
1646 | } | ||
1647 | |||
1648 | static void _rtl92ce_hal_customized_behavior(struct ieee80211_hw *hw) | ||
1649 | { | ||
1650 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1651 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
1652 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1653 | |||
1654 | switch (rtlhal->oem_id) { | ||
1655 | case RT_CID_819x_HP: | ||
1656 | pcipriv->ledctl.led_opendrain = true; | ||
1657 | break; | ||
1658 | case RT_CID_819x_Lenovo: | ||
1659 | case RT_CID_DEFAULT: | ||
1660 | case RT_CID_TOSHIBA: | ||
1661 | case RT_CID_CCX: | ||
1662 | case RT_CID_819x_Acer: | ||
1663 | case RT_CID_WHQL: | ||
1664 | default: | ||
1665 | break; | ||
1666 | } | ||
1667 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1668 | ("RT Customized ID: 0x%02X\n", rtlhal->oem_id)); | ||
1669 | } | ||
1670 | |||
1671 | void rtl92ce_read_eeprom_info(struct ieee80211_hw *hw) | ||
1672 | { | ||
1673 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1674 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1675 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1676 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1677 | u8 tmp_u1b; | ||
1678 | |||
1679 | rtlhal->version = _rtl92ce_read_chip_version(hw); | ||
1680 | if (get_rf_type(rtlphy) == RF_1T1R) | ||
1681 | rtlpriv->dm.rfpath_rxenable[0] = true; | ||
1682 | else | ||
1683 | rtlpriv->dm.rfpath_rxenable[0] = | ||
1684 | rtlpriv->dm.rfpath_rxenable[1] = true; | ||
1685 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("VersionID = 0x%4x\n", | ||
1686 | rtlhal->version)); | ||
1687 | tmp_u1b = rtl_read_byte(rtlpriv, REG_9346CR); | ||
1688 | if (tmp_u1b & BIT(4)) { | ||
1689 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Boot from EEPROM\n")); | ||
1690 | rtlefuse->epromtype = EEPROM_93C46; | ||
1691 | } else { | ||
1692 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Boot from EFUSE\n")); | ||
1693 | rtlefuse->epromtype = EEPROM_BOOT_EFUSE; | ||
1694 | } | ||
1695 | if (tmp_u1b & BIT(5)) { | ||
1696 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload OK\n")); | ||
1697 | rtlefuse->autoload_failflag = false; | ||
1698 | _rtl92ce_read_adapter_info(hw); | ||
1699 | } else { | ||
1700 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("Autoload ERR!!\n")); | ||
1701 | } | ||
1702 | _rtl92ce_hal_customized_behavior(hw); | ||
1703 | } | ||
1704 | |||
1705 | static void rtl92ce_update_hal_rate_table(struct ieee80211_hw *hw, | ||
1706 | struct ieee80211_sta *sta) | ||
1707 | { | ||
1708 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1709 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
1710 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1711 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1712 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1713 | u32 ratr_value; | ||
1714 | u8 ratr_index = 0; | ||
1715 | u8 nmode = mac->ht_enable; | ||
1716 | u8 mimo_ps = IEEE80211_SMPS_OFF; | ||
1717 | u16 shortgi_rate; | ||
1718 | u32 tmp_ratr_value; | ||
1719 | u8 curtxbw_40mhz = mac->bw_40; | ||
1720 | u8 curshortgi_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40) ? | ||
1721 | 1 : 0; | ||
1722 | u8 curshortgi_20mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20) ? | ||
1723 | 1 : 0; | ||
1724 | enum wireless_mode wirelessmode = mac->mode; | ||
1725 | |||
1726 | if (rtlhal->current_bandtype == BAND_ON_5G) | ||
1727 | ratr_value = sta->supp_rates[1] << 4; | ||
1728 | else | ||
1729 | ratr_value = sta->supp_rates[0]; | ||
1730 | ratr_value |= (sta->ht_cap.mcs.rx_mask[1] << 20 | | ||
1731 | sta->ht_cap.mcs.rx_mask[0] << 12); | ||
1732 | switch (wirelessmode) { | ||
1733 | case WIRELESS_MODE_B: | ||
1734 | if (ratr_value & 0x0000000c) | ||
1735 | ratr_value &= 0x0000000d; | ||
1736 | else | ||
1737 | ratr_value &= 0x0000000f; | ||
1738 | break; | ||
1739 | case WIRELESS_MODE_G: | ||
1740 | ratr_value &= 0x00000FF5; | ||
1741 | break; | ||
1742 | case WIRELESS_MODE_N_24G: | ||
1743 | case WIRELESS_MODE_N_5G: | ||
1744 | nmode = 1; | ||
1745 | if (mimo_ps == IEEE80211_SMPS_STATIC) { | ||
1746 | ratr_value &= 0x0007F005; | ||
1747 | } else { | ||
1748 | u32 ratr_mask; | ||
1749 | |||
1750 | if (get_rf_type(rtlphy) == RF_1T2R || | ||
1751 | get_rf_type(rtlphy) == RF_1T1R) | ||
1752 | ratr_mask = 0x000ff005; | ||
1753 | else | ||
1754 | ratr_mask = 0x0f0ff005; | ||
1755 | |||
1756 | ratr_value &= ratr_mask; | ||
1757 | } | ||
1758 | break; | ||
1759 | default: | ||
1760 | if (rtlphy->rf_type == RF_1T2R) | ||
1761 | ratr_value &= 0x000ff0ff; | ||
1762 | else | ||
1763 | ratr_value &= 0x0f0ff0ff; | ||
1764 | |||
1765 | break; | ||
1766 | } | ||
1767 | |||
1768 | if ((rtlpcipriv->bt_coexist.bt_coexistence) && | ||
1769 | (rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4) && | ||
1770 | (rtlpcipriv->bt_coexist.bt_cur_state) && | ||
1771 | (rtlpcipriv->bt_coexist.bt_ant_isolation) && | ||
1772 | ((rtlpcipriv->bt_coexist.bt_service == BT_SCO) || | ||
1773 | (rtlpcipriv->bt_coexist.bt_service == BT_BUSY))) | ||
1774 | ratr_value &= 0x0fffcfc0; | ||
1775 | else | ||
1776 | ratr_value &= 0x0FFFFFFF; | ||
1777 | |||
1778 | if (nmode && ((curtxbw_40mhz && | ||
1779 | curshortgi_40mhz) || (!curtxbw_40mhz && | ||
1780 | curshortgi_20mhz))) { | ||
1781 | |||
1782 | ratr_value |= 0x10000000; | ||
1783 | tmp_ratr_value = (ratr_value >> 12); | ||
1784 | |||
1785 | for (shortgi_rate = 15; shortgi_rate > 0; shortgi_rate--) { | ||
1786 | if ((1 << shortgi_rate) & tmp_ratr_value) | ||
1787 | break; | ||
1788 | } | ||
1789 | |||
1790 | shortgi_rate = (shortgi_rate << 12) | (shortgi_rate << 8) | | ||
1791 | (shortgi_rate << 4) | (shortgi_rate); | ||
1792 | } | ||
1793 | |||
1794 | rtl_write_dword(rtlpriv, REG_ARFR0 + ratr_index * 4, ratr_value); | ||
1795 | |||
1796 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, | ||
1797 | ("%x\n", rtl_read_dword(rtlpriv, REG_ARFR0))); | ||
1798 | } | ||
1799 | |||
1800 | static void rtl92ce_update_hal_rate_mask(struct ieee80211_hw *hw, | ||
1801 | struct ieee80211_sta *sta, u8 rssi_level) | ||
1802 | { | ||
1803 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1804 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1805 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1806 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1807 | struct rtl_sta_info *sta_entry = NULL; | ||
1808 | u32 ratr_bitmap; | ||
1809 | u8 ratr_index; | ||
1810 | u8 curtxbw_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40) | ||
1811 | ? 1 : 0; | ||
1812 | u8 curshortgi_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40) ? | ||
1813 | 1 : 0; | ||
1814 | u8 curshortgi_20mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20) ? | ||
1815 | 1 : 0; | ||
1816 | enum wireless_mode wirelessmode = 0; | ||
1817 | bool shortgi = false; | ||
1818 | u8 rate_mask[5]; | ||
1819 | u8 macid = 0; | ||
1820 | u8 mimo_ps = IEEE80211_SMPS_OFF; | ||
1821 | |||
1822 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
1823 | wirelessmode = sta_entry->wireless_mode; | ||
1824 | if (mac->opmode == NL80211_IFTYPE_STATION) | ||
1825 | curtxbw_40mhz = mac->bw_40; | ||
1826 | else if (mac->opmode == NL80211_IFTYPE_AP || | ||
1827 | mac->opmode == NL80211_IFTYPE_ADHOC) | ||
1828 | macid = sta->aid + 1; | ||
1829 | |||
1830 | if (rtlhal->current_bandtype == BAND_ON_5G) | ||
1831 | ratr_bitmap = sta->supp_rates[1] << 4; | ||
1832 | else | ||
1833 | ratr_bitmap = sta->supp_rates[0]; | ||
1834 | ratr_bitmap |= (sta->ht_cap.mcs.rx_mask[1] << 20 | | ||
1835 | sta->ht_cap.mcs.rx_mask[0] << 12); | ||
1836 | switch (wirelessmode) { | ||
1837 | case WIRELESS_MODE_B: | ||
1838 | ratr_index = RATR_INX_WIRELESS_B; | ||
1839 | if (ratr_bitmap & 0x0000000c) | ||
1840 | ratr_bitmap &= 0x0000000d; | ||
1841 | else | ||
1842 | ratr_bitmap &= 0x0000000f; | ||
1843 | break; | ||
1844 | case WIRELESS_MODE_G: | ||
1845 | ratr_index = RATR_INX_WIRELESS_GB; | ||
1846 | |||
1847 | if (rssi_level == 1) | ||
1848 | ratr_bitmap &= 0x00000f00; | ||
1849 | else if (rssi_level == 2) | ||
1850 | ratr_bitmap &= 0x00000ff0; | ||
1851 | else | ||
1852 | ratr_bitmap &= 0x00000ff5; | ||
1853 | break; | ||
1854 | case WIRELESS_MODE_A: | ||
1855 | ratr_index = RATR_INX_WIRELESS_A; | ||
1856 | ratr_bitmap &= 0x00000ff0; | ||
1857 | break; | ||
1858 | case WIRELESS_MODE_N_24G: | ||
1859 | case WIRELESS_MODE_N_5G: | ||
1860 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
1861 | |||
1862 | if (mimo_ps == IEEE80211_SMPS_STATIC) { | ||
1863 | if (rssi_level == 1) | ||
1864 | ratr_bitmap &= 0x00070000; | ||
1865 | else if (rssi_level == 2) | ||
1866 | ratr_bitmap &= 0x0007f000; | ||
1867 | else | ||
1868 | ratr_bitmap &= 0x0007f005; | ||
1869 | } else { | ||
1870 | if (rtlphy->rf_type == RF_1T2R || | ||
1871 | rtlphy->rf_type == RF_1T1R) { | ||
1872 | if (curtxbw_40mhz) { | ||
1873 | if (rssi_level == 1) | ||
1874 | ratr_bitmap &= 0x000f0000; | ||
1875 | else if (rssi_level == 2) | ||
1876 | ratr_bitmap &= 0x000ff000; | ||
1877 | else | ||
1878 | ratr_bitmap &= 0x000ff015; | ||
1879 | } else { | ||
1880 | if (rssi_level == 1) | ||
1881 | ratr_bitmap &= 0x000f0000; | ||
1882 | else if (rssi_level == 2) | ||
1883 | ratr_bitmap &= 0x000ff000; | ||
1884 | else | ||
1885 | ratr_bitmap &= 0x000ff005; | ||
1886 | } | ||
1887 | } else { | ||
1888 | if (curtxbw_40mhz) { | ||
1889 | if (rssi_level == 1) | ||
1890 | ratr_bitmap &= 0x0f0f0000; | ||
1891 | else if (rssi_level == 2) | ||
1892 | ratr_bitmap &= 0x0f0ff000; | ||
1893 | else | ||
1894 | ratr_bitmap &= 0x0f0ff015; | ||
1895 | } else { | ||
1896 | if (rssi_level == 1) | ||
1897 | ratr_bitmap &= 0x0f0f0000; | ||
1898 | else if (rssi_level == 2) | ||
1899 | ratr_bitmap &= 0x0f0ff000; | ||
1900 | else | ||
1901 | ratr_bitmap &= 0x0f0ff005; | ||
1902 | } | ||
1903 | } | ||
1904 | } | ||
1905 | |||
1906 | if ((curtxbw_40mhz && curshortgi_40mhz) || | ||
1907 | (!curtxbw_40mhz && curshortgi_20mhz)) { | ||
1908 | |||
1909 | if (macid == 0) | ||
1910 | shortgi = true; | ||
1911 | else if (macid == 1) | ||
1912 | shortgi = false; | ||
1913 | } | ||
1914 | break; | ||
1915 | default: | ||
1916 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
1917 | |||
1918 | if (rtlphy->rf_type == RF_1T2R) | ||
1919 | ratr_bitmap &= 0x000ff0ff; | ||
1920 | else | ||
1921 | ratr_bitmap &= 0x0f0ff0ff; | ||
1922 | break; | ||
1923 | } | ||
1924 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, | ||
1925 | ("ratr_bitmap :%x\n", ratr_bitmap)); | ||
1926 | *(u32 *)&rate_mask = EF4BYTE((ratr_bitmap & 0x0fffffff) | | ||
1927 | (ratr_index << 28)); | ||
1928 | rate_mask[4] = macid | (shortgi ? 0x20 : 0x00) | 0x80; | ||
1929 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, ("Rate_index:%x, " | ||
1930 | "ratr_val:%x, %x:%x:%x:%x:%x\n", | ||
1931 | ratr_index, ratr_bitmap, | ||
1932 | rate_mask[0], rate_mask[1], | ||
1933 | rate_mask[2], rate_mask[3], | ||
1934 | rate_mask[4])); | ||
1935 | rtl92c_fill_h2c_cmd(hw, H2C_RA_MASK, 5, rate_mask); | ||
1936 | |||
1937 | if (macid != 0) | ||
1938 | sta_entry->ratr_index = ratr_index; | ||
1939 | } | ||
1940 | |||
1941 | void rtl92ce_update_hal_rate_tbl(struct ieee80211_hw *hw, | ||
1942 | struct ieee80211_sta *sta, u8 rssi_level) | ||
1943 | { | ||
1944 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1945 | |||
1946 | if (rtlpriv->dm.useramask) | ||
1947 | rtl92ce_update_hal_rate_mask(hw, sta, rssi_level); | ||
1948 | else | ||
1949 | rtl92ce_update_hal_rate_table(hw, sta); | ||
1950 | } | ||
1951 | |||
1952 | void rtl92ce_update_channel_access_setting(struct ieee80211_hw *hw) | ||
1953 | { | ||
1954 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1955 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1956 | u16 sifs_timer; | ||
1957 | |||
1958 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SLOT_TIME, | ||
1959 | (u8 *)&mac->slot_time); | ||
1960 | if (!mac->ht_enable) | ||
1961 | sifs_timer = 0x0a0a; | ||
1962 | else | ||
1963 | sifs_timer = 0x1010; | ||
1964 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SIFS, (u8 *)&sifs_timer); | ||
1965 | } | ||
1966 | |||
1967 | bool rtl92ce_gpio_radio_on_off_checking(struct ieee80211_hw *hw, u8 *valid) | ||
1968 | { | ||
1969 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1970 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1971 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1972 | enum rf_pwrstate e_rfpowerstate_toset, cur_rfstate; | ||
1973 | u8 u1tmp; | ||
1974 | bool actuallyset = false; | ||
1975 | unsigned long flag; | ||
1976 | |||
1977 | if (rtlpci->being_init_adapter) | ||
1978 | return false; | ||
1979 | |||
1980 | if (ppsc->swrf_processing) | ||
1981 | return false; | ||
1982 | |||
1983 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
1984 | if (ppsc->rfchange_inprogress) { | ||
1985 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
1986 | return false; | ||
1987 | } else { | ||
1988 | ppsc->rfchange_inprogress = true; | ||
1989 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
1990 | } | ||
1991 | |||
1992 | cur_rfstate = ppsc->rfpwr_state; | ||
1993 | |||
1994 | rtl_write_byte(rtlpriv, REG_MAC_PINMUX_CFG, rtl_read_byte(rtlpriv, | ||
1995 | REG_MAC_PINMUX_CFG)&~(BIT(3))); | ||
1996 | |||
1997 | u1tmp = rtl_read_byte(rtlpriv, REG_GPIO_IO_SEL); | ||
1998 | e_rfpowerstate_toset = (u1tmp & BIT(3)) ? ERFON : ERFOFF; | ||
1999 | |||
2000 | if ((ppsc->hwradiooff == true) && (e_rfpowerstate_toset == ERFON)) { | ||
2001 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
2002 | ("GPIOChangeRF - HW Radio ON, RF ON\n")); | ||
2003 | |||
2004 | e_rfpowerstate_toset = ERFON; | ||
2005 | ppsc->hwradiooff = false; | ||
2006 | actuallyset = true; | ||
2007 | } else if ((ppsc->hwradiooff == false) | ||
2008 | && (e_rfpowerstate_toset == ERFOFF)) { | ||
2009 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
2010 | ("GPIOChangeRF - HW Radio OFF, RF OFF\n")); | ||
2011 | |||
2012 | e_rfpowerstate_toset = ERFOFF; | ||
2013 | ppsc->hwradiooff = true; | ||
2014 | actuallyset = true; | ||
2015 | } | ||
2016 | |||
2017 | if (actuallyset) { | ||
2018 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2019 | ppsc->rfchange_inprogress = false; | ||
2020 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2021 | } else { | ||
2022 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC) | ||
2023 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
2024 | |||
2025 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2026 | ppsc->rfchange_inprogress = false; | ||
2027 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2028 | } | ||
2029 | |||
2030 | *valid = 1; | ||
2031 | return !ppsc->hwradiooff; | ||
2032 | |||
2033 | } | ||
2034 | |||
2035 | void rtl92ce_set_key(struct ieee80211_hw *hw, u32 key_index, | ||
2036 | u8 *p_macaddr, bool is_group, u8 enc_algo, | ||
2037 | bool is_wepkey, bool clear_all) | ||
2038 | { | ||
2039 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2040 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2041 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
2042 | u8 *macaddr = p_macaddr; | ||
2043 | u32 entry_id = 0; | ||
2044 | bool is_pairwise = false; | ||
2045 | |||
2046 | static u8 cam_const_addr[4][6] = { | ||
2047 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, | ||
2048 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, | ||
2049 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x02}, | ||
2050 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x03} | ||
2051 | }; | ||
2052 | static u8 cam_const_broad[] = { | ||
2053 | 0xff, 0xff, 0xff, 0xff, 0xff, 0xff | ||
2054 | }; | ||
2055 | |||
2056 | if (clear_all) { | ||
2057 | u8 idx = 0; | ||
2058 | u8 cam_offset = 0; | ||
2059 | u8 clear_number = 5; | ||
2060 | |||
2061 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("clear_all\n")); | ||
2062 | |||
2063 | for (idx = 0; idx < clear_number; idx++) { | ||
2064 | rtl_cam_mark_invalid(hw, cam_offset + idx); | ||
2065 | rtl_cam_empty_entry(hw, cam_offset + idx); | ||
2066 | |||
2067 | if (idx < 5) { | ||
2068 | memset(rtlpriv->sec.key_buf[idx], 0, | ||
2069 | MAX_KEY_LEN); | ||
2070 | rtlpriv->sec.key_len[idx] = 0; | ||
2071 | } | ||
2072 | } | ||
2073 | |||
2074 | } else { | ||
2075 | switch (enc_algo) { | ||
2076 | case WEP40_ENCRYPTION: | ||
2077 | enc_algo = CAM_WEP40; | ||
2078 | break; | ||
2079 | case WEP104_ENCRYPTION: | ||
2080 | enc_algo = CAM_WEP104; | ||
2081 | break; | ||
2082 | case TKIP_ENCRYPTION: | ||
2083 | enc_algo = CAM_TKIP; | ||
2084 | break; | ||
2085 | case AESCCMP_ENCRYPTION: | ||
2086 | enc_algo = CAM_AES; | ||
2087 | break; | ||
2088 | default: | ||
2089 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("switch case " | ||
2090 | "not process\n")); | ||
2091 | enc_algo = CAM_TKIP; | ||
2092 | break; | ||
2093 | } | ||
2094 | |||
2095 | if (is_wepkey || rtlpriv->sec.use_defaultkey) { | ||
2096 | macaddr = cam_const_addr[key_index]; | ||
2097 | entry_id = key_index; | ||
2098 | } else { | ||
2099 | if (is_group) { | ||
2100 | macaddr = cam_const_broad; | ||
2101 | entry_id = key_index; | ||
2102 | } else { | ||
2103 | if (mac->opmode == NL80211_IFTYPE_AP) { | ||
2104 | entry_id = rtl_cam_get_free_entry(hw, | ||
2105 | p_macaddr); | ||
2106 | if (entry_id >= TOTAL_CAM_ENTRY) { | ||
2107 | RT_TRACE(rtlpriv, COMP_SEC, | ||
2108 | DBG_EMERG, | ||
2109 | ("Can not find free hw" | ||
2110 | " security cam entry\n")); | ||
2111 | return; | ||
2112 | } | ||
2113 | } else { | ||
2114 | entry_id = CAM_PAIRWISE_KEY_POSITION; | ||
2115 | } | ||
2116 | |||
2117 | key_index = PAIRWISE_KEYIDX; | ||
2118 | is_pairwise = true; | ||
2119 | } | ||
2120 | } | ||
2121 | |||
2122 | if (rtlpriv->sec.key_len[key_index] == 0) { | ||
2123 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2124 | ("delete one entry, entry_id is %d\n", | ||
2125 | entry_id)); | ||
2126 | if (mac->opmode == NL80211_IFTYPE_AP) | ||
2127 | rtl_cam_del_entry(hw, p_macaddr); | ||
2128 | rtl_cam_delete_one_entry(hw, p_macaddr, entry_id); | ||
2129 | } else { | ||
2130 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2131 | ("The insert KEY length is %d\n", | ||
2132 | rtlpriv->sec.key_len[PAIRWISE_KEYIDX])); | ||
2133 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2134 | ("The insert KEY is %x %x\n", | ||
2135 | rtlpriv->sec.key_buf[0][0], | ||
2136 | rtlpriv->sec.key_buf[0][1])); | ||
2137 | |||
2138 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2139 | ("add one entry\n")); | ||
2140 | if (is_pairwise) { | ||
2141 | RT_PRINT_DATA(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2142 | "Pairwiase Key content :", | ||
2143 | rtlpriv->sec.pairwise_key, | ||
2144 | rtlpriv->sec. | ||
2145 | key_len[PAIRWISE_KEYIDX]); | ||
2146 | |||
2147 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2148 | ("set Pairwiase key\n")); | ||
2149 | |||
2150 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
2151 | entry_id, enc_algo, | ||
2152 | CAM_CONFIG_NO_USEDK, | ||
2153 | rtlpriv->sec. | ||
2154 | key_buf[key_index]); | ||
2155 | } else { | ||
2156 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2157 | ("set group key\n")); | ||
2158 | |||
2159 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
2160 | rtl_cam_add_one_entry(hw, | ||
2161 | rtlefuse->dev_addr, | ||
2162 | PAIRWISE_KEYIDX, | ||
2163 | CAM_PAIRWISE_KEY_POSITION, | ||
2164 | enc_algo, | ||
2165 | CAM_CONFIG_NO_USEDK, | ||
2166 | rtlpriv->sec.key_buf | ||
2167 | [entry_id]); | ||
2168 | } | ||
2169 | |||
2170 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
2171 | entry_id, enc_algo, | ||
2172 | CAM_CONFIG_NO_USEDK, | ||
2173 | rtlpriv->sec.key_buf[entry_id]); | ||
2174 | } | ||
2175 | |||
2176 | } | ||
2177 | } | ||
2178 | } | ||
2179 | |||
2180 | static void rtl8192ce_bt_var_init(struct ieee80211_hw *hw) | ||
2181 | { | ||
2182 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
2183 | |||
2184 | rtlpcipriv->bt_coexist.bt_coexistence = | ||
2185 | rtlpcipriv->bt_coexist.eeprom_bt_coexist; | ||
2186 | rtlpcipriv->bt_coexist.bt_ant_num = | ||
2187 | rtlpcipriv->bt_coexist.eeprom_bt_ant_num; | ||
2188 | rtlpcipriv->bt_coexist.bt_coexist_type = | ||
2189 | rtlpcipriv->bt_coexist.eeprom_bt_type; | ||
2190 | |||
2191 | if (rtlpcipriv->bt_coexist.reg_bt_iso == 2) | ||
2192 | rtlpcipriv->bt_coexist.bt_ant_isolation = | ||
2193 | rtlpcipriv->bt_coexist.eeprom_bt_ant_isolation; | ||
2194 | else | ||
2195 | rtlpcipriv->bt_coexist.bt_ant_isolation = | ||
2196 | rtlpcipriv->bt_coexist.reg_bt_iso; | ||
2197 | |||
2198 | rtlpcipriv->bt_coexist.bt_radio_shared_type = | ||
2199 | rtlpcipriv->bt_coexist.eeprom_bt_radio_shared; | ||
2200 | |||
2201 | if (rtlpcipriv->bt_coexist.bt_coexistence) { | ||
2202 | |||
2203 | if (rtlpcipriv->bt_coexist.reg_bt_sco == 1) | ||
2204 | rtlpcipriv->bt_coexist.bt_service = BT_OTHER_ACTION; | ||
2205 | else if (rtlpcipriv->bt_coexist.reg_bt_sco == 2) | ||
2206 | rtlpcipriv->bt_coexist.bt_service = BT_SCO; | ||
2207 | else if (rtlpcipriv->bt_coexist.reg_bt_sco == 4) | ||
2208 | rtlpcipriv->bt_coexist.bt_service = BT_BUSY; | ||
2209 | else if (rtlpcipriv->bt_coexist.reg_bt_sco == 5) | ||
2210 | rtlpcipriv->bt_coexist.bt_service = BT_OTHERBUSY; | ||
2211 | else | ||
2212 | rtlpcipriv->bt_coexist.bt_service = BT_IDLE; | ||
2213 | |||
2214 | rtlpcipriv->bt_coexist.bt_edca_ul = 0; | ||
2215 | rtlpcipriv->bt_coexist.bt_edca_dl = 0; | ||
2216 | rtlpcipriv->bt_coexist.bt_rssi_state = 0xff; | ||
2217 | } | ||
2218 | } | ||
2219 | |||
2220 | void rtl8192ce_read_bt_coexist_info_from_hwpg(struct ieee80211_hw *hw, | ||
2221 | bool auto_load_fail, u8 *hwinfo) | ||
2222 | { | ||
2223 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
2224 | u8 value; | ||
2225 | |||
2226 | if (!auto_load_fail) { | ||
2227 | rtlpcipriv->bt_coexist.eeprom_bt_coexist = | ||
2228 | ((hwinfo[RF_OPTION1] & 0xe0) >> 5); | ||
2229 | value = hwinfo[RF_OPTION4]; | ||
2230 | rtlpcipriv->bt_coexist.eeprom_bt_type = ((value & 0xe) >> 1); | ||
2231 | rtlpcipriv->bt_coexist.eeprom_bt_ant_num = (value & 0x1); | ||
2232 | rtlpcipriv->bt_coexist.eeprom_bt_ant_isolation = | ||
2233 | ((value & 0x10) >> 4); | ||
2234 | rtlpcipriv->bt_coexist.eeprom_bt_radio_shared = | ||
2235 | ((value & 0x20) >> 5); | ||
2236 | } else { | ||
2237 | rtlpcipriv->bt_coexist.eeprom_bt_coexist = 0; | ||
2238 | rtlpcipriv->bt_coexist.eeprom_bt_type = BT_2WIRE; | ||
2239 | rtlpcipriv->bt_coexist.eeprom_bt_ant_num = ANT_X2; | ||
2240 | rtlpcipriv->bt_coexist.eeprom_bt_ant_isolation = 0; | ||
2241 | rtlpcipriv->bt_coexist.eeprom_bt_radio_shared = BT_RADIO_SHARED; | ||
2242 | } | ||
2243 | |||
2244 | rtl8192ce_bt_var_init(hw); | ||
2245 | } | ||
2246 | |||
2247 | void rtl8192ce_bt_reg_init(struct ieee80211_hw *hw) | ||
2248 | { | ||
2249 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
2250 | |||
2251 | /* 0:Low, 1:High, 2:From Efuse. */ | ||
2252 | rtlpcipriv->bt_coexist.reg_bt_iso = 2; | ||
2253 | /* 0:Idle, 1:None-SCO, 2:SCO, 3:From Counter. */ | ||
2254 | rtlpcipriv->bt_coexist.reg_bt_sco = 3; | ||
2255 | /* 0:Disable BT control A-MPDU, 1:Enable BT control A-MPDU. */ | ||
2256 | rtlpcipriv->bt_coexist.reg_bt_sco = 0; | ||
2257 | } | ||
2258 | |||
2259 | |||
2260 | void rtl8192ce_bt_hw_init(struct ieee80211_hw *hw) | ||
2261 | { | ||
2262 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2263 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
2264 | struct rtl_pci_priv *rtlpcipriv = rtl_pcipriv(hw); | ||
2265 | |||
2266 | u8 u1_tmp; | ||
2267 | |||
2268 | if (rtlpcipriv->bt_coexist.bt_coexistence && | ||
2269 | ((rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC4) || | ||
2270 | rtlpcipriv->bt_coexist.bt_coexist_type == BT_CSR_BC8)) { | ||
2271 | |||
2272 | if (rtlpcipriv->bt_coexist.bt_ant_isolation) | ||
2273 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG, 0xa0); | ||
2274 | |||
2275 | u1_tmp = rtl_read_byte(rtlpriv, 0x4fd) & | ||
2276 | BIT_OFFSET_LEN_MASK_32(0, 1); | ||
2277 | u1_tmp = u1_tmp | | ||
2278 | ((rtlpcipriv->bt_coexist.bt_ant_isolation == 1) ? | ||
2279 | 0 : BIT_OFFSET_LEN_MASK_32(1, 1)) | | ||
2280 | ((rtlpcipriv->bt_coexist.bt_service == BT_SCO) ? | ||
2281 | 0 : BIT_OFFSET_LEN_MASK_32(2, 1)); | ||
2282 | rtl_write_byte(rtlpriv, 0x4fd, u1_tmp); | ||
2283 | |||
2284 | rtl_write_dword(rtlpriv, REG_BT_COEX_TABLE+4, 0xaaaa9aaa); | ||
2285 | rtl_write_dword(rtlpriv, REG_BT_COEX_TABLE+8, 0xffbd0040); | ||
2286 | rtl_write_dword(rtlpriv, REG_BT_COEX_TABLE+0xc, 0x40000010); | ||
2287 | |||
2288 | /* Config to 1T1R. */ | ||
2289 | if (rtlphy->rf_type == RF_1T1R) { | ||
2290 | u1_tmp = rtl_read_byte(rtlpriv, ROFDM0_TRXPATHENABLE); | ||
2291 | u1_tmp &= ~(BIT_OFFSET_LEN_MASK_32(1, 1)); | ||
2292 | rtl_write_byte(rtlpriv, ROFDM0_TRXPATHENABLE, u1_tmp); | ||
2293 | |||
2294 | u1_tmp = rtl_read_byte(rtlpriv, ROFDM1_TRXPATHENABLE); | ||
2295 | u1_tmp &= ~(BIT_OFFSET_LEN_MASK_32(1, 1)); | ||
2296 | rtl_write_byte(rtlpriv, ROFDM1_TRXPATHENABLE, u1_tmp); | ||
2297 | } | ||
2298 | } | ||
2299 | } | ||
2300 | |||
2301 | void rtl92ce_suspend(struct ieee80211_hw *hw) | ||
2302 | { | ||
2303 | } | ||
2304 | |||
2305 | void rtl92ce_resume(struct ieee80211_hw *hw) | ||
2306 | { | ||
2307 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/hw.h b/drivers/net/wireless/rtlwifi/rtl8192ce/hw.h new file mode 100644 index 000000000000..07dbe3e340a5 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/hw.h | |||
@@ -0,0 +1,78 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CE_HW_H__ | ||
31 | #define __RTL92CE_HW_H__ | ||
32 | |||
33 | static inline u8 _rtl92c_get_chnl_group(u8 chnl) | ||
34 | { | ||
35 | u8 group; | ||
36 | |||
37 | if (chnl < 3) | ||
38 | group = 0; | ||
39 | else if (chnl < 9) | ||
40 | group = 1; | ||
41 | else | ||
42 | group = 2; | ||
43 | return group; | ||
44 | } | ||
45 | |||
46 | void rtl92ce_get_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
47 | void rtl92ce_read_eeprom_info(struct ieee80211_hw *hw); | ||
48 | void rtl92ce_interrupt_recognized(struct ieee80211_hw *hw, | ||
49 | u32 *p_inta, u32 *p_intb); | ||
50 | int rtl92ce_hw_init(struct ieee80211_hw *hw); | ||
51 | void rtl92ce_card_disable(struct ieee80211_hw *hw); | ||
52 | void rtl92ce_enable_interrupt(struct ieee80211_hw *hw); | ||
53 | void rtl92ce_disable_interrupt(struct ieee80211_hw *hw); | ||
54 | int rtl92ce_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type); | ||
55 | void rtl92ce_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid); | ||
56 | void rtl92ce_set_qos(struct ieee80211_hw *hw, int aci); | ||
57 | void rtl92ce_set_beacon_related_registers(struct ieee80211_hw *hw); | ||
58 | void rtl92ce_set_beacon_interval(struct ieee80211_hw *hw); | ||
59 | void rtl92ce_update_interrupt_mask(struct ieee80211_hw *hw, | ||
60 | u32 add_msr, u32 rm_msr); | ||
61 | void rtl92ce_set_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
62 | void rtl92ce_update_hal_rate_tbl(struct ieee80211_hw *hw, | ||
63 | struct ieee80211_sta *sta, u8 rssi_level); | ||
64 | void rtl92ce_update_channel_access_setting(struct ieee80211_hw *hw); | ||
65 | bool rtl92ce_gpio_radio_on_off_checking(struct ieee80211_hw *hw, u8 *valid); | ||
66 | void rtl92ce_enable_hw_security_config(struct ieee80211_hw *hw); | ||
67 | void rtl92ce_set_key(struct ieee80211_hw *hw, u32 key_index, | ||
68 | u8 *p_macaddr, bool is_group, u8 enc_algo, | ||
69 | bool is_wepkey, bool clear_all); | ||
70 | |||
71 | void rtl8192ce_read_bt_coexist_info_from_hwpg(struct ieee80211_hw *hw, | ||
72 | bool autoload_fail, u8 *hwinfo); | ||
73 | void rtl8192ce_bt_reg_init(struct ieee80211_hw *hw); | ||
74 | void rtl8192ce_bt_hw_init(struct ieee80211_hw *hw); | ||
75 | void rtl92ce_suspend(struct ieee80211_hw *hw); | ||
76 | void rtl92ce_resume(struct ieee80211_hw *hw); | ||
77 | |||
78 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/led.c b/drivers/net/wireless/rtlwifi/rtl8192ce/led.c new file mode 100644 index 000000000000..9dd1ed7b6422 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/led.c | |||
@@ -0,0 +1,151 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "reg.h" | ||
33 | #include "led.h" | ||
34 | |||
35 | static void _rtl92ce_init_led(struct ieee80211_hw *hw, | ||
36 | struct rtl_led *pled, enum rtl_led_pin ledpin) | ||
37 | { | ||
38 | pled->hw = hw; | ||
39 | pled->ledpin = ledpin; | ||
40 | pled->ledon = false; | ||
41 | } | ||
42 | |||
43 | void rtl92ce_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
44 | { | ||
45 | u8 ledcfg; | ||
46 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
47 | |||
48 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
49 | ("LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pled->ledpin)); | ||
50 | |||
51 | ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2); | ||
52 | |||
53 | switch (pled->ledpin) { | ||
54 | case LED_PIN_GPIO0: | ||
55 | break; | ||
56 | case LED_PIN_LED0: | ||
57 | rtl_write_byte(rtlpriv, | ||
58 | REG_LEDCFG2, (ledcfg & 0xf0) | BIT(5) | BIT(6)); | ||
59 | break; | ||
60 | case LED_PIN_LED1: | ||
61 | rtl_write_byte(rtlpriv, REG_LEDCFG2, (ledcfg & 0x0f) | BIT(5)); | ||
62 | break; | ||
63 | default: | ||
64 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
65 | ("switch case not process\n")); | ||
66 | break; | ||
67 | } | ||
68 | pled->ledon = true; | ||
69 | } | ||
70 | |||
71 | void rtl92ce_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
72 | { | ||
73 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
74 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
75 | u8 ledcfg; | ||
76 | |||
77 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
78 | ("LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pled->ledpin)); | ||
79 | |||
80 | ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2); | ||
81 | |||
82 | switch (pled->ledpin) { | ||
83 | case LED_PIN_GPIO0: | ||
84 | break; | ||
85 | case LED_PIN_LED0: | ||
86 | ledcfg &= 0xf0; | ||
87 | if (pcipriv->ledctl.led_opendrain == true) | ||
88 | rtl_write_byte(rtlpriv, REG_LEDCFG2, | ||
89 | (ledcfg | BIT(1) | BIT(5) | BIT(6))); | ||
90 | else | ||
91 | rtl_write_byte(rtlpriv, REG_LEDCFG2, | ||
92 | (ledcfg | BIT(3) | BIT(5) | BIT(6))); | ||
93 | break; | ||
94 | case LED_PIN_LED1: | ||
95 | ledcfg &= 0x0f; | ||
96 | rtl_write_byte(rtlpriv, REG_LEDCFG2, (ledcfg | BIT(3))); | ||
97 | break; | ||
98 | default: | ||
99 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
100 | ("switch case not process\n")); | ||
101 | break; | ||
102 | } | ||
103 | pled->ledon = false; | ||
104 | } | ||
105 | |||
106 | void rtl92ce_init_sw_leds(struct ieee80211_hw *hw) | ||
107 | { | ||
108 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
109 | _rtl92ce_init_led(hw, &(pcipriv->ledctl.sw_led0), LED_PIN_LED0); | ||
110 | _rtl92ce_init_led(hw, &(pcipriv->ledctl.sw_led1), LED_PIN_LED1); | ||
111 | } | ||
112 | |||
113 | static void _rtl92ce_sw_led_control(struct ieee80211_hw *hw, | ||
114 | enum led_ctl_mode ledaction) | ||
115 | { | ||
116 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
117 | struct rtl_led *pLed0 = &(pcipriv->ledctl.sw_led0); | ||
118 | switch (ledaction) { | ||
119 | case LED_CTL_POWER_ON: | ||
120 | case LED_CTL_LINK: | ||
121 | case LED_CTL_NO_LINK: | ||
122 | rtl92ce_sw_led_on(hw, pLed0); | ||
123 | break; | ||
124 | case LED_CTL_POWER_OFF: | ||
125 | rtl92ce_sw_led_off(hw, pLed0); | ||
126 | break; | ||
127 | default: | ||
128 | break; | ||
129 | } | ||
130 | } | ||
131 | |||
132 | void rtl92ce_led_control(struct ieee80211_hw *hw, | ||
133 | enum led_ctl_mode ledaction) | ||
134 | { | ||
135 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
136 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
137 | |||
138 | if ((ppsc->rfoff_reason > RF_CHANGE_BY_PS) && | ||
139 | (ledaction == LED_CTL_TX || | ||
140 | ledaction == LED_CTL_RX || | ||
141 | ledaction == LED_CTL_SITE_SURVEY || | ||
142 | ledaction == LED_CTL_LINK || | ||
143 | ledaction == LED_CTL_NO_LINK || | ||
144 | ledaction == LED_CTL_START_TO_LINK || | ||
145 | ledaction == LED_CTL_POWER_ON)) { | ||
146 | return; | ||
147 | } | ||
148 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, ("ledaction %d.\n", | ||
149 | ledaction)); | ||
150 | _rtl92ce_sw_led_control(hw, ledaction); | ||
151 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/led.h b/drivers/net/wireless/rtlwifi/rtl8192ce/led.h new file mode 100644 index 000000000000..7dfccea2095b --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/led.h | |||
@@ -0,0 +1,38 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CE_LED_H__ | ||
31 | #define __RTL92CE_LED_H__ | ||
32 | |||
33 | void rtl92ce_init_sw_leds(struct ieee80211_hw *hw); | ||
34 | void rtl92ce_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
35 | void rtl92ce_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
36 | void rtl92ce_led_control(struct ieee80211_hw *hw, enum led_ctl_mode ledaction); | ||
37 | |||
38 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/phy.c b/drivers/net/wireless/rtlwifi/rtl8192ce/phy.c new file mode 100644 index 000000000000..abe0fcc75368 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/phy.c | |||
@@ -0,0 +1,635 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../ps.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "hw.h" | ||
36 | #include "phy.h" | ||
37 | #include "rf.h" | ||
38 | #include "dm.h" | ||
39 | #include "table.h" | ||
40 | |||
41 | static bool _rtl92c_phy_config_mac_with_headerfile(struct ieee80211_hw *hw); | ||
42 | |||
43 | u32 rtl92c_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
44 | enum radio_path rfpath, u32 regaddr, u32 bitmask) | ||
45 | { | ||
46 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
47 | u32 original_value, readback_value, bitshift; | ||
48 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
49 | unsigned long flags; | ||
50 | |||
51 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), " | ||
52 | "rfpath(%#x), bitmask(%#x)\n", | ||
53 | regaddr, rfpath, bitmask)); | ||
54 | |||
55 | spin_lock_irqsave(&rtlpriv->locks.rf_lock, flags); | ||
56 | |||
57 | if (rtlphy->rf_mode != RF_OP_BY_FW) { | ||
58 | original_value = _rtl92c_phy_rf_serial_read(hw, | ||
59 | rfpath, regaddr); | ||
60 | } else { | ||
61 | original_value = _rtl92c_phy_fw_rf_serial_read(hw, | ||
62 | rfpath, regaddr); | ||
63 | } | ||
64 | |||
65 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
66 | readback_value = (original_value & bitmask) >> bitshift; | ||
67 | |||
68 | spin_unlock_irqrestore(&rtlpriv->locks.rf_lock, flags); | ||
69 | |||
70 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
71 | ("regaddr(%#x), rfpath(%#x), " | ||
72 | "bitmask(%#x), original_value(%#x)\n", | ||
73 | regaddr, rfpath, bitmask, original_value)); | ||
74 | |||
75 | return readback_value; | ||
76 | } | ||
77 | |||
78 | bool rtl92c_phy_mac_config(struct ieee80211_hw *hw) | ||
79 | { | ||
80 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
81 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
82 | bool is92c = IS_92C_SERIAL(rtlhal->version); | ||
83 | bool rtstatus = _rtl92c_phy_config_mac_with_headerfile(hw); | ||
84 | |||
85 | if (is92c) | ||
86 | rtl_write_byte(rtlpriv, 0x14, 0x71); | ||
87 | return rtstatus; | ||
88 | } | ||
89 | |||
90 | bool rtl92c_phy_bb_config(struct ieee80211_hw *hw) | ||
91 | { | ||
92 | bool rtstatus = true; | ||
93 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
94 | u16 regval; | ||
95 | u32 regvaldw; | ||
96 | u8 reg_hwparafile = 1; | ||
97 | |||
98 | _rtl92c_phy_init_bb_rf_register_definition(hw); | ||
99 | regval = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
100 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, | ||
101 | regval | BIT(13) | BIT(0) | BIT(1)); | ||
102 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL, 0x83); | ||
103 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL + 1, 0xdb); | ||
104 | rtl_write_byte(rtlpriv, REG_RF_CTRL, RF_EN | RF_RSTB | RF_SDMRSTB); | ||
105 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, | ||
106 | FEN_PPLL | FEN_PCIEA | FEN_DIO_PCIE | | ||
107 | FEN_BB_GLB_RSTn | FEN_BBRSTB); | ||
108 | rtl_write_byte(rtlpriv, REG_AFE_XTAL_CTRL + 1, 0x80); | ||
109 | regvaldw = rtl_read_dword(rtlpriv, REG_LEDCFG0); | ||
110 | rtl_write_dword(rtlpriv, REG_LEDCFG0, regvaldw | BIT(23)); | ||
111 | if (reg_hwparafile == 1) | ||
112 | rtstatus = _rtl92c_phy_bb8192c_config_parafile(hw); | ||
113 | return rtstatus; | ||
114 | } | ||
115 | |||
116 | void rtl92ce_phy_set_rf_reg(struct ieee80211_hw *hw, | ||
117 | enum radio_path rfpath, | ||
118 | u32 regaddr, u32 bitmask, u32 data) | ||
119 | { | ||
120 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
121 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
122 | u32 original_value, bitshift; | ||
123 | unsigned long flags; | ||
124 | |||
125 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
126 | ("regaddr(%#x), bitmask(%#x), data(%#x), rfpath(%#x)\n", | ||
127 | regaddr, bitmask, data, rfpath)); | ||
128 | |||
129 | spin_lock_irqsave(&rtlpriv->locks.rf_lock, flags); | ||
130 | |||
131 | if (rtlphy->rf_mode != RF_OP_BY_FW) { | ||
132 | if (bitmask != RFREG_OFFSET_MASK) { | ||
133 | original_value = _rtl92c_phy_rf_serial_read(hw, | ||
134 | rfpath, | ||
135 | regaddr); | ||
136 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
137 | data = | ||
138 | ((original_value & (~bitmask)) | | ||
139 | (data << bitshift)); | ||
140 | } | ||
141 | |||
142 | _rtl92c_phy_rf_serial_write(hw, rfpath, regaddr, data); | ||
143 | } else { | ||
144 | if (bitmask != RFREG_OFFSET_MASK) { | ||
145 | original_value = _rtl92c_phy_fw_rf_serial_read(hw, | ||
146 | rfpath, | ||
147 | regaddr); | ||
148 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
149 | data = | ||
150 | ((original_value & (~bitmask)) | | ||
151 | (data << bitshift)); | ||
152 | } | ||
153 | _rtl92c_phy_fw_rf_serial_write(hw, rfpath, regaddr, data); | ||
154 | } | ||
155 | |||
156 | spin_unlock_irqrestore(&rtlpriv->locks.rf_lock, flags); | ||
157 | |||
158 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), " | ||
159 | "bitmask(%#x), data(%#x), " | ||
160 | "rfpath(%#x)\n", regaddr, | ||
161 | bitmask, data, rfpath)); | ||
162 | } | ||
163 | |||
164 | static bool _rtl92c_phy_config_mac_with_headerfile(struct ieee80211_hw *hw) | ||
165 | { | ||
166 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
167 | u32 i; | ||
168 | u32 arraylength; | ||
169 | u32 *ptrarray; | ||
170 | |||
171 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Read Rtl819XMACPHY_Array\n")); | ||
172 | arraylength = MAC_2T_ARRAYLENGTH; | ||
173 | ptrarray = RTL8192CEMAC_2T_ARRAY; | ||
174 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
175 | ("Img:RTL8192CEMAC_2T_ARRAY\n")); | ||
176 | for (i = 0; i < arraylength; i = i + 2) | ||
177 | rtl_write_byte(rtlpriv, ptrarray[i], (u8) ptrarray[i + 1]); | ||
178 | return true; | ||
179 | } | ||
180 | |||
181 | bool _rtl92ce_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
182 | u8 configtype) | ||
183 | { | ||
184 | int i; | ||
185 | u32 *phy_regarray_table; | ||
186 | u32 *agctab_array_table; | ||
187 | u16 phy_reg_arraylen, agctab_arraylen; | ||
188 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
189 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
190 | |||
191 | if (IS_92C_SERIAL(rtlhal->version)) { | ||
192 | agctab_arraylen = AGCTAB_2TARRAYLENGTH; | ||
193 | agctab_array_table = RTL8192CEAGCTAB_2TARRAY; | ||
194 | phy_reg_arraylen = PHY_REG_2TARRAY_LENGTH; | ||
195 | phy_regarray_table = RTL8192CEPHY_REG_2TARRAY; | ||
196 | } else { | ||
197 | agctab_arraylen = AGCTAB_1TARRAYLENGTH; | ||
198 | agctab_array_table = RTL8192CEAGCTAB_1TARRAY; | ||
199 | phy_reg_arraylen = PHY_REG_1TARRAY_LENGTH; | ||
200 | phy_regarray_table = RTL8192CEPHY_REG_1TARRAY; | ||
201 | } | ||
202 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
203 | for (i = 0; i < phy_reg_arraylen; i = i + 2) { | ||
204 | if (phy_regarray_table[i] == 0xfe) | ||
205 | mdelay(50); | ||
206 | else if (phy_regarray_table[i] == 0xfd) | ||
207 | mdelay(5); | ||
208 | else if (phy_regarray_table[i] == 0xfc) | ||
209 | mdelay(1); | ||
210 | else if (phy_regarray_table[i] == 0xfb) | ||
211 | udelay(50); | ||
212 | else if (phy_regarray_table[i] == 0xfa) | ||
213 | udelay(5); | ||
214 | else if (phy_regarray_table[i] == 0xf9) | ||
215 | udelay(1); | ||
216 | rtl_set_bbreg(hw, phy_regarray_table[i], MASKDWORD, | ||
217 | phy_regarray_table[i + 1]); | ||
218 | udelay(1); | ||
219 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
220 | ("The phy_regarray_table[0] is %x" | ||
221 | " Rtl819XPHY_REGArray[1] is %x\n", | ||
222 | phy_regarray_table[i], | ||
223 | phy_regarray_table[i + 1])); | ||
224 | } | ||
225 | } else if (configtype == BASEBAND_CONFIG_AGC_TAB) { | ||
226 | for (i = 0; i < agctab_arraylen; i = i + 2) { | ||
227 | rtl_set_bbreg(hw, agctab_array_table[i], MASKDWORD, | ||
228 | agctab_array_table[i + 1]); | ||
229 | udelay(1); | ||
230 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
231 | ("The agctab_array_table[0] is " | ||
232 | "%x Rtl819XPHY_REGArray[1] is %x\n", | ||
233 | agctab_array_table[i], | ||
234 | agctab_array_table[i + 1])); | ||
235 | } | ||
236 | } | ||
237 | return true; | ||
238 | } | ||
239 | |||
240 | bool _rtl92ce_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
241 | u8 configtype) | ||
242 | { | ||
243 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
244 | int i; | ||
245 | u32 *phy_regarray_table_pg; | ||
246 | u16 phy_regarray_pg_len; | ||
247 | |||
248 | phy_regarray_pg_len = PHY_REG_ARRAY_PGLENGTH; | ||
249 | phy_regarray_table_pg = RTL8192CEPHY_REG_ARRAY_PG; | ||
250 | |||
251 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
252 | for (i = 0; i < phy_regarray_pg_len; i = i + 3) { | ||
253 | if (phy_regarray_table_pg[i] == 0xfe) | ||
254 | mdelay(50); | ||
255 | else if (phy_regarray_table_pg[i] == 0xfd) | ||
256 | mdelay(5); | ||
257 | else if (phy_regarray_table_pg[i] == 0xfc) | ||
258 | mdelay(1); | ||
259 | else if (phy_regarray_table_pg[i] == 0xfb) | ||
260 | udelay(50); | ||
261 | else if (phy_regarray_table_pg[i] == 0xfa) | ||
262 | udelay(5); | ||
263 | else if (phy_regarray_table_pg[i] == 0xf9) | ||
264 | udelay(1); | ||
265 | |||
266 | _rtl92c_store_pwrIndex_diffrate_offset(hw, | ||
267 | phy_regarray_table_pg[i], | ||
268 | phy_regarray_table_pg[i + 1], | ||
269 | phy_regarray_table_pg[i + 2]); | ||
270 | } | ||
271 | } else { | ||
272 | |||
273 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, | ||
274 | ("configtype != BaseBand_Config_PHY_REG\n")); | ||
275 | } | ||
276 | return true; | ||
277 | } | ||
278 | |||
279 | bool rtl92c_phy_config_rf_with_headerfile(struct ieee80211_hw *hw, | ||
280 | enum radio_path rfpath) | ||
281 | { | ||
282 | |||
283 | int i; | ||
284 | bool rtstatus = true; | ||
285 | u32 *radioa_array_table; | ||
286 | u32 *radiob_array_table; | ||
287 | u16 radioa_arraylen, radiob_arraylen; | ||
288 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
289 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
290 | |||
291 | if (IS_92C_SERIAL(rtlhal->version)) { | ||
292 | radioa_arraylen = RADIOA_2TARRAYLENGTH; | ||
293 | radioa_array_table = RTL8192CERADIOA_2TARRAY; | ||
294 | radiob_arraylen = RADIOB_2TARRAYLENGTH; | ||
295 | radiob_array_table = RTL8192CE_RADIOB_2TARRAY; | ||
296 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
297 | ("Radio_A:RTL8192CERADIOA_2TARRAY\n")); | ||
298 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
299 | ("Radio_B:RTL8192CE_RADIOB_2TARRAY\n")); | ||
300 | } else { | ||
301 | radioa_arraylen = RADIOA_1TARRAYLENGTH; | ||
302 | radioa_array_table = RTL8192CE_RADIOA_1TARRAY; | ||
303 | radiob_arraylen = RADIOB_1TARRAYLENGTH; | ||
304 | radiob_array_table = RTL8192CE_RADIOB_1TARRAY; | ||
305 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
306 | ("Radio_A:RTL8192CE_RADIOA_1TARRAY\n")); | ||
307 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
308 | ("Radio_B:RTL8192CE_RADIOB_1TARRAY\n")); | ||
309 | } | ||
310 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Radio No %x\n", rfpath)); | ||
311 | rtstatus = true; | ||
312 | switch (rfpath) { | ||
313 | case RF90_PATH_A: | ||
314 | for (i = 0; i < radioa_arraylen; i = i + 2) { | ||
315 | if (radioa_array_table[i] == 0xfe) | ||
316 | mdelay(50); | ||
317 | else if (radioa_array_table[i] == 0xfd) | ||
318 | mdelay(5); | ||
319 | else if (radioa_array_table[i] == 0xfc) | ||
320 | mdelay(1); | ||
321 | else if (radioa_array_table[i] == 0xfb) | ||
322 | udelay(50); | ||
323 | else if (radioa_array_table[i] == 0xfa) | ||
324 | udelay(5); | ||
325 | else if (radioa_array_table[i] == 0xf9) | ||
326 | udelay(1); | ||
327 | else { | ||
328 | rtl_set_rfreg(hw, rfpath, radioa_array_table[i], | ||
329 | RFREG_OFFSET_MASK, | ||
330 | radioa_array_table[i + 1]); | ||
331 | udelay(1); | ||
332 | } | ||
333 | } | ||
334 | break; | ||
335 | case RF90_PATH_B: | ||
336 | for (i = 0; i < radiob_arraylen; i = i + 2) { | ||
337 | if (radiob_array_table[i] == 0xfe) { | ||
338 | mdelay(50); | ||
339 | } else if (radiob_array_table[i] == 0xfd) | ||
340 | mdelay(5); | ||
341 | else if (radiob_array_table[i] == 0xfc) | ||
342 | mdelay(1); | ||
343 | else if (radiob_array_table[i] == 0xfb) | ||
344 | udelay(50); | ||
345 | else if (radiob_array_table[i] == 0xfa) | ||
346 | udelay(5); | ||
347 | else if (radiob_array_table[i] == 0xf9) | ||
348 | udelay(1); | ||
349 | else { | ||
350 | rtl_set_rfreg(hw, rfpath, radiob_array_table[i], | ||
351 | RFREG_OFFSET_MASK, | ||
352 | radiob_array_table[i + 1]); | ||
353 | udelay(1); | ||
354 | } | ||
355 | } | ||
356 | break; | ||
357 | case RF90_PATH_C: | ||
358 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
359 | ("switch case not process\n")); | ||
360 | break; | ||
361 | case RF90_PATH_D: | ||
362 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
363 | ("switch case not process\n")); | ||
364 | break; | ||
365 | } | ||
366 | return true; | ||
367 | } | ||
368 | |||
369 | void rtl92ce_phy_set_bw_mode_callback(struct ieee80211_hw *hw) | ||
370 | { | ||
371 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
372 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
373 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
374 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
375 | u8 reg_bw_opmode; | ||
376 | u8 reg_prsr_rsc; | ||
377 | |||
378 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, | ||
379 | ("Switch to %s bandwidth\n", | ||
380 | rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20 ? | ||
381 | "20MHz" : "40MHz")) | ||
382 | |||
383 | if (is_hal_stop(rtlhal)) { | ||
384 | rtlphy->set_bwmode_inprogress = false; | ||
385 | return; | ||
386 | } | ||
387 | |||
388 | reg_bw_opmode = rtl_read_byte(rtlpriv, REG_BWOPMODE); | ||
389 | reg_prsr_rsc = rtl_read_byte(rtlpriv, REG_RRSR + 2); | ||
390 | |||
391 | switch (rtlphy->current_chan_bw) { | ||
392 | case HT_CHANNEL_WIDTH_20: | ||
393 | reg_bw_opmode |= BW_OPMODE_20MHZ; | ||
394 | rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode); | ||
395 | break; | ||
396 | case HT_CHANNEL_WIDTH_20_40: | ||
397 | reg_bw_opmode &= ~BW_OPMODE_20MHZ; | ||
398 | rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode); | ||
399 | reg_prsr_rsc = | ||
400 | (reg_prsr_rsc & 0x90) | (mac->cur_40_prime_sc << 5); | ||
401 | rtl_write_byte(rtlpriv, REG_RRSR + 2, reg_prsr_rsc); | ||
402 | break; | ||
403 | default: | ||
404 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
405 | ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw)); | ||
406 | break; | ||
407 | } | ||
408 | |||
409 | switch (rtlphy->current_chan_bw) { | ||
410 | case HT_CHANNEL_WIDTH_20: | ||
411 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x0); | ||
412 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x0); | ||
413 | rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 1); | ||
414 | break; | ||
415 | case HT_CHANNEL_WIDTH_20_40: | ||
416 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x1); | ||
417 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x1); | ||
418 | |||
419 | rtl_set_bbreg(hw, RCCK0_SYSTEM, BCCK_SIDEBAND, | ||
420 | (mac->cur_40_prime_sc >> 1)); | ||
421 | rtl_set_bbreg(hw, ROFDM1_LSTF, 0xC00, mac->cur_40_prime_sc); | ||
422 | rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 0); | ||
423 | |||
424 | rtl_set_bbreg(hw, 0x818, (BIT(26) | BIT(27)), | ||
425 | (mac->cur_40_prime_sc == | ||
426 | HAL_PRIME_CHNL_OFFSET_LOWER) ? 2 : 1); | ||
427 | break; | ||
428 | default: | ||
429 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
430 | ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw)); | ||
431 | break; | ||
432 | } | ||
433 | rtl92ce_phy_rf6052_set_bandwidth(hw, rtlphy->current_chan_bw); | ||
434 | rtlphy->set_bwmode_inprogress = false; | ||
435 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n")); | ||
436 | } | ||
437 | |||
438 | void _rtl92ce_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t) | ||
439 | { | ||
440 | u8 tmpreg; | ||
441 | u32 rf_a_mode = 0, rf_b_mode = 0, lc_cal; | ||
442 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
443 | |||
444 | tmpreg = rtl_read_byte(rtlpriv, 0xd03); | ||
445 | |||
446 | if ((tmpreg & 0x70) != 0) | ||
447 | rtl_write_byte(rtlpriv, 0xd03, tmpreg & 0x8F); | ||
448 | else | ||
449 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
450 | |||
451 | if ((tmpreg & 0x70) != 0) { | ||
452 | rf_a_mode = rtl_get_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS); | ||
453 | |||
454 | if (is2t) | ||
455 | rf_b_mode = rtl_get_rfreg(hw, RF90_PATH_B, 0x00, | ||
456 | MASK12BITS); | ||
457 | |||
458 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS, | ||
459 | (rf_a_mode & 0x8FFFF) | 0x10000); | ||
460 | |||
461 | if (is2t) | ||
462 | rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS, | ||
463 | (rf_b_mode & 0x8FFFF) | 0x10000); | ||
464 | } | ||
465 | lc_cal = rtl_get_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS); | ||
466 | |||
467 | rtl_set_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS, lc_cal | 0x08000); | ||
468 | |||
469 | mdelay(100); | ||
470 | |||
471 | if ((tmpreg & 0x70) != 0) { | ||
472 | rtl_write_byte(rtlpriv, 0xd03, tmpreg); | ||
473 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS, rf_a_mode); | ||
474 | |||
475 | if (is2t) | ||
476 | rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS, | ||
477 | rf_b_mode); | ||
478 | } else { | ||
479 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00); | ||
480 | } | ||
481 | } | ||
482 | |||
483 | static void _rtl92ce_phy_set_rf_sleep(struct ieee80211_hw *hw) | ||
484 | { | ||
485 | u32 u4b_tmp; | ||
486 | u8 delay = 5; | ||
487 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
488 | |||
489 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
490 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, RFREG_OFFSET_MASK, 0x00); | ||
491 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x40); | ||
492 | u4b_tmp = rtl_get_rfreg(hw, RF90_PATH_A, 0, RFREG_OFFSET_MASK); | ||
493 | while (u4b_tmp != 0 && delay > 0) { | ||
494 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x0); | ||
495 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, RFREG_OFFSET_MASK, 0x00); | ||
496 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x40); | ||
497 | u4b_tmp = rtl_get_rfreg(hw, RF90_PATH_A, 0, RFREG_OFFSET_MASK); | ||
498 | delay--; | ||
499 | } | ||
500 | if (delay == 0) { | ||
501 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, 0x00); | ||
502 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
503 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE3); | ||
504 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00); | ||
505 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
506 | ("Switch RF timeout !!!.\n")); | ||
507 | return; | ||
508 | } | ||
509 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, 0xE2); | ||
510 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x22); | ||
511 | } | ||
512 | |||
513 | static bool _rtl92ce_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
514 | enum rf_pwrstate rfpwr_state) | ||
515 | { | ||
516 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
517 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
518 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
519 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
520 | bool bresult = true; | ||
521 | u8 i, queue_id; | ||
522 | struct rtl8192_tx_ring *ring = NULL; | ||
523 | |||
524 | ppsc->set_rfpowerstate_inprogress = true; | ||
525 | switch (rfpwr_state) { | ||
526 | case ERFON:{ | ||
527 | if ((ppsc->rfpwr_state == ERFOFF) && | ||
528 | RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC)) { | ||
529 | bool rtstatus; | ||
530 | u32 InitializeCount = 0; | ||
531 | do { | ||
532 | InitializeCount++; | ||
533 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
534 | ("IPS Set eRf nic enable\n")); | ||
535 | rtstatus = rtl_ps_enable_nic(hw); | ||
536 | } while ((rtstatus != true) | ||
537 | && (InitializeCount < 10)); | ||
538 | RT_CLEAR_PS_LEVEL(ppsc, | ||
539 | RT_RF_OFF_LEVL_HALT_NIC); | ||
540 | } else { | ||
541 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
542 | ("Set ERFON sleeped:%d ms\n", | ||
543 | jiffies_to_msecs(jiffies - | ||
544 | ppsc-> | ||
545 | last_sleep_jiffies))); | ||
546 | ppsc->last_awake_jiffies = jiffies; | ||
547 | rtl92ce_phy_set_rf_on(hw); | ||
548 | } | ||
549 | if (mac->link_state == MAC80211_LINKED) { | ||
550 | rtlpriv->cfg->ops->led_control(hw, | ||
551 | LED_CTL_LINK); | ||
552 | } else { | ||
553 | rtlpriv->cfg->ops->led_control(hw, | ||
554 | LED_CTL_NO_LINK); | ||
555 | } | ||
556 | break; | ||
557 | } | ||
558 | case ERFOFF:{ | ||
559 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC) { | ||
560 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
561 | ("IPS Set eRf nic disable\n")); | ||
562 | rtl_ps_disable_nic(hw); | ||
563 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
564 | } else { | ||
565 | if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS) { | ||
566 | rtlpriv->cfg->ops->led_control(hw, | ||
567 | LED_CTL_NO_LINK); | ||
568 | } else { | ||
569 | rtlpriv->cfg->ops->led_control(hw, | ||
570 | LED_CTL_POWER_OFF); | ||
571 | } | ||
572 | } | ||
573 | break; | ||
574 | } | ||
575 | case ERFSLEEP:{ | ||
576 | if (ppsc->rfpwr_state == ERFOFF) | ||
577 | break; | ||
578 | for (queue_id = 0, i = 0; | ||
579 | queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) { | ||
580 | ring = &pcipriv->dev.tx_ring[queue_id]; | ||
581 | if (skb_queue_len(&ring->queue) == 0) { | ||
582 | queue_id++; | ||
583 | continue; | ||
584 | } else { | ||
585 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
586 | ("eRf Off/Sleep: %d times " | ||
587 | "TcbBusyQueue[%d] =%d before " | ||
588 | "doze!\n", (i + 1), queue_id, | ||
589 | skb_queue_len(&ring->queue))); | ||
590 | |||
591 | udelay(10); | ||
592 | i++; | ||
593 | } | ||
594 | if (i >= MAX_DOZE_WAITING_TIMES_9x) { | ||
595 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
596 | ("\n ERFSLEEP: %d times " | ||
597 | "TcbBusyQueue[%d] = %d !\n", | ||
598 | MAX_DOZE_WAITING_TIMES_9x, | ||
599 | queue_id, | ||
600 | skb_queue_len(&ring->queue))); | ||
601 | break; | ||
602 | } | ||
603 | } | ||
604 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
605 | ("Set ERFSLEEP awaked:%d ms\n", | ||
606 | jiffies_to_msecs(jiffies - | ||
607 | ppsc->last_awake_jiffies))); | ||
608 | ppsc->last_sleep_jiffies = jiffies; | ||
609 | _rtl92ce_phy_set_rf_sleep(hw); | ||
610 | break; | ||
611 | } | ||
612 | default: | ||
613 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
614 | ("switch case not process\n")); | ||
615 | bresult = false; | ||
616 | break; | ||
617 | } | ||
618 | if (bresult) | ||
619 | ppsc->rfpwr_state = rfpwr_state; | ||
620 | ppsc->set_rfpowerstate_inprogress = false; | ||
621 | return bresult; | ||
622 | } | ||
623 | |||
624 | bool rtl92c_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
625 | enum rf_pwrstate rfpwr_state) | ||
626 | { | ||
627 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
628 | |||
629 | bool bresult = false; | ||
630 | |||
631 | if (rfpwr_state == ppsc->rfpwr_state) | ||
632 | return bresult; | ||
633 | bresult = _rtl92ce_phy_set_rf_power_state(hw, rfpwr_state); | ||
634 | return bresult; | ||
635 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/phy.h b/drivers/net/wireless/rtlwifi/rtl8192ce/phy.h new file mode 100644 index 000000000000..be2c92adef33 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/phy.h | |||
@@ -0,0 +1,262 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_PHY_H__ | ||
31 | #define __RTL92C_PHY_H__ | ||
32 | |||
33 | #define MAX_PRECMD_CNT 16 | ||
34 | #define MAX_RFDEPENDCMD_CNT 16 | ||
35 | #define MAX_POSTCMD_CNT 16 | ||
36 | |||
37 | #define MAX_DOZE_WAITING_TIMES_9x 64 | ||
38 | |||
39 | #define RT_CANNOT_IO(hw) false | ||
40 | #define HIGHPOWER_RADIOA_ARRAYLEN 22 | ||
41 | |||
42 | #define IQK_ADDA_REG_NUM 16 | ||
43 | #define MAX_TOLERANCE 5 | ||
44 | #define IQK_DELAY_TIME 1 | ||
45 | |||
46 | #define APK_BB_REG_NUM 5 | ||
47 | #define APK_AFE_REG_NUM 16 | ||
48 | #define APK_CURVE_REG_NUM 4 | ||
49 | #define PATH_NUM 2 | ||
50 | |||
51 | #define LOOP_LIMIT 5 | ||
52 | #define MAX_STALL_TIME 50 | ||
53 | #define AntennaDiversityValue 0x80 | ||
54 | #define MAX_TXPWR_IDX_NMODE_92S 63 | ||
55 | #define Reset_Cnt_Limit 3 | ||
56 | |||
57 | #define IQK_ADDA_REG_NUM 16 | ||
58 | #define IQK_MAC_REG_NUM 4 | ||
59 | |||
60 | #define IQK_DELAY_TIME 1 | ||
61 | |||
62 | #define RF90_PATH_MAX 2 | ||
63 | |||
64 | #define CT_OFFSET_MAC_ADDR 0X16 | ||
65 | |||
66 | #define CT_OFFSET_CCK_TX_PWR_IDX 0x5A | ||
67 | #define CT_OFFSET_HT401S_TX_PWR_IDX 0x60 | ||
68 | #define CT_OFFSET_HT402S_TX_PWR_IDX_DIF 0x66 | ||
69 | #define CT_OFFSET_HT20_TX_PWR_IDX_DIFF 0x69 | ||
70 | #define CT_OFFSET_OFDM_TX_PWR_IDX_DIFF 0x6C | ||
71 | |||
72 | #define CT_OFFSET_HT40_MAX_PWR_OFFSET 0x6F | ||
73 | #define CT_OFFSET_HT20_MAX_PWR_OFFSET 0x72 | ||
74 | |||
75 | #define CT_OFFSET_CHANNEL_PLAH 0x75 | ||
76 | #define CT_OFFSET_THERMAL_METER 0x78 | ||
77 | #define CT_OFFSET_RF_OPTION 0x79 | ||
78 | #define CT_OFFSET_VERSION 0x7E | ||
79 | #define CT_OFFSET_CUSTOMER_ID 0x7F | ||
80 | |||
81 | #define RTL92C_MAX_PATH_NUM 2 | ||
82 | |||
83 | enum swchnlcmd_id { | ||
84 | CMDID_END, | ||
85 | CMDID_SET_TXPOWEROWER_LEVEL, | ||
86 | CMDID_BBREGWRITE10, | ||
87 | CMDID_WRITEPORT_ULONG, | ||
88 | CMDID_WRITEPORT_USHORT, | ||
89 | CMDID_WRITEPORT_UCHAR, | ||
90 | CMDID_RF_WRITEREG, | ||
91 | }; | ||
92 | |||
93 | struct swchnlcmd { | ||
94 | enum swchnlcmd_id cmdid; | ||
95 | u32 para1; | ||
96 | u32 para2; | ||
97 | u32 msdelay; | ||
98 | }; | ||
99 | |||
100 | enum hw90_block_e { | ||
101 | HW90_BLOCK_MAC = 0, | ||
102 | HW90_BLOCK_PHY0 = 1, | ||
103 | HW90_BLOCK_PHY1 = 2, | ||
104 | HW90_BLOCK_RF = 3, | ||
105 | HW90_BLOCK_MAXIMUM = 4, | ||
106 | }; | ||
107 | |||
108 | enum baseband_config_type { | ||
109 | BASEBAND_CONFIG_PHY_REG = 0, | ||
110 | BASEBAND_CONFIG_AGC_TAB = 1, | ||
111 | }; | ||
112 | |||
113 | enum ra_offset_area { | ||
114 | RA_OFFSET_LEGACY_OFDM1, | ||
115 | RA_OFFSET_LEGACY_OFDM2, | ||
116 | RA_OFFSET_HT_OFDM1, | ||
117 | RA_OFFSET_HT_OFDM2, | ||
118 | RA_OFFSET_HT_OFDM3, | ||
119 | RA_OFFSET_HT_OFDM4, | ||
120 | RA_OFFSET_HT_CCK, | ||
121 | }; | ||
122 | |||
123 | enum antenna_path { | ||
124 | ANTENNA_NONE, | ||
125 | ANTENNA_D, | ||
126 | ANTENNA_C, | ||
127 | ANTENNA_CD, | ||
128 | ANTENNA_B, | ||
129 | ANTENNA_BD, | ||
130 | ANTENNA_BC, | ||
131 | ANTENNA_BCD, | ||
132 | ANTENNA_A, | ||
133 | ANTENNA_AD, | ||
134 | ANTENNA_AC, | ||
135 | ANTENNA_ACD, | ||
136 | ANTENNA_AB, | ||
137 | ANTENNA_ABD, | ||
138 | ANTENNA_ABC, | ||
139 | ANTENNA_ABCD | ||
140 | }; | ||
141 | |||
142 | struct r_antenna_select_ofdm { | ||
143 | u32 r_tx_antenna:4; | ||
144 | u32 r_ant_l:4; | ||
145 | u32 r_ant_non_ht:4; | ||
146 | u32 r_ant_ht1:4; | ||
147 | u32 r_ant_ht2:4; | ||
148 | u32 r_ant_ht_s1:4; | ||
149 | u32 r_ant_non_ht_s1:4; | ||
150 | u32 ofdm_txsc:2; | ||
151 | u32 reserved:2; | ||
152 | }; | ||
153 | |||
154 | struct r_antenna_select_cck { | ||
155 | u8 r_cckrx_enable_2:2; | ||
156 | u8 r_cckrx_enable:2; | ||
157 | u8 r_ccktx_enable:4; | ||
158 | }; | ||
159 | |||
160 | struct efuse_contents { | ||
161 | u8 mac_addr[ETH_ALEN]; | ||
162 | u8 cck_tx_power_idx[6]; | ||
163 | u8 ht40_1s_tx_power_idx[6]; | ||
164 | u8 ht40_2s_tx_power_idx_diff[3]; | ||
165 | u8 ht20_tx_power_idx_diff[3]; | ||
166 | u8 ofdm_tx_power_idx_diff[3]; | ||
167 | u8 ht40_max_power_offset[3]; | ||
168 | u8 ht20_max_power_offset[3]; | ||
169 | u8 channel_plan; | ||
170 | u8 thermal_meter; | ||
171 | u8 rf_option[5]; | ||
172 | u8 version; | ||
173 | u8 oem_id; | ||
174 | u8 regulatory; | ||
175 | }; | ||
176 | |||
177 | struct tx_power_struct { | ||
178 | u8 cck[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
179 | u8 ht40_1s[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
180 | u8 ht40_2s[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
181 | u8 ht20_diff[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
182 | u8 legacy_ht_diff[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
183 | u8 legacy_ht_txpowerdiff; | ||
184 | u8 groupht20[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
185 | u8 groupht40[RTL92C_MAX_PATH_NUM][CHANNEL_MAX_NUMBER]; | ||
186 | u8 pwrgroup_cnt; | ||
187 | u32 mcs_original_offset[4][16]; | ||
188 | }; | ||
189 | |||
190 | bool rtl92c_phy_bb_config(struct ieee80211_hw *hw); | ||
191 | u32 rtl92c_phy_query_bb_reg(struct ieee80211_hw *hw, | ||
192 | u32 regaddr, u32 bitmask); | ||
193 | void rtl92c_phy_set_bb_reg(struct ieee80211_hw *hw, | ||
194 | u32 regaddr, u32 bitmask, u32 data); | ||
195 | u32 rtl92c_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
196 | enum radio_path rfpath, u32 regaddr, | ||
197 | u32 bitmask); | ||
198 | extern void rtl92ce_phy_set_rf_reg(struct ieee80211_hw *hw, | ||
199 | enum radio_path rfpath, u32 regaddr, | ||
200 | u32 bitmask, u32 data); | ||
201 | bool rtl92c_phy_mac_config(struct ieee80211_hw *hw); | ||
202 | bool rtl92ce_phy_bb_config(struct ieee80211_hw *hw); | ||
203 | bool rtl92c_phy_rf_config(struct ieee80211_hw *hw); | ||
204 | bool rtl92c_phy_config_rf_with_feaderfile(struct ieee80211_hw *hw, | ||
205 | enum radio_path rfpath); | ||
206 | void rtl92c_phy_get_hw_reg_originalvalue(struct ieee80211_hw *hw); | ||
207 | void rtl92c_phy_get_txpower_level(struct ieee80211_hw *hw, | ||
208 | long *powerlevel); | ||
209 | void rtl92c_phy_set_txpower_level(struct ieee80211_hw *hw, u8 channel); | ||
210 | bool rtl92c_phy_update_txpower_dbm(struct ieee80211_hw *hw, | ||
211 | long power_indbm); | ||
212 | void rtl92c_phy_scan_operation_backup(struct ieee80211_hw *hw, | ||
213 | u8 operation); | ||
214 | void rtl92c_phy_set_bw_mode(struct ieee80211_hw *hw, | ||
215 | enum nl80211_channel_type ch_type); | ||
216 | void rtl92c_phy_sw_chnl_callback(struct ieee80211_hw *hw); | ||
217 | u8 rtl92c_phy_sw_chnl(struct ieee80211_hw *hw); | ||
218 | void rtl92c_phy_iq_calibrate(struct ieee80211_hw *hw, bool b_recovery); | ||
219 | void rtl92c_phy_set_beacon_hw_reg(struct ieee80211_hw *hw, | ||
220 | u16 beaconinterval); | ||
221 | void rtl92c_phy_ap_calibrate(struct ieee80211_hw *hw, char delta); | ||
222 | void rtl92c_phy_lc_calibrate(struct ieee80211_hw *hw); | ||
223 | void _rtl92ce_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t); | ||
224 | void rtl92c_phy_set_rfpath_switch(struct ieee80211_hw *hw, bool bmain); | ||
225 | bool rtl92c_phy_config_rf_with_headerfile(struct ieee80211_hw *hw, | ||
226 | enum radio_path rfpath); | ||
227 | bool rtl8192_phy_check_is_legal_rfpath(struct ieee80211_hw *hw, | ||
228 | u32 rfpath); | ||
229 | bool rtl92c_phy_set_io_cmd(struct ieee80211_hw *hw, enum io_type iotype); | ||
230 | bool rtl92ce_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
231 | enum rf_pwrstate rfpwr_state); | ||
232 | void rtl92ce_phy_set_rf_on(struct ieee80211_hw *hw); | ||
233 | bool rtl92c_phy_set_io_cmd(struct ieee80211_hw *hw, enum io_type iotype); | ||
234 | void rtl92c_phy_set_io(struct ieee80211_hw *hw); | ||
235 | void rtl92c_bb_block_on(struct ieee80211_hw *hw); | ||
236 | u32 _rtl92c_phy_rf_serial_read(struct ieee80211_hw *hw, | ||
237 | enum radio_path rfpath, u32 offset); | ||
238 | u32 _rtl92c_phy_fw_rf_serial_read(struct ieee80211_hw *hw, | ||
239 | enum radio_path rfpath, u32 offset); | ||
240 | u32 _rtl92c_phy_calculate_bit_shift(u32 bitmask); | ||
241 | void _rtl92c_phy_rf_serial_write(struct ieee80211_hw *hw, | ||
242 | enum radio_path rfpath, u32 offset, | ||
243 | u32 data); | ||
244 | void _rtl92c_phy_fw_rf_serial_write(struct ieee80211_hw *hw, | ||
245 | enum radio_path rfpath, u32 offset, | ||
246 | u32 data); | ||
247 | void _rtl92c_store_pwrIndex_diffrate_offset(struct ieee80211_hw *hw, | ||
248 | u32 regaddr, u32 bitmask, | ||
249 | u32 data); | ||
250 | bool _rtl92ce_phy_config_mac_with_headerfile(struct ieee80211_hw *hw); | ||
251 | void _rtl92c_phy_init_bb_rf_register_definition(struct ieee80211_hw *hw); | ||
252 | bool _rtl92c_phy_bb8192c_config_parafile(struct ieee80211_hw *hw); | ||
253 | void _rtl92c_phy_set_rf_sleep(struct ieee80211_hw *hw); | ||
254 | bool rtl92c_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
255 | enum rf_pwrstate rfpwr_state); | ||
256 | bool _rtl92ce_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
257 | u8 configtype); | ||
258 | bool _rtl92ce_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
259 | u8 configtype); | ||
260 | void rtl92ce_phy_set_bw_mode_callback(struct ieee80211_hw *hw); | ||
261 | |||
262 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/reg.h b/drivers/net/wireless/rtlwifi/rtl8192ce/reg.h new file mode 100644 index 000000000000..598cecc63f41 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/reg.h | |||
@@ -0,0 +1,2090 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_REG_H__ | ||
31 | #define __RTL92C_REG_H__ | ||
32 | |||
33 | #define REG_SYS_ISO_CTRL 0x0000 | ||
34 | #define REG_SYS_FUNC_EN 0x0002 | ||
35 | #define REG_APS_FSMCO 0x0004 | ||
36 | #define REG_SYS_CLKR 0x0008 | ||
37 | #define REG_9346CR 0x000A | ||
38 | #define REG_EE_VPD 0x000C | ||
39 | #define REG_AFE_MISC 0x0010 | ||
40 | #define REG_SPS0_CTRL 0x0011 | ||
41 | #define REG_SPS_OCP_CFG 0x0018 | ||
42 | #define REG_RSV_CTRL 0x001C | ||
43 | #define REG_RF_CTRL 0x001F | ||
44 | #define REG_LDOA15_CTRL 0x0020 | ||
45 | #define REG_LDOV12D_CTRL 0x0021 | ||
46 | #define REG_LDOHCI12_CTRL 0x0022 | ||
47 | #define REG_LPLDO_CTRL 0x0023 | ||
48 | #define REG_AFE_XTAL_CTRL 0x0024 | ||
49 | #define REG_AFE_PLL_CTRL 0x0028 | ||
50 | #define REG_EFUSE_CTRL 0x0030 | ||
51 | #define REG_EFUSE_TEST 0x0034 | ||
52 | #define REG_PWR_DATA 0x0038 | ||
53 | #define REG_CAL_TIMER 0x003C | ||
54 | #define REG_ACLK_MON 0x003E | ||
55 | #define REG_GPIO_MUXCFG 0x0040 | ||
56 | #define REG_GPIO_IO_SEL 0x0042 | ||
57 | #define REG_MAC_PINMUX_CFG 0x0043 | ||
58 | #define REG_GPIO_PIN_CTRL 0x0044 | ||
59 | #define REG_GPIO_INTM 0x0048 | ||
60 | #define REG_LEDCFG0 0x004C | ||
61 | #define REG_LEDCFG1 0x004D | ||
62 | #define REG_LEDCFG2 0x004E | ||
63 | #define REG_LEDCFG3 0x004F | ||
64 | #define REG_FSIMR 0x0050 | ||
65 | #define REG_FSISR 0x0054 | ||
66 | #define REG_HSIMR 0x0058 | ||
67 | #define REG_HSISR 0x005c | ||
68 | |||
69 | /* RTL8723 WIFI/BT/GPS Multi-Function GPIO Pin Control. */ | ||
70 | #define REG_GPIO_PIN_CTRL_2 0x0060 | ||
71 | /* RTL8723 WIFI/BT/GPS Multi-Function GPIO Select. */ | ||
72 | #define REG_GPIO_IO_SEL_2 0x0062 | ||
73 | /* RTL8723 WIFI/BT/GPS Multi-Function control source. */ | ||
74 | #define REG_MULTI_FUNC_CTRL 0x0068 | ||
75 | |||
76 | #define REG_MCUFWDL 0x0080 | ||
77 | |||
78 | #define REG_HMEBOX_EXT_0 0x0088 | ||
79 | #define REG_HMEBOX_EXT_1 0x008A | ||
80 | #define REG_HMEBOX_EXT_2 0x008C | ||
81 | #define REG_HMEBOX_EXT_3 0x008E | ||
82 | |||
83 | #define REG_BIST_SCAN 0x00D0 | ||
84 | #define REG_BIST_RPT 0x00D4 | ||
85 | #define REG_BIST_ROM_RPT 0x00D8 | ||
86 | #define REG_USB_SIE_INTF 0x00E0 | ||
87 | #define REG_PCIE_MIO_INTF 0x00E4 | ||
88 | #define REG_PCIE_MIO_INTD 0x00E8 | ||
89 | #define REG_HPON_FSM 0x00EC | ||
90 | #define REG_SYS_CFG 0x00F0 | ||
91 | #define REG_GPIO_OUTSTS 0x00F4 /* For RTL8723 only.*/ | ||
92 | |||
93 | #define REG_CR 0x0100 | ||
94 | #define REG_PBP 0x0104 | ||
95 | #define REG_TRXDMA_CTRL 0x010C | ||
96 | #define REG_TRXFF_BNDY 0x0114 | ||
97 | #define REG_TRXFF_STATUS 0x0118 | ||
98 | #define REG_RXFF_PTR 0x011C | ||
99 | #define REG_HIMR 0x0120 | ||
100 | #define REG_HISR 0x0124 | ||
101 | #define REG_HIMRE 0x0128 | ||
102 | #define REG_HISRE 0x012C | ||
103 | #define REG_CPWM 0x012F | ||
104 | #define REG_FWIMR 0x0130 | ||
105 | #define REG_FWISR 0x0134 | ||
106 | #define REG_PKTBUF_DBG_CTRL 0x0140 | ||
107 | #define REG_PKTBUF_DBG_DATA_L 0x0144 | ||
108 | #define REG_PKTBUF_DBG_DATA_H 0x0148 | ||
109 | |||
110 | #define REG_TC0_CTRL 0x0150 | ||
111 | #define REG_TC1_CTRL 0x0154 | ||
112 | #define REG_TC2_CTRL 0x0158 | ||
113 | #define REG_TC3_CTRL 0x015C | ||
114 | #define REG_TC4_CTRL 0x0160 | ||
115 | #define REG_TCUNIT_BASE 0x0164 | ||
116 | #define REG_MBIST_START 0x0174 | ||
117 | #define REG_MBIST_DONE 0x0178 | ||
118 | #define REG_MBIST_FAIL 0x017C | ||
119 | #define REG_C2HEVT_MSG_NORMAL 0x01A0 | ||
120 | #define REG_C2HEVT_MSG_TEST 0x01B8 | ||
121 | #define REG_C2HEVT_CLEAR 0x01BF | ||
122 | #define REG_MCUTST_1 0x01c0 | ||
123 | #define REG_FMETHR 0x01C8 | ||
124 | #define REG_HMETFR 0x01CC | ||
125 | #define REG_HMEBOX_0 0x01D0 | ||
126 | #define REG_HMEBOX_1 0x01D4 | ||
127 | #define REG_HMEBOX_2 0x01D8 | ||
128 | #define REG_HMEBOX_3 0x01DC | ||
129 | |||
130 | #define REG_LLT_INIT 0x01E0 | ||
131 | #define REG_BB_ACCEESS_CTRL 0x01E8 | ||
132 | #define REG_BB_ACCESS_DATA 0x01EC | ||
133 | |||
134 | #define REG_RQPN 0x0200 | ||
135 | #define REG_FIFOPAGE 0x0204 | ||
136 | #define REG_TDECTRL 0x0208 | ||
137 | #define REG_TXDMA_OFFSET_CHK 0x020C | ||
138 | #define REG_TXDMA_STATUS 0x0210 | ||
139 | #define REG_RQPN_NPQ 0x0214 | ||
140 | |||
141 | #define REG_RXDMA_AGG_PG_TH 0x0280 | ||
142 | #define REG_RXPKT_NUM 0x0284 | ||
143 | #define REG_RXDMA_STATUS 0x0288 | ||
144 | |||
145 | #define REG_PCIE_CTRL_REG 0x0300 | ||
146 | #define REG_INT_MIG 0x0304 | ||
147 | #define REG_BCNQ_DESA 0x0308 | ||
148 | #define REG_HQ_DESA 0x0310 | ||
149 | #define REG_MGQ_DESA 0x0318 | ||
150 | #define REG_VOQ_DESA 0x0320 | ||
151 | #define REG_VIQ_DESA 0x0328 | ||
152 | #define REG_BEQ_DESA 0x0330 | ||
153 | #define REG_BKQ_DESA 0x0338 | ||
154 | #define REG_RX_DESA 0x0340 | ||
155 | #define REG_DBI 0x0348 | ||
156 | #define REG_MDIO 0x0354 | ||
157 | #define REG_DBG_SEL 0x0360 | ||
158 | #define REG_PCIE_HRPWM 0x0361 | ||
159 | #define REG_PCIE_HCPWM 0x0363 | ||
160 | #define REG_UART_CTRL 0x0364 | ||
161 | #define REG_UART_TX_DESA 0x0370 | ||
162 | #define REG_UART_RX_DESA 0x0378 | ||
163 | |||
164 | #define REG_HDAQ_DESA_NODEF 0x0000 | ||
165 | #define REG_CMDQ_DESA_NODEF 0x0000 | ||
166 | |||
167 | #define REG_VOQ_INFORMATION 0x0400 | ||
168 | #define REG_VIQ_INFORMATION 0x0404 | ||
169 | #define REG_BEQ_INFORMATION 0x0408 | ||
170 | #define REG_BKQ_INFORMATION 0x040C | ||
171 | #define REG_MGQ_INFORMATION 0x0410 | ||
172 | #define REG_HGQ_INFORMATION 0x0414 | ||
173 | #define REG_BCNQ_INFORMATION 0x0418 | ||
174 | |||
175 | #define REG_CPU_MGQ_INFORMATION 0x041C | ||
176 | #define REG_FWHW_TXQ_CTRL 0x0420 | ||
177 | #define REG_HWSEQ_CTRL 0x0423 | ||
178 | #define REG_TXPKTBUF_BCNQ_BDNY 0x0424 | ||
179 | #define REG_TXPKTBUF_MGQ_BDNY 0x0425 | ||
180 | #define REG_MULTI_BCNQ_EN 0x0426 | ||
181 | #define REG_MULTI_BCNQ_OFFSET 0x0427 | ||
182 | #define REG_SPEC_SIFS 0x0428 | ||
183 | #define REG_RL 0x042A | ||
184 | #define REG_DARFRC 0x0430 | ||
185 | #define REG_RARFRC 0x0438 | ||
186 | #define REG_RRSR 0x0440 | ||
187 | #define REG_ARFR0 0x0444 | ||
188 | #define REG_ARFR1 0x0448 | ||
189 | #define REG_ARFR2 0x044C | ||
190 | #define REG_ARFR3 0x0450 | ||
191 | #define REG_AGGLEN_LMT 0x0458 | ||
192 | #define REG_AMPDU_MIN_SPACE 0x045C | ||
193 | #define REG_TXPKTBUF_WMAC_LBK_BF_HD 0x045D | ||
194 | #define REG_FAST_EDCA_CTRL 0x0460 | ||
195 | #define REG_RD_RESP_PKT_TH 0x0463 | ||
196 | #define REG_INIRTS_RATE_SEL 0x0480 | ||
197 | #define REG_INIDATA_RATE_SEL 0x0484 | ||
198 | #define REG_POWER_STATUS 0x04A4 | ||
199 | #define REG_POWER_STAGE1 0x04B4 | ||
200 | #define REG_POWER_STAGE2 0x04B8 | ||
201 | #define REG_PKT_LIFE_TIME 0x04C0 | ||
202 | #define REG_STBC_SETTING 0x04C4 | ||
203 | #define REG_PROT_MODE_CTRL 0x04C8 | ||
204 | #define REG_BAR_MODE_CTRL 0x04CC | ||
205 | #define REG_RA_TRY_RATE_AGG_LMT 0x04CF | ||
206 | #define REG_NQOS_SEQ 0x04DC | ||
207 | #define REG_QOS_SEQ 0x04DE | ||
208 | #define REG_NEED_CPU_HANDLE 0x04E0 | ||
209 | #define REG_PKT_LOSE_RPT 0x04E1 | ||
210 | #define REG_PTCL_ERR_STATUS 0x04E2 | ||
211 | #define REG_DUMMY 0x04FC | ||
212 | |||
213 | #define REG_EDCA_VO_PARAM 0x0500 | ||
214 | #define REG_EDCA_VI_PARAM 0x0504 | ||
215 | #define REG_EDCA_BE_PARAM 0x0508 | ||
216 | #define REG_EDCA_BK_PARAM 0x050C | ||
217 | #define REG_BCNTCFG 0x0510 | ||
218 | #define REG_PIFS 0x0512 | ||
219 | #define REG_RDG_PIFS 0x0513 | ||
220 | #define REG_SIFS_CTX 0x0514 | ||
221 | #define REG_SIFS_TRX 0x0516 | ||
222 | #define REG_SIFS_CCK 0x0514 | ||
223 | #define REG_SIFS_OFDM 0x0516 | ||
224 | #define REG_AGGR_BREAK_TIME 0x051A | ||
225 | #define REG_SLOT 0x051B | ||
226 | #define REG_TX_PTCL_CTRL 0x0520 | ||
227 | #define REG_TXPAUSE 0x0522 | ||
228 | #define REG_DIS_TXREQ_CLR 0x0523 | ||
229 | #define REG_RD_CTRL 0x0524 | ||
230 | #define REG_TBTT_PROHIBIT 0x0540 | ||
231 | #define REG_RD_NAV_NXT 0x0544 | ||
232 | #define REG_NAV_PROT_LEN 0x0546 | ||
233 | #define REG_BCN_CTRL 0x0550 | ||
234 | #define REG_USTIME_TSF 0x0551 | ||
235 | #define REG_MBID_NUM 0x0552 | ||
236 | #define REG_DUAL_TSF_RST 0x0553 | ||
237 | #define REG_BCN_INTERVAL 0x0554 | ||
238 | #define REG_MBSSID_BCN_SPACE 0x0554 | ||
239 | #define REG_DRVERLYINT 0x0558 | ||
240 | #define REG_BCNDMATIM 0x0559 | ||
241 | #define REG_ATIMWND 0x055A | ||
242 | #define REG_BCN_MAX_ERR 0x055D | ||
243 | #define REG_RXTSF_OFFSET_CCK 0x055E | ||
244 | #define REG_RXTSF_OFFSET_OFDM 0x055F | ||
245 | #define REG_TSFTR 0x0560 | ||
246 | #define REG_INIT_TSFTR 0x0564 | ||
247 | #define REG_PSTIMER 0x0580 | ||
248 | #define REG_TIMER0 0x0584 | ||
249 | #define REG_TIMER1 0x0588 | ||
250 | #define REG_ACMHWCTRL 0x05C0 | ||
251 | #define REG_ACMRSTCTRL 0x05C1 | ||
252 | #define REG_ACMAVG 0x05C2 | ||
253 | #define REG_VO_ADMTIME 0x05C4 | ||
254 | #define REG_VI_ADMTIME 0x05C6 | ||
255 | #define REG_BE_ADMTIME 0x05C8 | ||
256 | #define REG_EDCA_RANDOM_GEN 0x05CC | ||
257 | #define REG_SCH_TXCMD 0x05D0 | ||
258 | |||
259 | #define REG_APSD_CTRL 0x0600 | ||
260 | #define REG_BWOPMODE 0x0603 | ||
261 | #define REG_TCR 0x0604 | ||
262 | #define REG_RCR 0x0608 | ||
263 | #define REG_RX_PKT_LIMIT 0x060C | ||
264 | #define REG_RX_DLK_TIME 0x060D | ||
265 | #define REG_RX_DRVINFO_SZ 0x060F | ||
266 | |||
267 | #define REG_MACID 0x0610 | ||
268 | #define REG_BSSID 0x0618 | ||
269 | #define REG_MAR 0x0620 | ||
270 | #define REG_MBIDCAMCFG 0x0628 | ||
271 | |||
272 | #define REG_USTIME_EDCA 0x0638 | ||
273 | #define REG_MAC_SPEC_SIFS 0x063A | ||
274 | #define REG_RESP_SIFS_CCK 0x063C | ||
275 | #define REG_RESP_SIFS_OFDM 0x063E | ||
276 | /* [15:8]SIFS_R2T_OFDM, [7:0]SIFS_R2T_CCK */ | ||
277 | #define REG_R2T_SIFS 0x063C | ||
278 | /* [15:8]SIFS_T2T_OFDM, [7:0]SIFS_T2T_CCK */ | ||
279 | #define REG_T2T_SIFS 0x063E | ||
280 | #define REG_ACKTO 0x0640 | ||
281 | #define REG_CTS2TO 0x0641 | ||
282 | #define REG_EIFS 0x0642 | ||
283 | |||
284 | #define REG_NAV_CTRL 0x0650 | ||
285 | #define REG_BACAMCMD 0x0654 | ||
286 | #define REG_BACAMCONTENT 0x0658 | ||
287 | #define REG_LBDLY 0x0660 | ||
288 | #define REG_FWDLY 0x0661 | ||
289 | #define REG_RXERR_RPT 0x0664 | ||
290 | #define REG_WMAC_TRXPTCL_CTL 0x0668 | ||
291 | |||
292 | #define REG_CAMCMD 0x0670 | ||
293 | #define REG_CAMWRITE 0x0674 | ||
294 | #define REG_CAMREAD 0x0678 | ||
295 | #define REG_CAMDBG 0x067C | ||
296 | #define REG_SECCFG 0x0680 | ||
297 | |||
298 | #define REG_WOW_CTRL 0x0690 | ||
299 | #define REG_PSSTATUS 0x0691 | ||
300 | #define REG_PS_RX_INFO 0x0692 | ||
301 | #define REG_LPNAV_CTRL 0x0694 | ||
302 | #define REG_WKFMCAM_CMD 0x0698 | ||
303 | #define REG_WKFMCAM_RWD 0x069C | ||
304 | #define REG_RXFLTMAP0 0x06A0 | ||
305 | #define REG_RXFLTMAP1 0x06A2 | ||
306 | #define REG_RXFLTMAP2 0x06A4 | ||
307 | #define REG_BCN_PSR_RPT 0x06A8 | ||
308 | #define REG_CALB32K_CTRL 0x06AC | ||
309 | #define REG_PKT_MON_CTRL 0x06B4 | ||
310 | #define REG_BT_COEX_TABLE 0x06C0 | ||
311 | #define REG_WMAC_RESP_TXINFO 0x06D8 | ||
312 | |||
313 | #define REG_USB_INFO 0xFE17 | ||
314 | #define REG_USB_SPECIAL_OPTION 0xFE55 | ||
315 | #define REG_USB_DMA_AGG_TO 0xFE5B | ||
316 | #define REG_USB_AGG_TO 0xFE5C | ||
317 | #define REG_USB_AGG_TH 0xFE5D | ||
318 | |||
319 | #define REG_TEST_USB_TXQS 0xFE48 | ||
320 | #define REG_TEST_SIE_VID 0xFE60 | ||
321 | #define REG_TEST_SIE_PID 0xFE62 | ||
322 | #define REG_TEST_SIE_OPTIONAL 0xFE64 | ||
323 | #define REG_TEST_SIE_CHIRP_K 0xFE65 | ||
324 | #define REG_TEST_SIE_PHY 0xFE66 | ||
325 | #define REG_TEST_SIE_MAC_ADDR 0xFE70 | ||
326 | #define REG_TEST_SIE_STRING 0xFE80 | ||
327 | |||
328 | #define REG_NORMAL_SIE_VID 0xFE60 | ||
329 | #define REG_NORMAL_SIE_PID 0xFE62 | ||
330 | #define REG_NORMAL_SIE_OPTIONAL 0xFE64 | ||
331 | #define REG_NORMAL_SIE_EP 0xFE65 | ||
332 | #define REG_NORMAL_SIE_PHY 0xFE68 | ||
333 | #define REG_NORMAL_SIE_MAC_ADDR 0xFE70 | ||
334 | #define REG_NORMAL_SIE_STRING 0xFE80 | ||
335 | |||
336 | #define CR9346 REG_9346CR | ||
337 | #define MSR (REG_CR + 2) | ||
338 | #define ISR REG_HISR | ||
339 | #define TSFR REG_TSFTR | ||
340 | |||
341 | #define MACIDR0 REG_MACID | ||
342 | #define MACIDR4 (REG_MACID + 4) | ||
343 | |||
344 | #define PBP REG_PBP | ||
345 | |||
346 | #define IDR0 MACIDR0 | ||
347 | #define IDR4 MACIDR4 | ||
348 | |||
349 | #define UNUSED_REGISTER 0x1BF | ||
350 | #define DCAM UNUSED_REGISTER | ||
351 | #define PSR UNUSED_REGISTER | ||
352 | #define BBADDR UNUSED_REGISTER | ||
353 | #define PHYDATAR UNUSED_REGISTER | ||
354 | |||
355 | #define INVALID_BBRF_VALUE 0x12345678 | ||
356 | |||
357 | #define MAX_MSS_DENSITY_2T 0x13 | ||
358 | #define MAX_MSS_DENSITY_1T 0x0A | ||
359 | |||
360 | #define CMDEEPROM_EN BIT(5) | ||
361 | #define CMDEEPROM_SEL BIT(4) | ||
362 | #define CMD9346CR_9356SEL BIT(4) | ||
363 | #define AUTOLOAD_EEPROM (CMDEEPROM_EN|CMDEEPROM_SEL) | ||
364 | #define AUTOLOAD_EFUSE CMDEEPROM_EN | ||
365 | |||
366 | #define GPIOSEL_GPIO 0 | ||
367 | #define GPIOSEL_ENBT BIT(5) | ||
368 | |||
369 | #define GPIO_IN REG_GPIO_PIN_CTRL | ||
370 | #define GPIO_OUT (REG_GPIO_PIN_CTRL+1) | ||
371 | #define GPIO_IO_SEL (REG_GPIO_PIN_CTRL+2) | ||
372 | #define GPIO_MOD (REG_GPIO_PIN_CTRL+3) | ||
373 | |||
374 | #define MSR_NOLINK 0x00 | ||
375 | #define MSR_ADHOC 0x01 | ||
376 | #define MSR_INFRA 0x02 | ||
377 | #define MSR_AP 0x03 | ||
378 | |||
379 | #define RRSR_RSC_OFFSET 21 | ||
380 | #define RRSR_SHORT_OFFSET 23 | ||
381 | #define RRSR_RSC_BW_40M 0x600000 | ||
382 | #define RRSR_RSC_UPSUBCHNL 0x400000 | ||
383 | #define RRSR_RSC_LOWSUBCHNL 0x200000 | ||
384 | #define RRSR_SHORT 0x800000 | ||
385 | #define RRSR_1M BIT(0) | ||
386 | #define RRSR_2M BIT(1) | ||
387 | #define RRSR_5_5M BIT(2) | ||
388 | #define RRSR_11M BIT(3) | ||
389 | #define RRSR_6M BIT(4) | ||
390 | #define RRSR_9M BIT(5) | ||
391 | #define RRSR_12M BIT(6) | ||
392 | #define RRSR_18M BIT(7) | ||
393 | #define RRSR_24M BIT(8) | ||
394 | #define RRSR_36M BIT(9) | ||
395 | #define RRSR_48M BIT(10) | ||
396 | #define RRSR_54M BIT(11) | ||
397 | #define RRSR_MCS0 BIT(12) | ||
398 | #define RRSR_MCS1 BIT(13) | ||
399 | #define RRSR_MCS2 BIT(14) | ||
400 | #define RRSR_MCS3 BIT(15) | ||
401 | #define RRSR_MCS4 BIT(16) | ||
402 | #define RRSR_MCS5 BIT(17) | ||
403 | #define RRSR_MCS6 BIT(18) | ||
404 | #define RRSR_MCS7 BIT(19) | ||
405 | #define BRSR_ACKSHORTPMB BIT(23) | ||
406 | |||
407 | #define RATR_1M 0x00000001 | ||
408 | #define RATR_2M 0x00000002 | ||
409 | #define RATR_55M 0x00000004 | ||
410 | #define RATR_11M 0x00000008 | ||
411 | #define RATR_6M 0x00000010 | ||
412 | #define RATR_9M 0x00000020 | ||
413 | #define RATR_12M 0x00000040 | ||
414 | #define RATR_18M 0x00000080 | ||
415 | #define RATR_24M 0x00000100 | ||
416 | #define RATR_36M 0x00000200 | ||
417 | #define RATR_48M 0x00000400 | ||
418 | #define RATR_54M 0x00000800 | ||
419 | #define RATR_MCS0 0x00001000 | ||
420 | #define RATR_MCS1 0x00002000 | ||
421 | #define RATR_MCS2 0x00004000 | ||
422 | #define RATR_MCS3 0x00008000 | ||
423 | #define RATR_MCS4 0x00010000 | ||
424 | #define RATR_MCS5 0x00020000 | ||
425 | #define RATR_MCS6 0x00040000 | ||
426 | #define RATR_MCS7 0x00080000 | ||
427 | #define RATR_MCS8 0x00100000 | ||
428 | #define RATR_MCS9 0x00200000 | ||
429 | #define RATR_MCS10 0x00400000 | ||
430 | #define RATR_MCS11 0x00800000 | ||
431 | #define RATR_MCS12 0x01000000 | ||
432 | #define RATR_MCS13 0x02000000 | ||
433 | #define RATR_MCS14 0x04000000 | ||
434 | #define RATR_MCS15 0x08000000 | ||
435 | |||
436 | #define RATE_1M BIT(0) | ||
437 | #define RATE_2M BIT(1) | ||
438 | #define RATE_5_5M BIT(2) | ||
439 | #define RATE_11M BIT(3) | ||
440 | #define RATE_6M BIT(4) | ||
441 | #define RATE_9M BIT(5) | ||
442 | #define RATE_12M BIT(6) | ||
443 | #define RATE_18M BIT(7) | ||
444 | #define RATE_24M BIT(8) | ||
445 | #define RATE_36M BIT(9) | ||
446 | #define RATE_48M BIT(10) | ||
447 | #define RATE_54M BIT(11) | ||
448 | #define RATE_MCS0 BIT(12) | ||
449 | #define RATE_MCS1 BIT(13) | ||
450 | #define RATE_MCS2 BIT(14) | ||
451 | #define RATE_MCS3 BIT(15) | ||
452 | #define RATE_MCS4 BIT(16) | ||
453 | #define RATE_MCS5 BIT(17) | ||
454 | #define RATE_MCS6 BIT(18) | ||
455 | #define RATE_MCS7 BIT(19) | ||
456 | #define RATE_MCS8 BIT(20) | ||
457 | #define RATE_MCS9 BIT(21) | ||
458 | #define RATE_MCS10 BIT(22) | ||
459 | #define RATE_MCS11 BIT(23) | ||
460 | #define RATE_MCS12 BIT(24) | ||
461 | #define RATE_MCS13 BIT(25) | ||
462 | #define RATE_MCS14 BIT(26) | ||
463 | #define RATE_MCS15 BIT(27) | ||
464 | |||
465 | #define RATE_ALL_CCK (RATR_1M | RATR_2M | RATR_55M | RATR_11M) | ||
466 | #define RATE_ALL_OFDM_AG (RATR_6M | RATR_9M | RATR_12M | RATR_18M \ | ||
467 | | RATR_24M | RATR_36M | RATR_48M | RATR_54M) | ||
468 | #define RATE_ALL_OFDM_1SS (RATR_MCS0 | RATR_MCS1 | RATR_MCS2 | \ | ||
469 | RATR_MCS3 | RATR_MCS4 | RATR_MCS5 | \ | ||
470 | RATR_MCS6 | RATR_MCS7) | ||
471 | #define RATE_ALL_OFDM_2SS (RATR_MCS8 | RATR_MCS9 | RATR_MCS10 | \ | ||
472 | RATR_MCS11 | RATR_MCS12 | RATR_MCS13 | \ | ||
473 | RATR_MCS14 | RATR_MCS15) | ||
474 | |||
475 | #define BW_OPMODE_20MHZ BIT(2) | ||
476 | #define BW_OPMODE_5G BIT(1) | ||
477 | #define BW_OPMODE_11J BIT(0) | ||
478 | |||
479 | #define CAM_VALID BIT(15) | ||
480 | #define CAM_NOTVALID 0x0000 | ||
481 | #define CAM_USEDK BIT(5) | ||
482 | |||
483 | #define CAM_NONE 0x0 | ||
484 | #define CAM_WEP40 0x01 | ||
485 | #define CAM_TKIP 0x02 | ||
486 | #define CAM_AES 0x04 | ||
487 | #define CAM_WEP104 0x05 | ||
488 | |||
489 | #define TOTAL_CAM_ENTRY 32 | ||
490 | #define HALF_CAM_ENTRY 16 | ||
491 | |||
492 | #define CAM_WRITE BIT(16) | ||
493 | #define CAM_READ 0x00000000 | ||
494 | #define CAM_POLLINIG BIT(31) | ||
495 | |||
496 | #define SCR_USEDK 0x01 | ||
497 | #define SCR_TXSEC_ENABLE 0x02 | ||
498 | #define SCR_RXSEC_ENABLE 0x04 | ||
499 | |||
500 | #define WOW_PMEN BIT(0) | ||
501 | #define WOW_WOMEN BIT(1) | ||
502 | #define WOW_MAGIC BIT(2) | ||
503 | #define WOW_UWF BIT(3) | ||
504 | |||
505 | #define IMR8190_DISABLED 0x0 | ||
506 | #define IMR_BCNDMAINT6 BIT(31) | ||
507 | #define IMR_BCNDMAINT5 BIT(30) | ||
508 | #define IMR_BCNDMAINT4 BIT(29) | ||
509 | #define IMR_BCNDMAINT3 BIT(28) | ||
510 | #define IMR_BCNDMAINT2 BIT(27) | ||
511 | #define IMR_BCNDMAINT1 BIT(26) | ||
512 | #define IMR_BCNDOK8 BIT(25) | ||
513 | #define IMR_BCNDOK7 BIT(24) | ||
514 | #define IMR_BCNDOK6 BIT(23) | ||
515 | #define IMR_BCNDOK5 BIT(22) | ||
516 | #define IMR_BCNDOK4 BIT(21) | ||
517 | #define IMR_BCNDOK3 BIT(20) | ||
518 | #define IMR_BCNDOK2 BIT(19) | ||
519 | #define IMR_BCNDOK1 BIT(18) | ||
520 | #define IMR_TIMEOUT2 BIT(17) | ||
521 | #define IMR_TIMEOUT1 BIT(16) | ||
522 | #define IMR_TXFOVW BIT(15) | ||
523 | #define IMR_PSTIMEOUT BIT(14) | ||
524 | #define IMR_BCNINT BIT(13) | ||
525 | #define IMR_RXFOVW BIT(12) | ||
526 | #define IMR_RDU BIT(11) | ||
527 | #define IMR_ATIMEND BIT(10) | ||
528 | #define IMR_BDOK BIT(9) | ||
529 | #define IMR_HIGHDOK BIT(8) | ||
530 | #define IMR_TBDOK BIT(7) | ||
531 | #define IMR_MGNTDOK BIT(6) | ||
532 | #define IMR_TBDER BIT(5) | ||
533 | #define IMR_BKDOK BIT(4) | ||
534 | #define IMR_BEDOK BIT(3) | ||
535 | #define IMR_VIDOK BIT(2) | ||
536 | #define IMR_VODOK BIT(1) | ||
537 | #define IMR_ROK BIT(0) | ||
538 | |||
539 | #define IMR_TXERR BIT(11) | ||
540 | #define IMR_RXERR BIT(10) | ||
541 | #define IMR_C2HCMD BIT(9) | ||
542 | #define IMR_CPWM BIT(8) | ||
543 | #define IMR_OCPINT BIT(1) | ||
544 | #define IMR_WLANOFF BIT(0) | ||
545 | |||
546 | #define EFUSE_REAL_CONTENT_LEN 512 | ||
547 | |||
548 | #define EEPROM_DEFAULT_TSSI 0x0 | ||
549 | #define EEPROM_DEFAULT_TXPOWERDIFF 0x0 | ||
550 | #define EEPROM_DEFAULT_CRYSTALCAP 0x5 | ||
551 | #define EEPROM_DEFAULT_BOARDTYPE 0x02 | ||
552 | #define EEPROM_DEFAULT_TXPOWER 0x1010 | ||
553 | #define EEPROM_DEFAULT_HT2T_TXPWR 0x10 | ||
554 | |||
555 | #define EEPROM_DEFAULT_LEGACYHTTXPOWERDIFF 0x3 | ||
556 | #define EEPROM_DEFAULT_THERMALMETER 0x12 | ||
557 | #define EEPROM_DEFAULT_ANTTXPOWERDIFF 0x0 | ||
558 | #define EEPROM_DEFAULT_TXPWDIFF_CRYSTALCAP 0x5 | ||
559 | #define EEPROM_DEFAULT_TXPOWERLEVEL 0x22 | ||
560 | #define EEPROM_DEFAULT_HT40_2SDIFF 0x0 | ||
561 | #define EEPROM_DEFAULT_HT20_DIFF 2 | ||
562 | #define EEPROM_DEFAULT_LEGACYHTTXPOWERDIFF 0x3 | ||
563 | #define EEPROM_DEFAULT_HT40_PWRMAXOFFSET 0 | ||
564 | #define EEPROM_DEFAULT_HT20_PWRMAXOFFSET 0 | ||
565 | |||
566 | #define RF_OPTION1 0x79 | ||
567 | #define RF_OPTION2 0x7A | ||
568 | #define RF_OPTION3 0x7B | ||
569 | #define RF_OPTION4 0x7C | ||
570 | |||
571 | #define EEPROM_DEFAULT_PID 0x1234 | ||
572 | #define EEPROM_DEFAULT_VID 0x5678 | ||
573 | #define EEPROM_DEFAULT_CUSTOMERID 0xAB | ||
574 | #define EEPROM_DEFAULT_SUBCUSTOMERID 0xCD | ||
575 | #define EEPROM_DEFAULT_VERSION 0 | ||
576 | |||
577 | #define EEPROM_CHANNEL_PLAN_FCC 0x0 | ||
578 | #define EEPROM_CHANNEL_PLAN_IC 0x1 | ||
579 | #define EEPROM_CHANNEL_PLAN_ETSI 0x2 | ||
580 | #define EEPROM_CHANNEL_PLAN_SPAIN 0x3 | ||
581 | #define EEPROM_CHANNEL_PLAN_FRANCE 0x4 | ||
582 | #define EEPROM_CHANNEL_PLAN_MKK 0x5 | ||
583 | #define EEPROM_CHANNEL_PLAN_MKK1 0x6 | ||
584 | #define EEPROM_CHANNEL_PLAN_ISRAEL 0x7 | ||
585 | #define EEPROM_CHANNEL_PLAN_TELEC 0x8 | ||
586 | #define EEPROM_CHANNEL_PLAN_GLOBAL_DOMAIN 0x9 | ||
587 | #define EEPROM_CHANNEL_PLAN_WORLD_WIDE_13 0xA | ||
588 | #define EEPROM_CHANNEL_PLAN_NCC 0xB | ||
589 | #define EEPROM_CHANNEL_PLAN_BY_HW_MASK 0x80 | ||
590 | |||
591 | #define EEPROM_CID_DEFAULT 0x0 | ||
592 | #define EEPROM_CID_TOSHIBA 0x4 | ||
593 | #define EEPROM_CID_CCX 0x10 | ||
594 | #define EEPROM_CID_QMI 0x0D | ||
595 | #define EEPROM_CID_WHQL 0xFE | ||
596 | |||
597 | #define RTL8192_EEPROM_ID 0x8129 | ||
598 | |||
599 | #define RTL8190_EEPROM_ID 0x8129 | ||
600 | #define EEPROM_HPON 0x02 | ||
601 | #define EEPROM_CLK 0x06 | ||
602 | #define EEPROM_TESTR 0x08 | ||
603 | |||
604 | #define EEPROM_VID 0x0A | ||
605 | #define EEPROM_DID 0x0C | ||
606 | #define EEPROM_SVID 0x0E | ||
607 | #define EEPROM_SMID 0x10 | ||
608 | |||
609 | #define EEPROM_MAC_ADDR 0x16 | ||
610 | |||
611 | #define EEPROM_CCK_TX_PWR_INX 0x5A | ||
612 | #define EEPROM_HT40_1S_TX_PWR_INX 0x60 | ||
613 | #define EEPROM_HT40_2S_TX_PWR_INX_DIFF 0x66 | ||
614 | #define EEPROM_HT20_TX_PWR_INX_DIFF 0x69 | ||
615 | #define EEPROM_OFDM_TX_PWR_INX_DIFF 0x6C | ||
616 | #define EEPROM_HT40_MAX_PWR_OFFSET 0x6F | ||
617 | #define EEPROM_HT20_MAX_PWR_OFFSET 0x72 | ||
618 | |||
619 | #define EEPROM_TSSI_A 0x76 | ||
620 | #define EEPROM_TSSI_B 0x77 | ||
621 | #define EEPROM_THERMAL_METER 0x78 | ||
622 | #define EEPROM_XTAL_K 0x78 | ||
623 | #define EEPROM_RF_OPT1 0x79 | ||
624 | #define EEPROM_RF_OPT2 0x7A | ||
625 | #define EEPROM_RF_OPT3 0x7B | ||
626 | #define EEPROM_RF_OPT4 0x7C | ||
627 | #define EEPROM_CHANNEL_PLAN 0x7D | ||
628 | #define EEPROM_VERSION 0x7E | ||
629 | #define EEPROM_CUSTOMER_ID 0x7F | ||
630 | |||
631 | #define EEPROM_PWRDIFF 0x54 | ||
632 | |||
633 | #define EEPROM_TXPOWERCCK 0x5A | ||
634 | #define EEPROM_TXPOWERHT40_1S 0x60 | ||
635 | #define EEPROM_TXPOWERHT40_2SDIFF 0x66 | ||
636 | #define EEPROM_TXPOWERHT20DIFF 0x69 | ||
637 | #define EEPROM_TXPOWER_OFDMDIFF 0x6C | ||
638 | |||
639 | #define EEPROM_TXPWR_GROUP 0x6F | ||
640 | |||
641 | #define EEPROM_TSSI_A 0x76 | ||
642 | #define EEPROM_TSSI_B 0x77 | ||
643 | #define EEPROM_THERMAL_METER 0x78 | ||
644 | |||
645 | #define EEPROM_CHANNELPLAN 0x75 | ||
646 | |||
647 | #define RF_OPTION1 0x79 | ||
648 | #define RF_OPTION2 0x7A | ||
649 | #define RF_OPTION3 0x7B | ||
650 | #define RF_OPTION4 0x7C | ||
651 | |||
652 | #define STOPBECON BIT(6) | ||
653 | #define STOPHIGHT BIT(5) | ||
654 | #define STOPMGT BIT(4) | ||
655 | #define STOPVO BIT(3) | ||
656 | #define STOPVI BIT(2) | ||
657 | #define STOPBE BIT(1) | ||
658 | #define STOPBK BIT(0) | ||
659 | |||
660 | #define RCR_APPFCS BIT(31) | ||
661 | #define RCR_APP_FCS BIT(31) | ||
662 | #define RCR_APP_MIC BIT(30) | ||
663 | #define RCR_APP_ICV BIT(29) | ||
664 | #define RCR_APP_PHYSTS BIT(28) | ||
665 | #define RCR_APP_PHYST_RXFF BIT(28) | ||
666 | #define RCR_APP_BA_SSN BIT(27) | ||
667 | #define RCR_ENMBID BIT(24) | ||
668 | #define RCR_LSIGEN BIT(23) | ||
669 | #define RCR_MFBEN BIT(22) | ||
670 | #define RCR_HTC_LOC_CTRL BIT(14) | ||
671 | #define RCR_AMF BIT(13) | ||
672 | #define RCR_ACF BIT(12) | ||
673 | #define RCR_ADF BIT(11) | ||
674 | #define RCR_AICV BIT(9) | ||
675 | #define RCR_ACRC32 BIT(8) | ||
676 | #define RCR_CBSSID_BCN BIT(7) | ||
677 | #define RCR_CBSSID_DATA BIT(6) | ||
678 | #define RCR_CBSSID RCR_CBSSID_DATA | ||
679 | #define RCR_APWRMGT BIT(5) | ||
680 | #define RCR_ADD3 BIT(4) | ||
681 | #define RCR_AB BIT(3) | ||
682 | #define RCR_AM BIT(2) | ||
683 | #define RCR_APM BIT(1) | ||
684 | #define RCR_AAP BIT(0) | ||
685 | #define RCR_MXDMA_OFFSET 8 | ||
686 | #define RCR_FIFO_OFFSET 13 | ||
687 | |||
688 | #define RSV_CTRL 0x001C | ||
689 | #define RD_CTRL 0x0524 | ||
690 | |||
691 | #define REG_USB_INFO 0xFE17 | ||
692 | #define REG_USB_SPECIAL_OPTION 0xFE55 | ||
693 | |||
694 | #define REG_USB_DMA_AGG_TO 0xFE5B | ||
695 | #define REG_USB_AGG_TO 0xFE5C | ||
696 | #define REG_USB_AGG_TH 0xFE5D | ||
697 | |||
698 | #define REG_USB_VID 0xFE60 | ||
699 | #define REG_USB_PID 0xFE62 | ||
700 | #define REG_USB_OPTIONAL 0xFE64 | ||
701 | #define REG_USB_CHIRP_K 0xFE65 | ||
702 | #define REG_USB_PHY 0xFE66 | ||
703 | #define REG_USB_MAC_ADDR 0xFE70 | ||
704 | #define REG_USB_HRPWM 0xFE58 | ||
705 | #define REG_USB_HCPWM 0xFE57 | ||
706 | |||
707 | #define SW18_FPWM BIT(3) | ||
708 | |||
709 | #define ISO_MD2PP BIT(0) | ||
710 | #define ISO_UA2USB BIT(1) | ||
711 | #define ISO_UD2CORE BIT(2) | ||
712 | #define ISO_PA2PCIE BIT(3) | ||
713 | #define ISO_PD2CORE BIT(4) | ||
714 | #define ISO_IP2MAC BIT(5) | ||
715 | #define ISO_DIOP BIT(6) | ||
716 | #define ISO_DIOE BIT(7) | ||
717 | #define ISO_EB2CORE BIT(8) | ||
718 | #define ISO_DIOR BIT(9) | ||
719 | |||
720 | #define PWC_EV25V BIT(14) | ||
721 | #define PWC_EV12V BIT(15) | ||
722 | |||
723 | #define FEN_BBRSTB BIT(0) | ||
724 | #define FEN_BB_GLB_RSTn BIT(1) | ||
725 | #define FEN_USBA BIT(2) | ||
726 | #define FEN_UPLL BIT(3) | ||
727 | #define FEN_USBD BIT(4) | ||
728 | #define FEN_DIO_PCIE BIT(5) | ||
729 | #define FEN_PCIEA BIT(6) | ||
730 | #define FEN_PPLL BIT(7) | ||
731 | #define FEN_PCIED BIT(8) | ||
732 | #define FEN_DIOE BIT(9) | ||
733 | #define FEN_CPUEN BIT(10) | ||
734 | #define FEN_DCORE BIT(11) | ||
735 | #define FEN_ELDR BIT(12) | ||
736 | #define FEN_DIO_RF BIT(13) | ||
737 | #define FEN_HWPDN BIT(14) | ||
738 | #define FEN_MREGEN BIT(15) | ||
739 | |||
740 | #define PFM_LDALL BIT(0) | ||
741 | #define PFM_ALDN BIT(1) | ||
742 | #define PFM_LDKP BIT(2) | ||
743 | #define PFM_WOWL BIT(3) | ||
744 | #define EnPDN BIT(4) | ||
745 | #define PDN_PL BIT(5) | ||
746 | #define APFM_ONMAC BIT(8) | ||
747 | #define APFM_OFF BIT(9) | ||
748 | #define APFM_RSM BIT(10) | ||
749 | #define AFSM_HSUS BIT(11) | ||
750 | #define AFSM_PCIE BIT(12) | ||
751 | #define APDM_MAC BIT(13) | ||
752 | #define APDM_HOST BIT(14) | ||
753 | #define APDM_HPDN BIT(15) | ||
754 | #define RDY_MACON BIT(16) | ||
755 | #define SUS_HOST BIT(17) | ||
756 | #define ROP_ALD BIT(20) | ||
757 | #define ROP_PWR BIT(21) | ||
758 | #define ROP_SPS BIT(22) | ||
759 | #define SOP_MRST BIT(25) | ||
760 | #define SOP_FUSE BIT(26) | ||
761 | #define SOP_ABG BIT(27) | ||
762 | #define SOP_AMB BIT(28) | ||
763 | #define SOP_RCK BIT(29) | ||
764 | #define SOP_A8M BIT(30) | ||
765 | #define XOP_BTCK BIT(31) | ||
766 | |||
767 | #define ANAD16V_EN BIT(0) | ||
768 | #define ANA8M BIT(1) | ||
769 | #define MACSLP BIT(4) | ||
770 | #define LOADER_CLK_EN BIT(5) | ||
771 | #define _80M_SSC_DIS BIT(7) | ||
772 | #define _80M_SSC_EN_HO BIT(8) | ||
773 | #define PHY_SSC_RSTB BIT(9) | ||
774 | #define SEC_CLK_EN BIT(10) | ||
775 | #define MAC_CLK_EN BIT(11) | ||
776 | #define SYS_CLK_EN BIT(12) | ||
777 | #define RING_CLK_EN BIT(13) | ||
778 | |||
779 | #define BOOT_FROM_EEPROM BIT(4) | ||
780 | #define EEPROM_EN BIT(5) | ||
781 | |||
782 | #define AFE_BGEN BIT(0) | ||
783 | #define AFE_MBEN BIT(1) | ||
784 | #define MAC_ID_EN BIT(7) | ||
785 | |||
786 | #define WLOCK_ALL BIT(0) | ||
787 | #define WLOCK_00 BIT(1) | ||
788 | #define WLOCK_04 BIT(2) | ||
789 | #define WLOCK_08 BIT(3) | ||
790 | #define WLOCK_40 BIT(4) | ||
791 | #define R_DIS_PRST_0 BIT(5) | ||
792 | #define R_DIS_PRST_1 BIT(6) | ||
793 | #define LOCK_ALL_EN BIT(7) | ||
794 | |||
795 | #define RF_EN BIT(0) | ||
796 | #define RF_RSTB BIT(1) | ||
797 | #define RF_SDMRSTB BIT(2) | ||
798 | |||
799 | #define LDA15_EN BIT(0) | ||
800 | #define LDA15_STBY BIT(1) | ||
801 | #define LDA15_OBUF BIT(2) | ||
802 | #define LDA15_REG_VOS BIT(3) | ||
803 | #define _LDA15_VOADJ(x) (((x) & 0x7) << 4) | ||
804 | |||
805 | #define LDV12_EN BIT(0) | ||
806 | #define LDV12_SDBY BIT(1) | ||
807 | #define LPLDO_HSM BIT(2) | ||
808 | #define LPLDO_LSM_DIS BIT(3) | ||
809 | #define _LDV12_VADJ(x) (((x) & 0xF) << 4) | ||
810 | |||
811 | #define XTAL_EN BIT(0) | ||
812 | #define XTAL_BSEL BIT(1) | ||
813 | #define _XTAL_BOSC(x) (((x) & 0x3) << 2) | ||
814 | #define _XTAL_CADJ(x) (((x) & 0xF) << 4) | ||
815 | #define XTAL_GATE_USB BIT(8) | ||
816 | #define _XTAL_USB_DRV(x) (((x) & 0x3) << 9) | ||
817 | #define XTAL_GATE_AFE BIT(11) | ||
818 | #define _XTAL_AFE_DRV(x) (((x) & 0x3) << 12) | ||
819 | #define XTAL_RF_GATE BIT(14) | ||
820 | #define _XTAL_RF_DRV(x) (((x) & 0x3) << 15) | ||
821 | #define XTAL_GATE_DIG BIT(17) | ||
822 | #define _XTAL_DIG_DRV(x) (((x) & 0x3) << 18) | ||
823 | #define XTAL_BT_GATE BIT(20) | ||
824 | #define _XTAL_BT_DRV(x) (((x) & 0x3) << 21) | ||
825 | #define _XTAL_GPIO(x) (((x) & 0x7) << 23) | ||
826 | |||
827 | #define CKDLY_AFE BIT(26) | ||
828 | #define CKDLY_USB BIT(27) | ||
829 | #define CKDLY_DIG BIT(28) | ||
830 | #define CKDLY_BT BIT(29) | ||
831 | |||
832 | #define APLL_EN BIT(0) | ||
833 | #define APLL_320_EN BIT(1) | ||
834 | #define APLL_FREF_SEL BIT(2) | ||
835 | #define APLL_EDGE_SEL BIT(3) | ||
836 | #define APLL_WDOGB BIT(4) | ||
837 | #define APLL_LPFEN BIT(5) | ||
838 | |||
839 | #define APLL_REF_CLK_13MHZ 0x1 | ||
840 | #define APLL_REF_CLK_19_2MHZ 0x2 | ||
841 | #define APLL_REF_CLK_20MHZ 0x3 | ||
842 | #define APLL_REF_CLK_25MHZ 0x4 | ||
843 | #define APLL_REF_CLK_26MHZ 0x5 | ||
844 | #define APLL_REF_CLK_38_4MHZ 0x6 | ||
845 | #define APLL_REF_CLK_40MHZ 0x7 | ||
846 | |||
847 | #define APLL_320EN BIT(14) | ||
848 | #define APLL_80EN BIT(15) | ||
849 | #define APLL_1MEN BIT(24) | ||
850 | |||
851 | #define ALD_EN BIT(18) | ||
852 | #define EF_PD BIT(19) | ||
853 | #define EF_FLAG BIT(31) | ||
854 | |||
855 | #define EF_TRPT BIT(7) | ||
856 | #define LDOE25_EN BIT(31) | ||
857 | |||
858 | #define RSM_EN BIT(0) | ||
859 | #define Timer_EN BIT(4) | ||
860 | |||
861 | #define TRSW0EN BIT(2) | ||
862 | #define TRSW1EN BIT(3) | ||
863 | #define EROM_EN BIT(4) | ||
864 | #define EnBT BIT(5) | ||
865 | #define EnUart BIT(8) | ||
866 | #define Uart_910 BIT(9) | ||
867 | #define EnPMAC BIT(10) | ||
868 | #define SIC_SWRST BIT(11) | ||
869 | #define EnSIC BIT(12) | ||
870 | #define SIC_23 BIT(13) | ||
871 | #define EnHDP BIT(14) | ||
872 | #define SIC_LBK BIT(15) | ||
873 | |||
874 | #define LED0PL BIT(4) | ||
875 | #define LED1PL BIT(12) | ||
876 | #define LED0DIS BIT(7) | ||
877 | |||
878 | #define MCUFWDL_EN BIT(0) | ||
879 | #define MCUFWDL_RDY BIT(1) | ||
880 | #define FWDL_ChkSum_rpt BIT(2) | ||
881 | #define MACINI_RDY BIT(3) | ||
882 | #define BBINI_RDY BIT(4) | ||
883 | #define RFINI_RDY BIT(5) | ||
884 | #define WINTINI_RDY BIT(6) | ||
885 | #define CPRST BIT(23) | ||
886 | |||
887 | #define XCLK_VLD BIT(0) | ||
888 | #define ACLK_VLD BIT(1) | ||
889 | #define UCLK_VLD BIT(2) | ||
890 | #define PCLK_VLD BIT(3) | ||
891 | #define PCIRSTB BIT(4) | ||
892 | #define V15_VLD BIT(5) | ||
893 | #define TRP_B15V_EN BIT(7) | ||
894 | #define SIC_IDLE BIT(8) | ||
895 | #define BD_MAC2 BIT(9) | ||
896 | #define BD_MAC1 BIT(10) | ||
897 | #define IC_MACPHY_MODE BIT(11) | ||
898 | #define BT_FUNC BIT(16) | ||
899 | #define VENDOR_ID BIT(19) | ||
900 | #define PAD_HWPD_IDN BIT(22) | ||
901 | #define TRP_VAUX_EN BIT(23) | ||
902 | #define TRP_BT_EN BIT(24) | ||
903 | #define BD_PKG_SEL BIT(25) | ||
904 | #define BD_HCI_SEL BIT(26) | ||
905 | #define TYPE_ID BIT(27) | ||
906 | #define RF_RL_ID (BIT(31) | BIT(30) | BIT(29) | BIT(28)) | ||
907 | |||
908 | #define CHIP_VER_RTL_MASK 0xF000 | ||
909 | #define CHIP_VER_RTL_SHIFT 12 | ||
910 | |||
911 | #define REG_LBMODE (REG_CR + 3) | ||
912 | |||
913 | #define HCI_TXDMA_EN BIT(0) | ||
914 | #define HCI_RXDMA_EN BIT(1) | ||
915 | #define TXDMA_EN BIT(2) | ||
916 | #define RXDMA_EN BIT(3) | ||
917 | #define PROTOCOL_EN BIT(4) | ||
918 | #define SCHEDULE_EN BIT(5) | ||
919 | #define MACTXEN BIT(6) | ||
920 | #define MACRXEN BIT(7) | ||
921 | #define ENSWBCN BIT(8) | ||
922 | #define ENSEC BIT(9) | ||
923 | |||
924 | #define _NETTYPE(x) (((x) & 0x3) << 16) | ||
925 | #define MASK_NETTYPE 0x30000 | ||
926 | #define NT_NO_LINK 0x0 | ||
927 | #define NT_LINK_AD_HOC 0x1 | ||
928 | #define NT_LINK_AP 0x2 | ||
929 | #define NT_AS_AP 0x3 | ||
930 | |||
931 | #define _LBMODE(x) (((x) & 0xF) << 24) | ||
932 | #define MASK_LBMODE 0xF000000 | ||
933 | #define LOOPBACK_NORMAL 0x0 | ||
934 | #define LOOPBACK_IMMEDIATELY 0xB | ||
935 | #define LOOPBACK_MAC_DELAY 0x3 | ||
936 | #define LOOPBACK_PHY 0x1 | ||
937 | #define LOOPBACK_DMA 0x7 | ||
938 | |||
939 | #define GET_RX_PAGE_SIZE(value) ((value) & 0xF) | ||
940 | #define GET_TX_PAGE_SIZE(value) (((value) & 0xF0) >> 4) | ||
941 | #define _PSRX_MASK 0xF | ||
942 | #define _PSTX_MASK 0xF0 | ||
943 | #define _PSRX(x) (x) | ||
944 | #define _PSTX(x) ((x) << 4) | ||
945 | |||
946 | #define PBP_64 0x0 | ||
947 | #define PBP_128 0x1 | ||
948 | #define PBP_256 0x2 | ||
949 | #define PBP_512 0x3 | ||
950 | #define PBP_1024 0x4 | ||
951 | |||
952 | #define RXDMA_ARBBW_EN BIT(0) | ||
953 | #define RXSHFT_EN BIT(1) | ||
954 | #define RXDMA_AGG_EN BIT(2) | ||
955 | #define QS_VO_QUEUE BIT(8) | ||
956 | #define QS_VI_QUEUE BIT(9) | ||
957 | #define QS_BE_QUEUE BIT(10) | ||
958 | #define QS_BK_QUEUE BIT(11) | ||
959 | #define QS_MANAGER_QUEUE BIT(12) | ||
960 | #define QS_HIGH_QUEUE BIT(13) | ||
961 | |||
962 | #define HQSEL_VOQ BIT(0) | ||
963 | #define HQSEL_VIQ BIT(1) | ||
964 | #define HQSEL_BEQ BIT(2) | ||
965 | #define HQSEL_BKQ BIT(3) | ||
966 | #define HQSEL_MGTQ BIT(4) | ||
967 | #define HQSEL_HIQ BIT(5) | ||
968 | |||
969 | #define _TXDMA_HIQ_MAP(x) (((x)&0x3) << 14) | ||
970 | #define _TXDMA_MGQ_MAP(x) (((x)&0x3) << 12) | ||
971 | #define _TXDMA_BKQ_MAP(x) (((x)&0x3) << 10) | ||
972 | #define _TXDMA_BEQ_MAP(x) (((x)&0x3) << 8) | ||
973 | #define _TXDMA_VIQ_MAP(x) (((x)&0x3) << 6) | ||
974 | #define _TXDMA_VOQ_MAP(x) (((x)&0x3) << 4) | ||
975 | |||
976 | #define QUEUE_LOW 1 | ||
977 | #define QUEUE_NORMAL 2 | ||
978 | #define QUEUE_HIGH 3 | ||
979 | |||
980 | #define _LLT_NO_ACTIVE 0x0 | ||
981 | #define _LLT_WRITE_ACCESS 0x1 | ||
982 | #define _LLT_READ_ACCESS 0x2 | ||
983 | |||
984 | #define _LLT_INIT_DATA(x) ((x) & 0xFF) | ||
985 | #define _LLT_INIT_ADDR(x) (((x) & 0xFF) << 8) | ||
986 | #define _LLT_OP(x) (((x) & 0x3) << 30) | ||
987 | #define _LLT_OP_VALUE(x) (((x) >> 30) & 0x3) | ||
988 | |||
989 | #define BB_WRITE_READ_MASK (BIT(31) | BIT(30)) | ||
990 | #define BB_WRITE_EN BIT(30) | ||
991 | #define BB_READ_EN BIT(31) | ||
992 | |||
993 | #define _HPQ(x) ((x) & 0xFF) | ||
994 | #define _LPQ(x) (((x) & 0xFF) << 8) | ||
995 | #define _PUBQ(x) (((x) & 0xFF) << 16) | ||
996 | #define _NPQ(x) ((x) & 0xFF) | ||
997 | |||
998 | #define HPQ_PUBLIC_DIS BIT(24) | ||
999 | #define LPQ_PUBLIC_DIS BIT(25) | ||
1000 | #define LD_RQPN BIT(31) | ||
1001 | |||
1002 | #define BCN_VALID BIT(16) | ||
1003 | #define BCN_HEAD(x) (((x) & 0xFF) << 8) | ||
1004 | #define BCN_HEAD_MASK 0xFF00 | ||
1005 | |||
1006 | #define BLK_DESC_NUM_SHIFT 4 | ||
1007 | #define BLK_DESC_NUM_MASK 0xF | ||
1008 | |||
1009 | #define DROP_DATA_EN BIT(9) | ||
1010 | |||
1011 | #define EN_AMPDU_RTY_NEW BIT(7) | ||
1012 | |||
1013 | #define _INIRTSMCS_SEL(x) ((x) & 0x3F) | ||
1014 | |||
1015 | #define _SPEC_SIFS_CCK(x) ((x) & 0xFF) | ||
1016 | #define _SPEC_SIFS_OFDM(x) (((x) & 0xFF) << 8) | ||
1017 | |||
1018 | #define RATE_REG_BITMAP_ALL 0xFFFFF | ||
1019 | |||
1020 | #define _RRSC_BITMAP(x) ((x) & 0xFFFFF) | ||
1021 | |||
1022 | #define _RRSR_RSC(x) (((x) & 0x3) << 21) | ||
1023 | #define RRSR_RSC_RESERVED 0x0 | ||
1024 | #define RRSR_RSC_UPPER_SUBCHANNEL 0x1 | ||
1025 | #define RRSR_RSC_LOWER_SUBCHANNEL 0x2 | ||
1026 | #define RRSR_RSC_DUPLICATE_MODE 0x3 | ||
1027 | |||
1028 | #define USE_SHORT_G1 BIT(20) | ||
1029 | |||
1030 | #define _AGGLMT_MCS0(x) ((x) & 0xF) | ||
1031 | #define _AGGLMT_MCS1(x) (((x) & 0xF) << 4) | ||
1032 | #define _AGGLMT_MCS2(x) (((x) & 0xF) << 8) | ||
1033 | #define _AGGLMT_MCS3(x) (((x) & 0xF) << 12) | ||
1034 | #define _AGGLMT_MCS4(x) (((x) & 0xF) << 16) | ||
1035 | #define _AGGLMT_MCS5(x) (((x) & 0xF) << 20) | ||
1036 | #define _AGGLMT_MCS6(x) (((x) & 0xF) << 24) | ||
1037 | #define _AGGLMT_MCS7(x) (((x) & 0xF) << 28) | ||
1038 | |||
1039 | #define RETRY_LIMIT_SHORT_SHIFT 8 | ||
1040 | #define RETRY_LIMIT_LONG_SHIFT 0 | ||
1041 | |||
1042 | #define _DARF_RC1(x) ((x) & 0x1F) | ||
1043 | #define _DARF_RC2(x) (((x) & 0x1F) << 8) | ||
1044 | #define _DARF_RC3(x) (((x) & 0x1F) << 16) | ||
1045 | #define _DARF_RC4(x) (((x) & 0x1F) << 24) | ||
1046 | #define _DARF_RC5(x) ((x) & 0x1F) | ||
1047 | #define _DARF_RC6(x) (((x) & 0x1F) << 8) | ||
1048 | #define _DARF_RC7(x) (((x) & 0x1F) << 16) | ||
1049 | #define _DARF_RC8(x) (((x) & 0x1F) << 24) | ||
1050 | |||
1051 | #define _RARF_RC1(x) ((x) & 0x1F) | ||
1052 | #define _RARF_RC2(x) (((x) & 0x1F) << 8) | ||
1053 | #define _RARF_RC3(x) (((x) & 0x1F) << 16) | ||
1054 | #define _RARF_RC4(x) (((x) & 0x1F) << 24) | ||
1055 | #define _RARF_RC5(x) ((x) & 0x1F) | ||
1056 | #define _RARF_RC6(x) (((x) & 0x1F) << 8) | ||
1057 | #define _RARF_RC7(x) (((x) & 0x1F) << 16) | ||
1058 | #define _RARF_RC8(x) (((x) & 0x1F) << 24) | ||
1059 | |||
1060 | #define AC_PARAM_TXOP_OFFSET 16 | ||
1061 | #define AC_PARAM_TXOP_LIMIT_OFFSET 16 | ||
1062 | #define AC_PARAM_ECW_MAX_OFFSET 12 | ||
1063 | #define AC_PARAM_ECW_MIN_OFFSET 8 | ||
1064 | #define AC_PARAM_AIFS_OFFSET 0 | ||
1065 | |||
1066 | #define _AIFS(x) (x) | ||
1067 | #define _ECW_MAX_MIN(x) ((x) << 8) | ||
1068 | #define _TXOP_LIMIT(x) ((x) << 16) | ||
1069 | |||
1070 | #define _BCNIFS(x) ((x) & 0xFF) | ||
1071 | #define _BCNECW(x) ((((x) & 0xF)) << 8) | ||
1072 | |||
1073 | #define _LRL(x) ((x) & 0x3F) | ||
1074 | #define _SRL(x) (((x) & 0x3F) << 8) | ||
1075 | |||
1076 | #define _SIFS_CCK_CTX(x) ((x) & 0xFF) | ||
1077 | #define _SIFS_CCK_TRX(x) (((x) & 0xFF) << 8); | ||
1078 | |||
1079 | #define _SIFS_OFDM_CTX(x) ((x) & 0xFF) | ||
1080 | #define _SIFS_OFDM_TRX(x) (((x) & 0xFF) << 8); | ||
1081 | |||
1082 | #define _TBTT_PROHIBIT_HOLD(x) (((x) & 0xFF) << 8) | ||
1083 | |||
1084 | #define DIS_EDCA_CNT_DWN BIT(11) | ||
1085 | |||
1086 | #define EN_MBSSID BIT(1) | ||
1087 | #define EN_TXBCN_RPT BIT(2) | ||
1088 | #define EN_BCN_FUNCTION BIT(3) | ||
1089 | |||
1090 | #define TSFTR_RST BIT(0) | ||
1091 | #define TSFTR1_RST BIT(1) | ||
1092 | |||
1093 | #define STOP_BCNQ BIT(6) | ||
1094 | |||
1095 | #define DIS_TSF_UDT0_NORMAL_CHIP BIT(4) | ||
1096 | #define DIS_TSF_UDT0_TEST_CHIP BIT(5) | ||
1097 | |||
1098 | #define AcmHw_HwEn BIT(0) | ||
1099 | #define AcmHw_BeqEn BIT(1) | ||
1100 | #define AcmHw_ViqEn BIT(2) | ||
1101 | #define AcmHw_VoqEn BIT(3) | ||
1102 | #define AcmHw_BeqStatus BIT(4) | ||
1103 | #define AcmHw_ViqStatus BIT(5) | ||
1104 | #define AcmHw_VoqStatus BIT(6) | ||
1105 | |||
1106 | #define APSDOFF BIT(6) | ||
1107 | #define APSDOFF_STATUS BIT(7) | ||
1108 | |||
1109 | #define BW_20MHZ BIT(2) | ||
1110 | |||
1111 | #define RATE_BITMAP_ALL 0xFFFFF | ||
1112 | |||
1113 | #define RATE_RRSR_CCK_ONLY_1M 0xFFFF1 | ||
1114 | |||
1115 | #define TSFRST BIT(0) | ||
1116 | #define DIS_GCLK BIT(1) | ||
1117 | #define PAD_SEL BIT(2) | ||
1118 | #define PWR_ST BIT(6) | ||
1119 | #define PWRBIT_OW_EN BIT(7) | ||
1120 | #define ACRC BIT(8) | ||
1121 | #define CFENDFORM BIT(9) | ||
1122 | #define ICV BIT(10) | ||
1123 | |||
1124 | #define AAP BIT(0) | ||
1125 | #define APM BIT(1) | ||
1126 | #define AM BIT(2) | ||
1127 | #define AB BIT(3) | ||
1128 | #define ADD3 BIT(4) | ||
1129 | #define APWRMGT BIT(5) | ||
1130 | #define CBSSID BIT(6) | ||
1131 | #define CBSSID_DATA BIT(6) | ||
1132 | #define CBSSID_BCN BIT(7) | ||
1133 | #define ACRC32 BIT(8) | ||
1134 | #define AICV BIT(9) | ||
1135 | #define ADF BIT(11) | ||
1136 | #define ACF BIT(12) | ||
1137 | #define AMF BIT(13) | ||
1138 | #define HTC_LOC_CTRL BIT(14) | ||
1139 | #define UC_DATA_EN BIT(16) | ||
1140 | #define BM_DATA_EN BIT(17) | ||
1141 | #define MFBEN BIT(22) | ||
1142 | #define LSIGEN BIT(23) | ||
1143 | #define EnMBID BIT(24) | ||
1144 | #define APP_BASSN BIT(27) | ||
1145 | #define APP_PHYSTS BIT(28) | ||
1146 | #define APP_ICV BIT(29) | ||
1147 | #define APP_MIC BIT(30) | ||
1148 | #define APP_FCS BIT(31) | ||
1149 | |||
1150 | #define _MIN_SPACE(x) ((x) & 0x7) | ||
1151 | #define _SHORT_GI_PADDING(x) (((x) & 0x1F) << 3) | ||
1152 | |||
1153 | #define RXERR_TYPE_OFDM_PPDU 0 | ||
1154 | #define RXERR_TYPE_OFDM_FALSE_ALARM 1 | ||
1155 | #define RXERR_TYPE_OFDM_MPDU_OK 2 | ||
1156 | #define RXERR_TYPE_OFDM_MPDU_FAIL 3 | ||
1157 | #define RXERR_TYPE_CCK_PPDU 4 | ||
1158 | #define RXERR_TYPE_CCK_FALSE_ALARM 5 | ||
1159 | #define RXERR_TYPE_CCK_MPDU_OK 6 | ||
1160 | #define RXERR_TYPE_CCK_MPDU_FAIL 7 | ||
1161 | #define RXERR_TYPE_HT_PPDU 8 | ||
1162 | #define RXERR_TYPE_HT_FALSE_ALARM 9 | ||
1163 | #define RXERR_TYPE_HT_MPDU_TOTAL 10 | ||
1164 | #define RXERR_TYPE_HT_MPDU_OK 11 | ||
1165 | #define RXERR_TYPE_HT_MPDU_FAIL 12 | ||
1166 | #define RXERR_TYPE_RX_FULL_DROP 15 | ||
1167 | |||
1168 | #define RXERR_COUNTER_MASK 0xFFFFF | ||
1169 | #define RXERR_RPT_RST BIT(27) | ||
1170 | #define _RXERR_RPT_SEL(type) ((type) << 28) | ||
1171 | |||
1172 | #define SCR_TxUseDK BIT(0) | ||
1173 | #define SCR_RxUseDK BIT(1) | ||
1174 | #define SCR_TxEncEnable BIT(2) | ||
1175 | #define SCR_RxDecEnable BIT(3) | ||
1176 | #define SCR_SKByA2 BIT(4) | ||
1177 | #define SCR_NoSKMC BIT(5) | ||
1178 | #define SCR_TXBCUSEDK BIT(6) | ||
1179 | #define SCR_RXBCUSEDK BIT(7) | ||
1180 | |||
1181 | #define USB_IS_HIGH_SPEED 0 | ||
1182 | #define USB_IS_FULL_SPEED 1 | ||
1183 | #define USB_SPEED_MASK BIT(5) | ||
1184 | |||
1185 | #define USB_NORMAL_SIE_EP_MASK 0xF | ||
1186 | #define USB_NORMAL_SIE_EP_SHIFT 4 | ||
1187 | |||
1188 | #define USB_TEST_EP_MASK 0x30 | ||
1189 | #define USB_TEST_EP_SHIFT 4 | ||
1190 | |||
1191 | #define USB_AGG_EN BIT(3) | ||
1192 | |||
1193 | #define MAC_ADDR_LEN 6 | ||
1194 | #define LAST_ENTRY_OF_TX_PKT_BUFFER 255 | ||
1195 | |||
1196 | #define POLLING_LLT_THRESHOLD 20 | ||
1197 | #define POLLING_READY_TIMEOUT_COUNT 1000 | ||
1198 | |||
1199 | #define MAX_MSS_DENSITY_2T 0x13 | ||
1200 | #define MAX_MSS_DENSITY_1T 0x0A | ||
1201 | |||
1202 | #define EPROM_CMD_OPERATING_MODE_MASK ((1<<7)|(1<<6)) | ||
1203 | #define EPROM_CMD_CONFIG 0x3 | ||
1204 | #define EPROM_CMD_LOAD 1 | ||
1205 | |||
1206 | #define HWSET_MAX_SIZE_92S HWSET_MAX_SIZE | ||
1207 | |||
1208 | #define WL_HWPDN_EN BIT(0) | ||
1209 | |||
1210 | #define HAL_8192C_HW_GPIO_WPS_BIT BIT(2) | ||
1211 | |||
1212 | #define RPMAC_RESET 0x100 | ||
1213 | #define RPMAC_TXSTART 0x104 | ||
1214 | #define RPMAC_TXLEGACYSIG 0x108 | ||
1215 | #define RPMAC_TXHTSIG1 0x10c | ||
1216 | #define RPMAC_TXHTSIG2 0x110 | ||
1217 | #define RPMAC_PHYDEBUG 0x114 | ||
1218 | #define RPMAC_TXPACKETNUM 0x118 | ||
1219 | #define RPMAC_TXIDLE 0x11c | ||
1220 | #define RPMAC_TXMACHEADER0 0x120 | ||
1221 | #define RPMAC_TXMACHEADER1 0x124 | ||
1222 | #define RPMAC_TXMACHEADER2 0x128 | ||
1223 | #define RPMAC_TXMACHEADER3 0x12c | ||
1224 | #define RPMAC_TXMACHEADER4 0x130 | ||
1225 | #define RPMAC_TXMACHEADER5 0x134 | ||
1226 | #define RPMAC_TXDADATYPE 0x138 | ||
1227 | #define RPMAC_TXRANDOMSEED 0x13c | ||
1228 | #define RPMAC_CCKPLCPPREAMBLE 0x140 | ||
1229 | #define RPMAC_CCKPLCPHEADER 0x144 | ||
1230 | #define RPMAC_CCKCRC16 0x148 | ||
1231 | #define RPMAC_OFDMRXCRC32OK 0x170 | ||
1232 | #define RPMAC_OFDMRXCRC32Er 0x174 | ||
1233 | #define RPMAC_OFDMRXPARITYER 0x178 | ||
1234 | #define RPMAC_OFDMRXCRC8ER 0x17c | ||
1235 | #define RPMAC_CCKCRXRC16ER 0x180 | ||
1236 | #define RPMAC_CCKCRXRC32ER 0x184 | ||
1237 | #define RPMAC_CCKCRXRC32OK 0x188 | ||
1238 | #define RPMAC_TXSTATUS 0x18c | ||
1239 | |||
1240 | #define RFPGA0_RFMOD 0x800 | ||
1241 | |||
1242 | #define RFPGA0_TXINFO 0x804 | ||
1243 | #define RFPGA0_PSDFUNCTION 0x808 | ||
1244 | |||
1245 | #define RFPGA0_TXGAINSTAGE 0x80c | ||
1246 | |||
1247 | #define RFPGA0_RFTIMING1 0x810 | ||
1248 | #define RFPGA0_RFTIMING2 0x814 | ||
1249 | |||
1250 | #define RFPGA0_XA_HSSIPARAMETER1 0x820 | ||
1251 | #define RFPGA0_XA_HSSIPARAMETER2 0x824 | ||
1252 | #define RFPGA0_XB_HSSIPARAMETER1 0x828 | ||
1253 | #define RFPGA0_XB_HSSIPARAMETER2 0x82c | ||
1254 | |||
1255 | #define RFPGA0_XA_LSSIPARAMETER 0x840 | ||
1256 | #define RFPGA0_XB_LSSIPARAMETER 0x844 | ||
1257 | |||
1258 | #define RFPGA0_RFWAKEUPPARAMETER 0x850 | ||
1259 | #define RFPGA0_RFSLEEPUPPARAMETER 0x854 | ||
1260 | |||
1261 | #define RFPGA0_XAB_SWITCHCONTROL 0x858 | ||
1262 | #define RFPGA0_XCD_SWITCHCONTROL 0x85c | ||
1263 | |||
1264 | #define RFPGA0_XA_RFINTERFACEOE 0x860 | ||
1265 | #define RFPGA0_XB_RFINTERFACEOE 0x864 | ||
1266 | |||
1267 | #define RFPGA0_XAB_RFINTERFACESW 0x870 | ||
1268 | #define RFPGA0_XCD_RFINTERFACESW 0x874 | ||
1269 | |||
1270 | #define rFPGA0_XAB_RFPARAMETER 0x878 | ||
1271 | #define rFPGA0_XCD_RFPARAMETER 0x87c | ||
1272 | |||
1273 | #define RFPGA0_ANALOGPARAMETER1 0x880 | ||
1274 | #define RFPGA0_ANALOGPARAMETER2 0x884 | ||
1275 | #define RFPGA0_ANALOGPARAMETER3 0x888 | ||
1276 | #define RFPGA0_ANALOGPARAMETER4 0x88c | ||
1277 | |||
1278 | #define RFPGA0_XA_LSSIREADBACK 0x8a0 | ||
1279 | #define RFPGA0_XB_LSSIREADBACK 0x8a4 | ||
1280 | #define RFPGA0_XC_LSSIREADBACK 0x8a8 | ||
1281 | #define RFPGA0_XD_LSSIREADBACK 0x8ac | ||
1282 | |||
1283 | #define RFPGA0_PSDREPORT 0x8b4 | ||
1284 | #define TRANSCEIVEA_HSPI_READBACK 0x8b8 | ||
1285 | #define TRANSCEIVEB_HSPI_READBACK 0x8bc | ||
1286 | #define RFPGA0_XAB_RFINTERFACERB 0x8e0 | ||
1287 | #define RFPGA0_XCD_RFINTERFACERB 0x8e4 | ||
1288 | |||
1289 | #define RFPGA1_RFMOD 0x900 | ||
1290 | |||
1291 | #define RFPGA1_TXBLOCK 0x904 | ||
1292 | #define RFPGA1_DEBUGSELECT 0x908 | ||
1293 | #define RFPGA1_TXINFO 0x90c | ||
1294 | |||
1295 | #define RCCK0_SYSTEM 0xa00 | ||
1296 | |||
1297 | #define RCCK0_AFESETTING 0xa04 | ||
1298 | #define RCCK0_CCA 0xa08 | ||
1299 | |||
1300 | #define RCCK0_RXAGC1 0xa0c | ||
1301 | #define RCCK0_RXAGC2 0xa10 | ||
1302 | |||
1303 | #define RCCK0_RXHP 0xa14 | ||
1304 | |||
1305 | #define RCCK0_DSPPARAMETER1 0xa18 | ||
1306 | #define RCCK0_DSPPARAMETER2 0xa1c | ||
1307 | |||
1308 | #define RCCK0_TXFILTER1 0xa20 | ||
1309 | #define RCCK0_TXFILTER2 0xa24 | ||
1310 | #define RCCK0_DEBUGPORT 0xa28 | ||
1311 | #define RCCK0_FALSEALARMREPORT 0xa2c | ||
1312 | #define RCCK0_TRSSIREPORT 0xa50 | ||
1313 | #define RCCK0_RXREPORT 0xa54 | ||
1314 | #define RCCK0_FACOUNTERLOWER 0xa5c | ||
1315 | #define RCCK0_FACOUNTERUPPER 0xa58 | ||
1316 | |||
1317 | #define ROFDM0_LSTF 0xc00 | ||
1318 | |||
1319 | #define ROFDM0_TRXPATHENABLE 0xc04 | ||
1320 | #define ROFDM0_TRMUXPAR 0xc08 | ||
1321 | #define ROFDM0_TRSWISOLATION 0xc0c | ||
1322 | |||
1323 | #define ROFDM0_XARXAFE 0xc10 | ||
1324 | #define ROFDM0_XARXIQIMBALANCE 0xc14 | ||
1325 | #define ROFDM0_XBRXAFE 0xc18 | ||
1326 | #define ROFDM0_XBRXIQIMBALANCE 0xc1c | ||
1327 | #define ROFDM0_XCRXAFE 0xc20 | ||
1328 | #define ROFDM0_XCRXIQIMBANLANCE 0xc24 | ||
1329 | #define ROFDM0_XDRXAFE 0xc28 | ||
1330 | #define ROFDM0_XDRXIQIMBALANCE 0xc2c | ||
1331 | |||
1332 | #define ROFDM0_RXDETECTOR1 0xc30 | ||
1333 | #define ROFDM0_RXDETECTOR2 0xc34 | ||
1334 | #define ROFDM0_RXDETECTOR3 0xc38 | ||
1335 | #define ROFDM0_RXDETECTOR4 0xc3c | ||
1336 | |||
1337 | #define ROFDM0_RXDSP 0xc40 | ||
1338 | #define ROFDM0_CFOANDDAGC 0xc44 | ||
1339 | #define ROFDM0_CCADROPTHRESHOLD 0xc48 | ||
1340 | #define ROFDM0_ECCATHRESHOLD 0xc4c | ||
1341 | |||
1342 | #define ROFDM0_XAAGCCORE1 0xc50 | ||
1343 | #define ROFDM0_XAAGCCORE2 0xc54 | ||
1344 | #define ROFDM0_XBAGCCORE1 0xc58 | ||
1345 | #define ROFDM0_XBAGCCORE2 0xc5c | ||
1346 | #define ROFDM0_XCAGCCORE1 0xc60 | ||
1347 | #define ROFDM0_XCAGCCORE2 0xc64 | ||
1348 | #define ROFDM0_XDAGCCORE1 0xc68 | ||
1349 | #define ROFDM0_XDAGCCORE2 0xc6c | ||
1350 | |||
1351 | #define ROFDM0_AGCPARAMETER1 0xc70 | ||
1352 | #define ROFDM0_AGCPARAMETER2 0xc74 | ||
1353 | #define ROFDM0_AGCRSSITABLE 0xc78 | ||
1354 | #define ROFDM0_HTSTFAGC 0xc7c | ||
1355 | |||
1356 | #define ROFDM0_XATXIQIMBALANCE 0xc80 | ||
1357 | #define ROFDM0_XATXAFE 0xc84 | ||
1358 | #define ROFDM0_XBTXIQIMBALANCE 0xc88 | ||
1359 | #define ROFDM0_XBTXAFE 0xc8c | ||
1360 | #define ROFDM0_XCTXIQIMBALANCE 0xc90 | ||
1361 | #define ROFDM0_XCTXAFE 0xc94 | ||
1362 | #define ROFDM0_XDTXIQIMBALANCE 0xc98 | ||
1363 | #define ROFDM0_XDTXAFE 0xc9c | ||
1364 | |||
1365 | #define ROFDM0_RXIQEXTANTA 0xca0 | ||
1366 | |||
1367 | #define ROFDM0_RXHPPARAMETER 0xce0 | ||
1368 | #define ROFDM0_TXPSEUDONOISEWGT 0xce4 | ||
1369 | #define ROFDM0_FRAMESYNC 0xcf0 | ||
1370 | #define ROFDM0_DFSREPORT 0xcf4 | ||
1371 | #define ROFDM0_TXCOEFF1 0xca4 | ||
1372 | #define ROFDM0_TXCOEFF2 0xca8 | ||
1373 | #define ROFDM0_TXCOEFF3 0xcac | ||
1374 | #define ROFDM0_TXCOEFF4 0xcb0 | ||
1375 | #define ROFDM0_TXCOEFF5 0xcb4 | ||
1376 | #define ROFDM0_TXCOEFF6 0xcb8 | ||
1377 | |||
1378 | #define ROFDM1_LSTF 0xd00 | ||
1379 | #define ROFDM1_TRXPATHENABLE 0xd04 | ||
1380 | |||
1381 | #define ROFDM1_CF0 0xd08 | ||
1382 | #define ROFDM1_CSI1 0xd10 | ||
1383 | #define ROFDM1_SBD 0xd14 | ||
1384 | #define ROFDM1_CSI2 0xd18 | ||
1385 | #define ROFDM1_CFOTRACKING 0xd2c | ||
1386 | #define ROFDM1_TRXMESAURE1 0xd34 | ||
1387 | #define ROFDM1_INTFDET 0xd3c | ||
1388 | #define ROFDM1_PSEUDONOISESTATEAB 0xd50 | ||
1389 | #define ROFDM1_PSEUDONOISESTATECD 0xd54 | ||
1390 | #define ROFDM1_RXPSEUDONOISEWGT 0xd58 | ||
1391 | |||
1392 | #define ROFDM_PHYCOUNTER1 0xda0 | ||
1393 | #define ROFDM_PHYCOUNTER2 0xda4 | ||
1394 | #define ROFDM_PHYCOUNTER3 0xda8 | ||
1395 | |||
1396 | #define ROFDM_SHORTCFOAB 0xdac | ||
1397 | #define ROFDM_SHORTCFOCD 0xdb0 | ||
1398 | #define ROFDM_LONGCFOAB 0xdb4 | ||
1399 | #define ROFDM_LONGCFOCD 0xdb8 | ||
1400 | #define ROFDM_TAILCF0AB 0xdbc | ||
1401 | #define ROFDM_TAILCF0CD 0xdc0 | ||
1402 | #define ROFDM_PWMEASURE1 0xdc4 | ||
1403 | #define ROFDM_PWMEASURE2 0xdc8 | ||
1404 | #define ROFDM_BWREPORT 0xdcc | ||
1405 | #define ROFDM_AGCREPORT 0xdd0 | ||
1406 | #define ROFDM_RXSNR 0xdd4 | ||
1407 | #define ROFDM_RXEVMCSI 0xdd8 | ||
1408 | #define ROFDM_SIGREPORT 0xddc | ||
1409 | |||
1410 | #define RTXAGC_A_RATE18_06 0xe00 | ||
1411 | #define RTXAGC_A_RATE54_24 0xe04 | ||
1412 | #define RTXAGC_A_CCK1_MCS32 0xe08 | ||
1413 | #define RTXAGC_A_MCS03_MCS00 0xe10 | ||
1414 | #define RTXAGC_A_MCS07_MCS04 0xe14 | ||
1415 | #define RTXAGC_A_MCS11_MCS08 0xe18 | ||
1416 | #define RTXAGC_A_MCS15_MCS12 0xe1c | ||
1417 | |||
1418 | #define RTXAGC_B_RATE18_06 0x830 | ||
1419 | #define RTXAGC_B_RATE54_24 0x834 | ||
1420 | #define RTXAGC_B_CCK1_55_MCS32 0x838 | ||
1421 | #define RTXAGC_B_MCS03_MCS00 0x83c | ||
1422 | #define RTXAGC_B_MCS07_MCS04 0x848 | ||
1423 | #define RTXAGC_B_MCS11_MCS08 0x84c | ||
1424 | #define RTXAGC_B_MCS15_MCS12 0x868 | ||
1425 | #define RTXAGC_B_CCK11_A_CCK2_11 0x86c | ||
1426 | |||
1427 | #define RZEBRA1_HSSIENABLE 0x0 | ||
1428 | #define RZEBRA1_TRXENABLE1 0x1 | ||
1429 | #define RZEBRA1_TRXENABLE2 0x2 | ||
1430 | #define RZEBRA1_AGC 0x4 | ||
1431 | #define RZEBRA1_CHARGEPUMP 0x5 | ||
1432 | #define RZEBRA1_CHANNEL 0x7 | ||
1433 | |||
1434 | #define RZEBRA1_TXGAIN 0x8 | ||
1435 | #define RZEBRA1_TXLPF 0x9 | ||
1436 | #define RZEBRA1_RXLPF 0xb | ||
1437 | #define RZEBRA1_RXHPFCORNER 0xc | ||
1438 | |||
1439 | #define RGLOBALCTRL 0 | ||
1440 | #define RRTL8256_TXLPF 19 | ||
1441 | #define RRTL8256_RXLPF 11 | ||
1442 | #define RRTL8258_TXLPF 0x11 | ||
1443 | #define RRTL8258_RXLPF 0x13 | ||
1444 | #define RRTL8258_RSSILPF 0xa | ||
1445 | |||
1446 | #define RF_AC 0x00 | ||
1447 | |||
1448 | #define RF_IQADJ_G1 0x01 | ||
1449 | #define RF_IQADJ_G2 0x02 | ||
1450 | #define RF_POW_TRSW 0x05 | ||
1451 | |||
1452 | #define RF_GAIN_RX 0x06 | ||
1453 | #define RF_GAIN_TX 0x07 | ||
1454 | |||
1455 | #define RF_TXM_IDAC 0x08 | ||
1456 | #define RF_BS_IQGEN 0x0F | ||
1457 | |||
1458 | #define RF_MODE1 0x10 | ||
1459 | #define RF_MODE2 0x11 | ||
1460 | |||
1461 | #define RF_RX_AGC_HP 0x12 | ||
1462 | #define RF_TX_AGC 0x13 | ||
1463 | #define RF_BIAS 0x14 | ||
1464 | #define RF_IPA 0x15 | ||
1465 | #define RF_POW_ABILITY 0x17 | ||
1466 | #define RF_MODE_AG 0x18 | ||
1467 | #define RRFCHANNEL 0x18 | ||
1468 | #define RF_CHNLBW 0x18 | ||
1469 | #define RF_TOP 0x19 | ||
1470 | |||
1471 | #define RF_RX_G1 0x1A | ||
1472 | #define RF_RX_G2 0x1B | ||
1473 | |||
1474 | #define RF_RX_BB2 0x1C | ||
1475 | #define RF_RX_BB1 0x1D | ||
1476 | |||
1477 | #define RF_RCK1 0x1E | ||
1478 | #define RF_RCK2 0x1F | ||
1479 | |||
1480 | #define RF_TX_G1 0x20 | ||
1481 | #define RF_TX_G2 0x21 | ||
1482 | #define RF_TX_G3 0x22 | ||
1483 | |||
1484 | #define RF_TX_BB1 0x23 | ||
1485 | #define RF_T_METER 0x24 | ||
1486 | |||
1487 | #define RF_SYN_G1 0x25 | ||
1488 | #define RF_SYN_G2 0x26 | ||
1489 | #define RF_SYN_G3 0x27 | ||
1490 | #define RF_SYN_G4 0x28 | ||
1491 | #define RF_SYN_G5 0x29 | ||
1492 | #define RF_SYN_G6 0x2A | ||
1493 | #define RF_SYN_G7 0x2B | ||
1494 | #define RF_SYN_G8 0x2C | ||
1495 | |||
1496 | #define RF_RCK_OS 0x30 | ||
1497 | #define RF_TXPA_G1 0x31 | ||
1498 | #define RF_TXPA_G2 0x32 | ||
1499 | #define RF_TXPA_G3 0x33 | ||
1500 | |||
1501 | #define BBBRESETB 0x100 | ||
1502 | #define BGLOBALRESETB 0x200 | ||
1503 | #define BOFDMTXSTART 0x4 | ||
1504 | #define BCCKTXSTART 0x8 | ||
1505 | #define BCRC32DEBUG 0x100 | ||
1506 | #define BPMACLOOPBACK 0x10 | ||
1507 | #define BTXLSIG 0xffffff | ||
1508 | #define BOFDMTXRATE 0xf | ||
1509 | #define BOFDMTXRESERVED 0x10 | ||
1510 | #define BOFDMTXLENGTH 0x1ffe0 | ||
1511 | #define BOFDMTXPARITY 0x20000 | ||
1512 | #define BTXHTSIG1 0xffffff | ||
1513 | #define BTXHTMCSRATE 0x7f | ||
1514 | #define BTXHTBW 0x80 | ||
1515 | #define BTXHTLENGTH 0xffff00 | ||
1516 | #define BTXHTSIG2 0xffffff | ||
1517 | #define BTXHTSMOOTHING 0x1 | ||
1518 | #define BTXHTSOUNDING 0x2 | ||
1519 | #define BTXHTRESERVED 0x4 | ||
1520 | #define BTXHTAGGREATION 0x8 | ||
1521 | #define BTXHTSTBC 0x30 | ||
1522 | #define BTXHTADVANCECODING 0x40 | ||
1523 | #define BTXHTSHORTGI 0x80 | ||
1524 | #define BTXHTNUMBERHT_LTF 0x300 | ||
1525 | #define BTXHTCRC8 0x3fc00 | ||
1526 | #define BCOUNTERRESET 0x10000 | ||
1527 | #define BNUMOFOFDMTX 0xffff | ||
1528 | #define BNUMOFCCKTX 0xffff0000 | ||
1529 | #define BTXIDLEINTERVAL 0xffff | ||
1530 | #define BOFDMSERVICE 0xffff0000 | ||
1531 | #define BTXMACHEADER 0xffffffff | ||
1532 | #define BTXDATAINIT 0xff | ||
1533 | #define BTXHTMODE 0x100 | ||
1534 | #define BTXDATATYPE 0x30000 | ||
1535 | #define BTXRANDOMSEED 0xffffffff | ||
1536 | #define BCCKTXPREAMBLE 0x1 | ||
1537 | #define BCCKTXSFD 0xffff0000 | ||
1538 | #define BCCKTXSIG 0xff | ||
1539 | #define BCCKTXSERVICE 0xff00 | ||
1540 | #define BCCKLENGTHEXT 0x8000 | ||
1541 | #define BCCKTXLENGHT 0xffff0000 | ||
1542 | #define BCCKTXCRC16 0xffff | ||
1543 | #define BCCKTXSTATUS 0x1 | ||
1544 | #define BOFDMTXSTATUS 0x2 | ||
1545 | #define IS_BB_REG_OFFSET_92S(_Offset) \ | ||
1546 | ((_Offset >= 0x800) && (_Offset <= 0xfff)) | ||
1547 | |||
1548 | #define BRFMOD 0x1 | ||
1549 | #define BJAPANMODE 0x2 | ||
1550 | #define BCCKTXSC 0x30 | ||
1551 | #define BCCKEN 0x1000000 | ||
1552 | #define BOFDMEN 0x2000000 | ||
1553 | |||
1554 | #define BOFDMRXADCPHASE 0x10000 | ||
1555 | #define BOFDMTXDACPHASE 0x40000 | ||
1556 | #define BXATXAGC 0x3f | ||
1557 | |||
1558 | #define BXBTXAGC 0xf00 | ||
1559 | #define BXCTXAGC 0xf000 | ||
1560 | #define BXDTXAGC 0xf0000 | ||
1561 | |||
1562 | #define BPASTART 0xf0000000 | ||
1563 | #define BTRSTART 0x00f00000 | ||
1564 | #define BRFSTART 0x0000f000 | ||
1565 | #define BBBSTART 0x000000f0 | ||
1566 | #define BBBCCKSTART 0x0000000f | ||
1567 | #define BPAEND 0xf | ||
1568 | #define BTREND 0x0f000000 | ||
1569 | #define BRFEND 0x000f0000 | ||
1570 | #define BCCAMASK 0x000000f0 | ||
1571 | #define BR2RCCAMASK 0x00000f00 | ||
1572 | #define BHSSI_R2TDELAY 0xf8000000 | ||
1573 | #define BHSSI_T2RDELAY 0xf80000 | ||
1574 | #define BCONTXHSSI 0x400 | ||
1575 | #define BIGFROMCCK 0x200 | ||
1576 | #define BAGCADDRESS 0x3f | ||
1577 | #define BRXHPTX 0x7000 | ||
1578 | #define BRXHP2RX 0x38000 | ||
1579 | #define BRXHPCCKINI 0xc0000 | ||
1580 | #define BAGCTXCODE 0xc00000 | ||
1581 | #define BAGCRXCODE 0x300000 | ||
1582 | |||
1583 | #define B3WIREDATALENGTH 0x800 | ||
1584 | #define B3WIREADDREAALENGTH 0x400 | ||
1585 | |||
1586 | #define B3WIRERFPOWERDOWN 0x1 | ||
1587 | #define B5GPAPEPOLARITY 0x40000000 | ||
1588 | #define B2GPAPEPOLARITY 0x80000000 | ||
1589 | #define BRFSW_TXDEFAULTANT 0x3 | ||
1590 | #define BRFSW_TXOPTIONANT 0x30 | ||
1591 | #define BRFSW_RXDEFAULTANT 0x300 | ||
1592 | #define BRFSW_RXOPTIONANT 0x3000 | ||
1593 | #define BRFSI_3WIREDATA 0x1 | ||
1594 | #define BRFSI_3WIRECLOCK 0x2 | ||
1595 | #define BRFSI_3WIRELOAD 0x4 | ||
1596 | #define BRFSI_3WIRERW 0x8 | ||
1597 | #define BRFSI_3WIRE 0xf | ||
1598 | |||
1599 | #define BRFSI_RFENV 0x10 | ||
1600 | |||
1601 | #define BRFSI_TRSW 0x20 | ||
1602 | #define BRFSI_TRSWB 0x40 | ||
1603 | #define BRFSI_ANTSW 0x100 | ||
1604 | #define BRFSI_ANTSWB 0x200 | ||
1605 | #define BRFSI_PAPE 0x400 | ||
1606 | #define BRFSI_PAPE5G 0x800 | ||
1607 | #define BBANDSELECT 0x1 | ||
1608 | #define BHTSIG2_GI 0x80 | ||
1609 | #define BHTSIG2_SMOOTHING 0x01 | ||
1610 | #define BHTSIG2_SOUNDING 0x02 | ||
1611 | #define BHTSIG2_AGGREATON 0x08 | ||
1612 | #define BHTSIG2_STBC 0x30 | ||
1613 | #define BHTSIG2_ADVCODING 0x40 | ||
1614 | #define BHTSIG2_NUMOFHTLTF 0x300 | ||
1615 | #define BHTSIG2_CRC8 0x3fc | ||
1616 | #define BHTSIG1_MCS 0x7f | ||
1617 | #define BHTSIG1_BANDWIDTH 0x80 | ||
1618 | #define BHTSIG1_HTLENGTH 0xffff | ||
1619 | #define BLSIG_RATE 0xf | ||
1620 | #define BLSIG_RESERVED 0x10 | ||
1621 | #define BLSIG_LENGTH 0x1fffe | ||
1622 | #define BLSIG_PARITY 0x20 | ||
1623 | #define BCCKRXPHASE 0x4 | ||
1624 | |||
1625 | #define BLSSIREADADDRESS 0x7f800000 | ||
1626 | #define BLSSIREADEDGE 0x80000000 | ||
1627 | |||
1628 | #define BLSSIREADBACKDATA 0xfffff | ||
1629 | |||
1630 | #define BLSSIREADOKFLAG 0x1000 | ||
1631 | #define BCCKSAMPLERATE 0x8 | ||
1632 | #define BREGULATOR0STANDBY 0x1 | ||
1633 | #define BREGULATORPLLSTANDBY 0x2 | ||
1634 | #define BREGULATOR1STANDBY 0x4 | ||
1635 | #define BPLLPOWERUP 0x8 | ||
1636 | #define BDPLLPOWERUP 0x10 | ||
1637 | #define BDA10POWERUP 0x20 | ||
1638 | #define BAD7POWERUP 0x200 | ||
1639 | #define BDA6POWERUP 0x2000 | ||
1640 | #define BXTALPOWERUP 0x4000 | ||
1641 | #define B40MDCLKPOWERUP 0x8000 | ||
1642 | #define BDA6DEBUGMODE 0x20000 | ||
1643 | #define BDA6SWING 0x380000 | ||
1644 | |||
1645 | #define BADCLKPHASE 0x4000000 | ||
1646 | #define B80MCLKDELAY 0x18000000 | ||
1647 | #define BAFEWATCHDOGENABLE 0x20000000 | ||
1648 | |||
1649 | #define BXTALCAP01 0xc0000000 | ||
1650 | #define BXTALCAP23 0x3 | ||
1651 | #define BXTALCAP92X 0x0f000000 | ||
1652 | #define BXTALCAP 0x0f000000 | ||
1653 | |||
1654 | #define BINTDIFCLKENABLE 0x400 | ||
1655 | #define BEXTSIGCLKENABLE 0x800 | ||
1656 | #define BBANDGAP_MBIAS_POWERUP 0x10000 | ||
1657 | #define BAD11SH_GAIN 0xc0000 | ||
1658 | #define BAD11NPUT_RANGE 0x700000 | ||
1659 | #define BAD110P_CURRENT 0x3800000 | ||
1660 | #define BLPATH_LOOPBACK 0x4000000 | ||
1661 | #define BQPATH_LOOPBACK 0x8000000 | ||
1662 | #define BAFE_LOOPBACK 0x10000000 | ||
1663 | #define BDA10_SWING 0x7e0 | ||
1664 | #define BDA10_REVERSE 0x800 | ||
1665 | #define BDA_CLK_SOURCE 0x1000 | ||
1666 | #define BDA7INPUT_RANGE 0x6000 | ||
1667 | #define BDA7_GAIN 0x38000 | ||
1668 | #define BDA7OUTPUT_CM_MODE 0x40000 | ||
1669 | #define BDA7INPUT_CM_MODE 0x380000 | ||
1670 | #define BDA7CURRENT 0xc00000 | ||
1671 | #define BREGULATOR_ADJUST 0x7000000 | ||
1672 | #define BAD11POWERUP_ATTX 0x1 | ||
1673 | #define BDA10PS_ATTX 0x10 | ||
1674 | #define BAD11POWERUP_ATRX 0x100 | ||
1675 | #define BDA10PS_ATRX 0x1000 | ||
1676 | #define BCCKRX_AGC_FORMAT 0x200 | ||
1677 | #define BPSDFFT_SAMPLE_POINT 0xc000 | ||
1678 | #define BPSD_AVERAGE_NUM 0x3000 | ||
1679 | #define BIQPATH_CONTROL 0xc00 | ||
1680 | #define BPSD_FREQ 0x3ff | ||
1681 | #define BPSD_ANTENNA_PATH 0x30 | ||
1682 | #define BPSD_IQ_SWITCH 0x40 | ||
1683 | #define BPSD_RX_TRIGGER 0x400000 | ||
1684 | #define BPSD_TX_TRIGGER 0x80000000 | ||
1685 | #define BPSD_SINE_TONE_SCALE 0x7f000000 | ||
1686 | #define BPSD_REPORT 0xffff | ||
1687 | |||
1688 | #define BOFDM_TXSC 0x30000000 | ||
1689 | #define BCCK_TXON 0x1 | ||
1690 | #define BOFDM_TXON 0x2 | ||
1691 | #define BDEBUG_PAGE 0xfff | ||
1692 | #define BDEBUG_ITEM 0xff | ||
1693 | #define BANTL 0x10 | ||
1694 | #define BANT_NONHT 0x100 | ||
1695 | #define BANT_HT1 0x1000 | ||
1696 | #define BANT_HT2 0x10000 | ||
1697 | #define BANT_HT1S1 0x100000 | ||
1698 | #define BANT_NONHTS1 0x1000000 | ||
1699 | |||
1700 | #define BCCK_BBMODE 0x3 | ||
1701 | #define BCCK_TXPOWERSAVING 0x80 | ||
1702 | #define BCCK_RXPOWERSAVING 0x40 | ||
1703 | |||
1704 | #define BCCK_SIDEBAND 0x10 | ||
1705 | |||
1706 | #define BCCK_SCRAMBLE 0x8 | ||
1707 | #define BCCK_ANTDIVERSITY 0x8000 | ||
1708 | #define BCCK_CARRIER_RECOVERY 0x4000 | ||
1709 | #define BCCK_TXRATE 0x3000 | ||
1710 | #define BCCK_DCCANCEL 0x0800 | ||
1711 | #define BCCK_ISICANCEL 0x0400 | ||
1712 | #define BCCK_MATCH_FILTER 0x0200 | ||
1713 | #define BCCK_EQUALIZER 0x0100 | ||
1714 | #define BCCK_PREAMBLE_DETECT 0x800000 | ||
1715 | #define BCCK_FAST_FALSECCA 0x400000 | ||
1716 | #define BCCK_CH_ESTSTART 0x300000 | ||
1717 | #define BCCK_CCA_COUNT 0x080000 | ||
1718 | #define BCCK_CS_LIM 0x070000 | ||
1719 | #define BCCK_BIST_MODE 0x80000000 | ||
1720 | #define BCCK_CCAMASK 0x40000000 | ||
1721 | #define BCCK_TX_DAC_PHASE 0x4 | ||
1722 | #define BCCK_RX_ADC_PHASE 0x20000000 | ||
1723 | #define BCCKR_CP_MODE 0x0100 | ||
1724 | #define BCCK_TXDC_OFFSET 0xf0 | ||
1725 | #define BCCK_RXDC_OFFSET 0xf | ||
1726 | #define BCCK_CCA_MODE 0xc000 | ||
1727 | #define BCCK_FALSECS_LIM 0x3f00 | ||
1728 | #define BCCK_CS_RATIO 0xc00000 | ||
1729 | #define BCCK_CORGBIT_SEL 0x300000 | ||
1730 | #define BCCK_PD_LIM 0x0f0000 | ||
1731 | #define BCCK_NEWCCA 0x80000000 | ||
1732 | #define BCCK_RXHP_OF_IG 0x8000 | ||
1733 | #define BCCK_RXIG 0x7f00 | ||
1734 | #define BCCK_LNA_POLARITY 0x800000 | ||
1735 | #define BCCK_RX1ST_BAIN 0x7f0000 | ||
1736 | #define BCCK_RF_EXTEND 0x20000000 | ||
1737 | #define BCCK_RXAGC_SATLEVEL 0x1f000000 | ||
1738 | #define BCCK_RXAGC_SATCOUNT 0xe0 | ||
1739 | #define bCCKRxRFSettle 0x1f | ||
1740 | #define BCCK_FIXED_RXAGC 0x8000 | ||
1741 | #define BCCK_ANTENNA_POLARITY 0x2000 | ||
1742 | #define BCCK_TXFILTER_TYPE 0x0c00 | ||
1743 | #define BCCK_RXAGC_REPORTTYPE 0x0300 | ||
1744 | #define BCCK_RXDAGC_EN 0x80000000 | ||
1745 | #define BCCK_RXDAGC_PERIOD 0x20000000 | ||
1746 | #define BCCK_RXDAGC_SATLEVEL 0x1f000000 | ||
1747 | #define BCCK_TIMING_RECOVERY 0x800000 | ||
1748 | #define BCCK_TXC0 0x3f0000 | ||
1749 | #define BCCK_TXC1 0x3f000000 | ||
1750 | #define BCCK_TXC2 0x3f | ||
1751 | #define BCCK_TXC3 0x3f00 | ||
1752 | #define BCCK_TXC4 0x3f0000 | ||
1753 | #define BCCK_TXC5 0x3f000000 | ||
1754 | #define BCCK_TXC6 0x3f | ||
1755 | #define BCCK_TXC7 0x3f00 | ||
1756 | #define BCCK_DEBUGPORT 0xff0000 | ||
1757 | #define BCCK_DAC_DEBUG 0x0f000000 | ||
1758 | #define BCCK_FALSEALARM_ENABLE 0x8000 | ||
1759 | #define BCCK_FALSEALARM_READ 0x4000 | ||
1760 | #define BCCK_TRSSI 0x7f | ||
1761 | #define BCCK_RXAGC_REPORT 0xfe | ||
1762 | #define BCCK_RXREPORT_ANTSEL 0x80000000 | ||
1763 | #define BCCK_RXREPORT_MFOFF 0x40000000 | ||
1764 | #define BCCK_RXREPORT_SQLOSS 0x20000000 | ||
1765 | #define BCCK_RXREPORT_PKTLOSS 0x10000000 | ||
1766 | #define BCCK_RXREPORT_LOCKEDBIT 0x08000000 | ||
1767 | #define BCCK_RXREPORT_RATEERROR 0x04000000 | ||
1768 | #define BCCK_RXREPORT_RXRATE 0x03000000 | ||
1769 | #define BCCK_RXFA_COUNTER_LOWER 0xff | ||
1770 | #define BCCK_RXFA_COUNTER_UPPER 0xff000000 | ||
1771 | #define BCCK_RXHPAGC_START 0xe000 | ||
1772 | #define BCCK_RXHPAGC_FINAL 0x1c00 | ||
1773 | #define BCCK_RXFALSEALARM_ENABLE 0x8000 | ||
1774 | #define BCCK_FACOUNTER_FREEZE 0x4000 | ||
1775 | #define BCCK_TXPATH_SEL 0x10000000 | ||
1776 | #define BCCK_DEFAULT_RXPATH 0xc000000 | ||
1777 | #define BCCK_OPTION_RXPATH 0x3000000 | ||
1778 | |||
1779 | #define BNUM_OFSTF 0x3 | ||
1780 | #define BSHIFT_L 0xc0 | ||
1781 | #define BGI_TH 0xc | ||
1782 | #define BRXPATH_A 0x1 | ||
1783 | #define BRXPATH_B 0x2 | ||
1784 | #define BRXPATH_C 0x4 | ||
1785 | #define BRXPATH_D 0x8 | ||
1786 | #define BTXPATH_A 0x1 | ||
1787 | #define BTXPATH_B 0x2 | ||
1788 | #define BTXPATH_C 0x4 | ||
1789 | #define BTXPATH_D 0x8 | ||
1790 | #define BTRSSI_FREQ 0x200 | ||
1791 | #define BADC_BACKOFF 0x3000 | ||
1792 | #define BDFIR_BACKOFF 0xc000 | ||
1793 | #define BTRSSI_LATCH_PHASE 0x10000 | ||
1794 | #define BRX_LDC_OFFSET 0xff | ||
1795 | #define BRX_QDC_OFFSET 0xff00 | ||
1796 | #define BRX_DFIR_MODE 0x1800000 | ||
1797 | #define BRX_DCNF_TYPE 0xe000000 | ||
1798 | #define BRXIQIMB_A 0x3ff | ||
1799 | #define BRXIQIMB_B 0xfc00 | ||
1800 | #define BRXIQIMB_C 0x3f0000 | ||
1801 | #define BRXIQIMB_D 0xffc00000 | ||
1802 | #define BDC_DC_NOTCH 0x60000 | ||
1803 | #define BRXNB_NOTCH 0x1f000000 | ||
1804 | #define BPD_TH 0xf | ||
1805 | #define BPD_TH_OPT2 0xc000 | ||
1806 | #define BPWED_TH 0x700 | ||
1807 | #define BIFMF_WIN_L 0x800 | ||
1808 | #define BPD_OPTION 0x1000 | ||
1809 | #define BMF_WIN_L 0xe000 | ||
1810 | #define BBW_SEARCH_L 0x30000 | ||
1811 | #define BWIN_ENH_L 0xc0000 | ||
1812 | #define BBW_TH 0x700000 | ||
1813 | #define BED_TH2 0x3800000 | ||
1814 | #define BBW_OPTION 0x4000000 | ||
1815 | #define BRADIO_TH 0x18000000 | ||
1816 | #define BWINDOW_L 0xe0000000 | ||
1817 | #define BSBD_OPTION 0x1 | ||
1818 | #define BFRAME_TH 0x1c | ||
1819 | #define BFS_OPTION 0x60 | ||
1820 | #define BDC_SLOPE_CHECK 0x80 | ||
1821 | #define BFGUARD_COUNTER_DC_L 0xe00 | ||
1822 | #define BFRAME_WEIGHT_SHORT 0x7000 | ||
1823 | #define BSUB_TUNE 0xe00000 | ||
1824 | #define BFRAME_DC_LENGTH 0xe000000 | ||
1825 | #define BSBD_START_OFFSET 0x30000000 | ||
1826 | #define BFRAME_TH_2 0x7 | ||
1827 | #define BFRAME_GI2_TH 0x38 | ||
1828 | #define BGI2_SYNC_EN 0x40 | ||
1829 | #define BSARCH_SHORT_EARLY 0x300 | ||
1830 | #define BSARCH_SHORT_LATE 0xc00 | ||
1831 | #define BSARCH_GI2_LATE 0x70000 | ||
1832 | #define BCFOANTSUM 0x1 | ||
1833 | #define BCFOACC 0x2 | ||
1834 | #define BCFOSTARTOFFSET 0xc | ||
1835 | #define BCFOLOOPBACK 0x70 | ||
1836 | #define BCFOSUMWEIGHT 0x80 | ||
1837 | #define BDAGCENABLE 0x10000 | ||
1838 | #define BTXIQIMB_A 0x3ff | ||
1839 | #define BTXIQIMB_b 0xfc00 | ||
1840 | #define BTXIQIMB_C 0x3f0000 | ||
1841 | #define BTXIQIMB_D 0xffc00000 | ||
1842 | #define BTXIDCOFFSET 0xff | ||
1843 | #define BTXIQDCOFFSET 0xff00 | ||
1844 | #define BTXDFIRMODE 0x10000 | ||
1845 | #define BTXPESUDO_NOISEON 0x4000000 | ||
1846 | #define BTXPESUDO_NOISE_A 0xff | ||
1847 | #define BTXPESUDO_NOISE_B 0xff00 | ||
1848 | #define BTXPESUDO_NOISE_C 0xff0000 | ||
1849 | #define BTXPESUDO_NOISE_D 0xff000000 | ||
1850 | #define BCCA_DROPOPTION 0x20000 | ||
1851 | #define BCCA_DROPTHRES 0xfff00000 | ||
1852 | #define BEDCCA_H 0xf | ||
1853 | #define BEDCCA_L 0xf0 | ||
1854 | #define BLAMBDA_ED 0x300 | ||
1855 | #define BRX_INITIALGAIN 0x7f | ||
1856 | #define BRX_ANTDIV_EN 0x80 | ||
1857 | #define BRX_AGC_ADDRESS_FOR_LNA 0x7f00 | ||
1858 | #define BRX_HIGHPOWER_FLOW 0x8000 | ||
1859 | #define BRX_AGC_FREEZE_THRES 0xc0000 | ||
1860 | #define BRX_FREEZESTEP_AGC1 0x300000 | ||
1861 | #define BRX_FREEZESTEP_AGC2 0xc00000 | ||
1862 | #define BRX_FREEZESTEP_AGC3 0x3000000 | ||
1863 | #define BRX_FREEZESTEP_AGC0 0xc000000 | ||
1864 | #define BRXRSSI_CMP_EN 0x10000000 | ||
1865 | #define BRXQUICK_AGCEN 0x20000000 | ||
1866 | #define BRXAGC_FREEZE_THRES_MODE 0x40000000 | ||
1867 | #define BRX_OVERFLOW_CHECKTYPE 0x80000000 | ||
1868 | #define BRX_AGCSHIFT 0x7f | ||
1869 | #define BTRSW_TRI_ONLY 0x80 | ||
1870 | #define BPOWER_THRES 0x300 | ||
1871 | #define BRXAGC_EN 0x1 | ||
1872 | #define BRXAGC_TOGETHER_EN 0x2 | ||
1873 | #define BRXAGC_MIN 0x4 | ||
1874 | #define BRXHP_INI 0x7 | ||
1875 | #define BRXHP_TRLNA 0x70 | ||
1876 | #define BRXHP_RSSI 0x700 | ||
1877 | #define BRXHP_BBP1 0x7000 | ||
1878 | #define BRXHP_BBP2 0x70000 | ||
1879 | #define BRXHP_BBP3 0x700000 | ||
1880 | #define BRSSI_H 0x7f0000 | ||
1881 | #define BRSSI_GEN 0x7f000000 | ||
1882 | #define BRXSETTLE_TRSW 0x7 | ||
1883 | #define BRXSETTLE_LNA 0x38 | ||
1884 | #define BRXSETTLE_RSSI 0x1c0 | ||
1885 | #define BRXSETTLE_BBP 0xe00 | ||
1886 | #define BRXSETTLE_RXHP 0x7000 | ||
1887 | #define BRXSETTLE_ANTSW_RSSI 0x38000 | ||
1888 | #define BRXSETTLE_ANTSW 0xc0000 | ||
1889 | #define BRXPROCESS_TIME_DAGC 0x300000 | ||
1890 | #define BRXSETTLE_HSSI 0x400000 | ||
1891 | #define BRXPROCESS_TIME_BBPPW 0x800000 | ||
1892 | #define BRXANTENNA_POWER_SHIFT 0x3000000 | ||
1893 | #define BRSSI_TABLE_SELECT 0xc000000 | ||
1894 | #define BRXHP_FINAL 0x7000000 | ||
1895 | #define BRXHPSETTLE_BBP 0x7 | ||
1896 | #define BRXHTSETTLE_HSSI 0x8 | ||
1897 | #define BRXHTSETTLE_RXHP 0x70 | ||
1898 | #define BRXHTSETTLE_BBPPW 0x80 | ||
1899 | #define BRXHTSETTLE_IDLE 0x300 | ||
1900 | #define BRXHTSETTLE_RESERVED 0x1c00 | ||
1901 | #define BRXHT_RXHP_EN 0x8000 | ||
1902 | #define BRXAGC_FREEZE_THRES 0x30000 | ||
1903 | #define BRXAGC_TOGETHEREN 0x40000 | ||
1904 | #define BRXHTAGC_MIN 0x80000 | ||
1905 | #define BRXHTAGC_EN 0x100000 | ||
1906 | #define BRXHTDAGC_EN 0x200000 | ||
1907 | #define BRXHT_RXHP_BBP 0x1c00000 | ||
1908 | #define BRXHT_RXHP_FINAL 0xe0000000 | ||
1909 | #define BRXPW_RADIO_TH 0x3 | ||
1910 | #define BRXPW_RADIO_EN 0x4 | ||
1911 | #define BRXMF_HOLD 0x3800 | ||
1912 | #define BRXPD_DELAY_TH1 0x38 | ||
1913 | #define BRXPD_DELAY_TH2 0x1c0 | ||
1914 | #define BRXPD_DC_COUNT_MAX 0x600 | ||
1915 | #define BRXPD_DELAY_TH 0x8000 | ||
1916 | #define BRXPROCESS_DELAY 0xf0000 | ||
1917 | #define BRXSEARCHRANGE_GI2_EARLY 0x700000 | ||
1918 | #define BRXFRAME_FUARD_COUNTER_L 0x3800000 | ||
1919 | #define BRXSGI_GUARD_L 0xc000000 | ||
1920 | #define BRXSGI_SEARCH_L 0x30000000 | ||
1921 | #define BRXSGI_TH 0xc0000000 | ||
1922 | #define BDFSCNT0 0xff | ||
1923 | #define BDFSCNT1 0xff00 | ||
1924 | #define BDFSFLAG 0xf0000 | ||
1925 | #define BMF_WEIGHT_SUM 0x300000 | ||
1926 | #define BMINIDX_TH 0x7f000000 | ||
1927 | #define BDAFORMAT 0x40000 | ||
1928 | #define BTXCH_EMU_ENABLE 0x01000000 | ||
1929 | #define BTRSW_ISOLATION_A 0x7f | ||
1930 | #define BTRSW_ISOLATION_B 0x7f00 | ||
1931 | #define BTRSW_ISOLATION_C 0x7f0000 | ||
1932 | #define BTRSW_ISOLATION_D 0x7f000000 | ||
1933 | #define BEXT_LNA_GAIN 0x7c00 | ||
1934 | |||
1935 | #define BSTBC_EN 0x4 | ||
1936 | #define BANTENNA_MAPPING 0x10 | ||
1937 | #define BNSS 0x20 | ||
1938 | #define BCFO_ANTSUM_ID 0x200 | ||
1939 | #define BPHY_COUNTER_RESET 0x8000000 | ||
1940 | #define BCFO_REPORT_GET 0x4000000 | ||
1941 | #define BOFDM_CONTINUE_TX 0x10000000 | ||
1942 | #define BOFDM_SINGLE_CARRIER 0x20000000 | ||
1943 | #define BOFDM_SINGLE_TONE 0x40000000 | ||
1944 | #define BHT_DETECT 0x100 | ||
1945 | #define BCFOEN 0x10000 | ||
1946 | #define BCFOVALUE 0xfff00000 | ||
1947 | #define BSIGTONE_RE 0x3f | ||
1948 | #define BSIGTONE_IM 0x7f00 | ||
1949 | #define BCOUNTER_CCA 0xffff | ||
1950 | #define BCOUNTER_PARITYFAIL 0xffff0000 | ||
1951 | #define BCOUNTER_RATEILLEGAL 0xffff | ||
1952 | #define BCOUNTER_CRC8FAIL 0xffff0000 | ||
1953 | #define BCOUNTER_MCSNOSUPPORT 0xffff | ||
1954 | #define BCOUNTER_FASTSYNC 0xffff | ||
1955 | #define BSHORTCFO 0xfff | ||
1956 | #define BSHORTCFOT_LENGTH 12 | ||
1957 | #define BSHORTCFOF_LENGTH 11 | ||
1958 | #define BLONGCFO 0x7ff | ||
1959 | #define BLONGCFOT_LENGTH 11 | ||
1960 | #define BLONGCFOF_LENGTH 11 | ||
1961 | #define BTAILCFO 0x1fff | ||
1962 | #define BTAILCFOT_LENGTH 13 | ||
1963 | #define BTAILCFOF_LENGTH 12 | ||
1964 | #define BNOISE_EN_PWDB 0xffff | ||
1965 | #define BCC_POWER_DB 0xffff0000 | ||
1966 | #define BMOISE_PWDB 0xffff | ||
1967 | #define BPOWERMEAST_LENGTH 10 | ||
1968 | #define BPOWERMEASF_LENGTH 3 | ||
1969 | #define BRX_HT_BW 0x1 | ||
1970 | #define BRXSC 0x6 | ||
1971 | #define BRX_HT 0x8 | ||
1972 | #define BNB_INTF_DET_ON 0x1 | ||
1973 | #define BINTF_WIN_LEN_CFG 0x30 | ||
1974 | #define BNB_INTF_TH_CFG 0x1c0 | ||
1975 | #define BRFGAIN 0x3f | ||
1976 | #define BTABLESEL 0x40 | ||
1977 | #define BTRSW 0x80 | ||
1978 | #define BRXSNR_A 0xff | ||
1979 | #define BRXSNR_B 0xff00 | ||
1980 | #define BRXSNR_C 0xff0000 | ||
1981 | #define BRXSNR_D 0xff000000 | ||
1982 | #define BSNR_EVMT_LENGTH 8 | ||
1983 | #define BSNR_EVMF_LENGTH 1 | ||
1984 | #define BCSI1ST 0xff | ||
1985 | #define BCSI2ND 0xff00 | ||
1986 | #define BRXEVM1ST 0xff0000 | ||
1987 | #define BRXEVM2ND 0xff000000 | ||
1988 | #define BSIGEVM 0xff | ||
1989 | #define BPWDB 0xff00 | ||
1990 | #define BSGIEN 0x10000 | ||
1991 | |||
1992 | #define BSFACTOR_QMA1 0xf | ||
1993 | #define BSFACTOR_QMA2 0xf0 | ||
1994 | #define BSFACTOR_QMA3 0xf00 | ||
1995 | #define BSFACTOR_QMA4 0xf000 | ||
1996 | #define BSFACTOR_QMA5 0xf0000 | ||
1997 | #define BSFACTOR_QMA6 0xf0000 | ||
1998 | #define BSFACTOR_QMA7 0xf00000 | ||
1999 | #define BSFACTOR_QMA8 0xf000000 | ||
2000 | #define BSFACTOR_QMA9 0xf0000000 | ||
2001 | #define BCSI_SCHEME 0x100000 | ||
2002 | |||
2003 | #define BNOISE_LVL_TOP_SET 0x3 | ||
2004 | #define BCHSMOOTH 0x4 | ||
2005 | #define BCHSMOOTH_CFG1 0x38 | ||
2006 | #define BCHSMOOTH_CFG2 0x1c0 | ||
2007 | #define BCHSMOOTH_CFG3 0xe00 | ||
2008 | #define BCHSMOOTH_CFG4 0x7000 | ||
2009 | #define BMRCMODE 0x800000 | ||
2010 | #define BTHEVMCFG 0x7000000 | ||
2011 | |||
2012 | #define BLOOP_FIT_TYPE 0x1 | ||
2013 | #define BUPD_CFO 0x40 | ||
2014 | #define BUPD_CFO_OFFDATA 0x80 | ||
2015 | #define BADV_UPD_CFO 0x100 | ||
2016 | #define BADV_TIME_CTRL 0x800 | ||
2017 | #define BUPD_CLKO 0x1000 | ||
2018 | #define BFC 0x6000 | ||
2019 | #define BTRACKING_MODE 0x8000 | ||
2020 | #define BPHCMP_ENABLE 0x10000 | ||
2021 | #define BUPD_CLKO_LTF 0x20000 | ||
2022 | #define BCOM_CH_CFO 0x40000 | ||
2023 | #define BCSI_ESTI_MODE 0x80000 | ||
2024 | #define BADV_UPD_EQZ 0x100000 | ||
2025 | #define BUCHCFG 0x7000000 | ||
2026 | #define BUPDEQZ 0x8000000 | ||
2027 | |||
2028 | #define BRX_PESUDO_NOISE_ON 0x20000000 | ||
2029 | #define BRX_PESUDO_NOISE_A 0xff | ||
2030 | #define BRX_PESUDO_NOISE_B 0xff00 | ||
2031 | #define BRX_PESUDO_NOISE_C 0xff0000 | ||
2032 | #define BRX_PESUDO_NOISE_D 0xff000000 | ||
2033 | #define BRX_PESUDO_NOISESTATE_A 0xffff | ||
2034 | #define BRX_PESUDO_NOISESTATE_B 0xffff0000 | ||
2035 | #define BRX_PESUDO_NOISESTATE_C 0xffff | ||
2036 | #define BRX_PESUDO_NOISESTATE_D 0xffff0000 | ||
2037 | |||
2038 | #define BZEBRA1_HSSIENABLE 0x8 | ||
2039 | #define BZEBRA1_TRXCONTROL 0xc00 | ||
2040 | #define BZEBRA1_TRXGAINSETTING 0x07f | ||
2041 | #define BZEBRA1_RXCOUNTER 0xc00 | ||
2042 | #define BZEBRA1_TXCHANGEPUMP 0x38 | ||
2043 | #define BZEBRA1_RXCHANGEPUMP 0x7 | ||
2044 | #define BZEBRA1_CHANNEL_NUM 0xf80 | ||
2045 | #define BZEBRA1_TXLPFBW 0x400 | ||
2046 | #define BZEBRA1_RXLPFBW 0x600 | ||
2047 | |||
2048 | #define BRTL8256REG_MODE_CTRL1 0x100 | ||
2049 | #define BRTL8256REG_MODE_CTRL0 0x40 | ||
2050 | #define BRTL8256REG_TXLPFBW 0x18 | ||
2051 | #define BRTL8256REG_RXLPFBW 0x600 | ||
2052 | |||
2053 | #define BRTL8258_TXLPFBW 0xc | ||
2054 | #define BRTL8258_RXLPFBW 0xc00 | ||
2055 | #define BRTL8258_RSSILPFBW 0xc0 | ||
2056 | |||
2057 | #define BBYTE0 0x1 | ||
2058 | #define BBYTE1 0x2 | ||
2059 | #define BBYTE2 0x4 | ||
2060 | #define BBYTE3 0x8 | ||
2061 | #define BWORD0 0x3 | ||
2062 | #define BWORD1 0xc | ||
2063 | #define BWORD 0xf | ||
2064 | |||
2065 | #define MASKBYTE0 0xff | ||
2066 | #define MASKBYTE1 0xff00 | ||
2067 | #define MASKBYTE2 0xff0000 | ||
2068 | #define MASKBYTE3 0xff000000 | ||
2069 | #define MASKHWORD 0xffff0000 | ||
2070 | #define MASKLWORD 0x0000ffff | ||
2071 | #define MASKDWORD 0xffffffff | ||
2072 | #define MASK12BITS 0xfff | ||
2073 | #define MASKH4BITS 0xf0000000 | ||
2074 | #define MASKOFDM_D 0xffc00000 | ||
2075 | #define MASKCCK 0x3f3f3f3f | ||
2076 | |||
2077 | #define MASK4BITS 0x0f | ||
2078 | #define MASK20BITS 0xfffff | ||
2079 | #define RFREG_OFFSET_MASK 0xfffff | ||
2080 | |||
2081 | #define BENABLE 0x1 | ||
2082 | #define BDISABLE 0x0 | ||
2083 | |||
2084 | #define LEFT_ANTENNA 0x0 | ||
2085 | #define RIGHT_ANTENNA 0x1 | ||
2086 | |||
2087 | #define TCHECK_TXSTATUS 500 | ||
2088 | #define TUPDATE_RXCOUNTER 100 | ||
2089 | |||
2090 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/rf.c b/drivers/net/wireless/rtlwifi/rtl8192ce/rf.c new file mode 100644 index 000000000000..90d0f2cf3b27 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/rf.c | |||
@@ -0,0 +1,523 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "reg.h" | ||
32 | #include "def.h" | ||
33 | #include "phy.h" | ||
34 | #include "rf.h" | ||
35 | #include "dm.h" | ||
36 | |||
37 | static bool _rtl92ce_phy_rf6052_config_parafile(struct ieee80211_hw *hw); | ||
38 | |||
39 | void rtl92ce_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, u8 bandwidth) | ||
40 | { | ||
41 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
42 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
43 | |||
44 | switch (bandwidth) { | ||
45 | case HT_CHANNEL_WIDTH_20: | ||
46 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
47 | 0xfffff3ff) | 0x0400); | ||
48 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
49 | rtlphy->rfreg_chnlval[0]); | ||
50 | break; | ||
51 | case HT_CHANNEL_WIDTH_20_40: | ||
52 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
53 | 0xfffff3ff)); | ||
54 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
55 | rtlphy->rfreg_chnlval[0]); | ||
56 | break; | ||
57 | default: | ||
58 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
59 | ("unknown bandwidth: %#X\n", bandwidth)); | ||
60 | break; | ||
61 | } | ||
62 | } | ||
63 | |||
64 | void rtl92ce_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
65 | u8 *ppowerlevel) | ||
66 | { | ||
67 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
68 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
69 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
70 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
71 | u32 tx_agc[2] = {0, 0}, tmpval; | ||
72 | bool turbo_scanoff = false; | ||
73 | u8 idx1, idx2; | ||
74 | u8 *ptr; | ||
75 | |||
76 | if (rtlefuse->eeprom_regulatory != 0) | ||
77 | turbo_scanoff = true; | ||
78 | |||
79 | if (mac->act_scanning == true) { | ||
80 | tx_agc[RF90_PATH_A] = 0x3f3f3f3f; | ||
81 | tx_agc[RF90_PATH_B] = 0x3f3f3f3f; | ||
82 | |||
83 | if (turbo_scanoff) { | ||
84 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
85 | tx_agc[idx1] = ppowerlevel[idx1] | | ||
86 | (ppowerlevel[idx1] << 8) | | ||
87 | (ppowerlevel[idx1] << 16) | | ||
88 | (ppowerlevel[idx1] << 24); | ||
89 | } | ||
90 | } | ||
91 | } else { | ||
92 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
93 | tx_agc[idx1] = ppowerlevel[idx1] | | ||
94 | (ppowerlevel[idx1] << 8) | | ||
95 | (ppowerlevel[idx1] << 16) | | ||
96 | (ppowerlevel[idx1] << 24); | ||
97 | } | ||
98 | |||
99 | if (rtlefuse->eeprom_regulatory == 0) { | ||
100 | tmpval = | ||
101 | (rtlphy->mcs_txpwrlevel_origoffset[0][6]) + | ||
102 | (rtlphy->mcs_txpwrlevel_origoffset[0][7] << | ||
103 | 8); | ||
104 | tx_agc[RF90_PATH_A] += tmpval; | ||
105 | |||
106 | tmpval = (rtlphy->mcs_txpwrlevel_origoffset[0][14]) + | ||
107 | (rtlphy->mcs_txpwrlevel_origoffset[0][15] << | ||
108 | 24); | ||
109 | tx_agc[RF90_PATH_B] += tmpval; | ||
110 | } | ||
111 | } | ||
112 | |||
113 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
114 | ptr = (u8 *) (&(tx_agc[idx1])); | ||
115 | for (idx2 = 0; idx2 < 4; idx2++) { | ||
116 | if (*ptr > RF6052_MAX_TX_PWR) | ||
117 | *ptr = RF6052_MAX_TX_PWR; | ||
118 | ptr++; | ||
119 | } | ||
120 | } | ||
121 | |||
122 | tmpval = tx_agc[RF90_PATH_A] & 0xff; | ||
123 | rtl_set_bbreg(hw, RTXAGC_A_CCK1_MCS32, MASKBYTE1, tmpval); | ||
124 | |||
125 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
126 | ("CCK PWR 1M (rf-A) = 0x%x (reg 0x%x)\n", tmpval, | ||
127 | RTXAGC_A_CCK1_MCS32)); | ||
128 | |||
129 | tmpval = tx_agc[RF90_PATH_A] >> 8; | ||
130 | |||
131 | tmpval = tmpval & 0xff00ffff; | ||
132 | |||
133 | rtl_set_bbreg(hw, RTXAGC_B_CCK11_A_CCK2_11, 0xffffff00, tmpval); | ||
134 | |||
135 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
136 | ("CCK PWR 2~11M (rf-A) = 0x%x (reg 0x%x)\n", tmpval, | ||
137 | RTXAGC_B_CCK11_A_CCK2_11)); | ||
138 | |||
139 | tmpval = tx_agc[RF90_PATH_B] >> 24; | ||
140 | rtl_set_bbreg(hw, RTXAGC_B_CCK11_A_CCK2_11, MASKBYTE0, tmpval); | ||
141 | |||
142 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
143 | ("CCK PWR 11M (rf-B) = 0x%x (reg 0x%x)\n", tmpval, | ||
144 | RTXAGC_B_CCK11_A_CCK2_11)); | ||
145 | |||
146 | tmpval = tx_agc[RF90_PATH_B] & 0x00ffffff; | ||
147 | rtl_set_bbreg(hw, RTXAGC_B_CCK1_55_MCS32, 0xffffff00, tmpval); | ||
148 | |||
149 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
150 | ("CCK PWR 1~5.5M (rf-B) = 0x%x (reg 0x%x)\n", tmpval, | ||
151 | RTXAGC_B_CCK1_55_MCS32)); | ||
152 | } | ||
153 | |||
154 | static void rtl92c_phy_get_power_base(struct ieee80211_hw *hw, | ||
155 | u8 *ppowerlevel, u8 channel, | ||
156 | u32 *ofdmbase, u32 *mcsbase) | ||
157 | { | ||
158 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
159 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
160 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
161 | u32 powerBase0, powerBase1; | ||
162 | u8 legacy_pwrdiff, ht20_pwrdiff; | ||
163 | u8 i, powerlevel[2]; | ||
164 | |||
165 | for (i = 0; i < 2; i++) { | ||
166 | powerlevel[i] = ppowerlevel[i]; | ||
167 | legacy_pwrdiff = rtlefuse->txpwr_legacyhtdiff[i][channel - 1]; | ||
168 | powerBase0 = powerlevel[i] + legacy_pwrdiff; | ||
169 | |||
170 | powerBase0 = (powerBase0 << 24) | (powerBase0 << 16) | | ||
171 | (powerBase0 << 8) | powerBase0; | ||
172 | *(ofdmbase + i) = powerBase0; | ||
173 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
174 | (" [OFDM power base index rf(%c) = 0x%x]\n", | ||
175 | ((i == 0) ? 'A' : 'B'), *(ofdmbase + i))); | ||
176 | } | ||
177 | |||
178 | for (i = 0; i < 2; i++) { | ||
179 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20) { | ||
180 | ht20_pwrdiff = rtlefuse->txpwr_ht20diff[i][channel - 1]; | ||
181 | powerlevel[i] += ht20_pwrdiff; | ||
182 | } | ||
183 | powerBase1 = powerlevel[i]; | ||
184 | powerBase1 = (powerBase1 << 24) | | ||
185 | (powerBase1 << 16) | (powerBase1 << 8) | powerBase1; | ||
186 | |||
187 | *(mcsbase + i) = powerBase1; | ||
188 | |||
189 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
190 | (" [MCS power base index rf(%c) = 0x%x]\n", | ||
191 | ((i == 0) ? 'A' : 'B'), *(mcsbase + i))); | ||
192 | } | ||
193 | } | ||
194 | |||
195 | static void _rtl92c_get_txpower_writeval_by_regulatory(struct ieee80211_hw *hw, | ||
196 | u8 channel, u8 index, | ||
197 | u32 *powerBase0, | ||
198 | u32 *powerBase1, | ||
199 | u32 *p_outwriteval) | ||
200 | { | ||
201 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
202 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
203 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
204 | u8 i, chnlgroup = 0, pwr_diff_limit[4]; | ||
205 | u32 writeVal, customer_limit, rf; | ||
206 | |||
207 | for (rf = 0; rf < 2; rf++) { | ||
208 | switch (rtlefuse->eeprom_regulatory) { | ||
209 | case 0: | ||
210 | chnlgroup = 0; | ||
211 | |||
212 | writeVal = | ||
213 | rtlphy->mcs_txpwrlevel_origoffset[chnlgroup][index + | ||
214 | (rf ? 8 : 0)] | ||
215 | + ((index < 2) ? powerBase0[rf] : powerBase1[rf]); | ||
216 | |||
217 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
218 | ("RTK better performance, " | ||
219 | "writeVal(%c) = 0x%x\n", | ||
220 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
221 | break; | ||
222 | case 1: | ||
223 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
224 | writeVal = ((index < 2) ? powerBase0[rf] : | ||
225 | powerBase1[rf]); | ||
226 | |||
227 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
228 | ("Realtek regulatory, 40MHz, " | ||
229 | "writeVal(%c) = 0x%x\n", | ||
230 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
231 | } else { | ||
232 | if (rtlphy->pwrgroup_cnt == 1) | ||
233 | chnlgroup = 0; | ||
234 | if (rtlphy->pwrgroup_cnt >= 3) { | ||
235 | if (channel <= 3) | ||
236 | chnlgroup = 0; | ||
237 | else if (channel >= 4 && channel <= 9) | ||
238 | chnlgroup = 1; | ||
239 | else if (channel > 9) | ||
240 | chnlgroup = 2; | ||
241 | if (rtlphy->pwrgroup_cnt == 4) | ||
242 | chnlgroup++; | ||
243 | } | ||
244 | |||
245 | writeVal = | ||
246 | rtlphy->mcs_txpwrlevel_origoffset[chnlgroup] | ||
247 | [index + (rf ? 8 : 0)] + ((index < 2) ? | ||
248 | powerBase0[rf] : | ||
249 | powerBase1[rf]); | ||
250 | |||
251 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
252 | ("Realtek regulatory, 20MHz, " | ||
253 | "writeVal(%c) = 0x%x\n", | ||
254 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
255 | } | ||
256 | break; | ||
257 | case 2: | ||
258 | writeVal = | ||
259 | ((index < 2) ? powerBase0[rf] : powerBase1[rf]); | ||
260 | |||
261 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
262 | ("Better regulatory, " | ||
263 | "writeVal(%c) = 0x%x\n", | ||
264 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
265 | break; | ||
266 | case 3: | ||
267 | chnlgroup = 0; | ||
268 | |||
269 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
270 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
271 | ("customer's limit, 40MHz " | ||
272 | "rf(%c) = 0x%x\n", | ||
273 | ((rf == 0) ? 'A' : 'B'), | ||
274 | rtlefuse->pwrgroup_ht40[rf][channel - | ||
275 | 1])); | ||
276 | } else { | ||
277 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
278 | ("customer's limit, 20MHz " | ||
279 | "rf(%c) = 0x%x\n", | ||
280 | ((rf == 0) ? 'A' : 'B'), | ||
281 | rtlefuse->pwrgroup_ht20[rf][channel - | ||
282 | 1])); | ||
283 | } | ||
284 | for (i = 0; i < 4; i++) { | ||
285 | pwr_diff_limit[i] = | ||
286 | (u8) ((rtlphy->mcs_txpwrlevel_origoffset | ||
287 | [chnlgroup][index + | ||
288 | (rf ? 8 : 0)] & (0x7f << (i * 8))) >> | ||
289 | (i * 8)); | ||
290 | |||
291 | if (rtlphy->current_chan_bw == | ||
292 | HT_CHANNEL_WIDTH_20_40) { | ||
293 | if (pwr_diff_limit[i] > | ||
294 | rtlefuse-> | ||
295 | pwrgroup_ht40[rf][channel - 1]) | ||
296 | pwr_diff_limit[i] = | ||
297 | rtlefuse->pwrgroup_ht40[rf] | ||
298 | [channel - 1]; | ||
299 | } else { | ||
300 | if (pwr_diff_limit[i] > | ||
301 | rtlefuse-> | ||
302 | pwrgroup_ht20[rf][channel - 1]) | ||
303 | pwr_diff_limit[i] = | ||
304 | rtlefuse->pwrgroup_ht20[rf] | ||
305 | [channel - 1]; | ||
306 | } | ||
307 | } | ||
308 | |||
309 | customer_limit = (pwr_diff_limit[3] << 24) | | ||
310 | (pwr_diff_limit[2] << 16) | | ||
311 | (pwr_diff_limit[1] << 8) | (pwr_diff_limit[0]); | ||
312 | |||
313 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
314 | ("Customer's limit rf(%c) = 0x%x\n", | ||
315 | ((rf == 0) ? 'A' : 'B'), customer_limit)); | ||
316 | |||
317 | writeVal = customer_limit + | ||
318 | ((index < 2) ? powerBase0[rf] : powerBase1[rf]); | ||
319 | |||
320 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
321 | ("Customer, writeVal rf(%c)= 0x%x\n", | ||
322 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
323 | break; | ||
324 | default: | ||
325 | chnlgroup = 0; | ||
326 | writeVal = | ||
327 | rtlphy->mcs_txpwrlevel_origoffset[chnlgroup] | ||
328 | [index + (rf ? 8 : 0)] | ||
329 | + ((index < 2) ? powerBase0[rf] : powerBase1[rf]); | ||
330 | |||
331 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
332 | ("RTK better performance, writeVal " | ||
333 | "rf(%c) = 0x%x\n", | ||
334 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
335 | break; | ||
336 | } | ||
337 | |||
338 | if (rtlpriv->dm.dynamic_txhighpower_lvl == TXHIGHPWRLEVEL_BT1) | ||
339 | writeVal = writeVal - 0x06060606; | ||
340 | else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
341 | TXHIGHPWRLEVEL_BT2) | ||
342 | writeVal = writeVal - 0x0c0c0c0c; | ||
343 | *(p_outwriteval + rf) = writeVal; | ||
344 | } | ||
345 | } | ||
346 | |||
347 | static void _rtl92c_write_ofdm_power_reg(struct ieee80211_hw *hw, | ||
348 | u8 index, u32 *pValue) | ||
349 | { | ||
350 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
351 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
352 | |||
353 | u16 regoffset_a[6] = { | ||
354 | RTXAGC_A_RATE18_06, RTXAGC_A_RATE54_24, | ||
355 | RTXAGC_A_MCS03_MCS00, RTXAGC_A_MCS07_MCS04, | ||
356 | RTXAGC_A_MCS11_MCS08, RTXAGC_A_MCS15_MCS12 | ||
357 | }; | ||
358 | u16 regoffset_b[6] = { | ||
359 | RTXAGC_B_RATE18_06, RTXAGC_B_RATE54_24, | ||
360 | RTXAGC_B_MCS03_MCS00, RTXAGC_B_MCS07_MCS04, | ||
361 | RTXAGC_B_MCS11_MCS08, RTXAGC_B_MCS15_MCS12 | ||
362 | }; | ||
363 | u8 i, rf, pwr_val[4]; | ||
364 | u32 writeVal; | ||
365 | u16 regoffset; | ||
366 | |||
367 | for (rf = 0; rf < 2; rf++) { | ||
368 | writeVal = pValue[rf]; | ||
369 | for (i = 0; i < 4; i++) { | ||
370 | pwr_val[i] = (u8) ((writeVal & (0x7f << | ||
371 | (i * 8))) >> (i * 8)); | ||
372 | |||
373 | if (pwr_val[i] > RF6052_MAX_TX_PWR) | ||
374 | pwr_val[i] = RF6052_MAX_TX_PWR; | ||
375 | } | ||
376 | writeVal = (pwr_val[3] << 24) | (pwr_val[2] << 16) | | ||
377 | (pwr_val[1] << 8) | pwr_val[0]; | ||
378 | |||
379 | if (rf == 0) | ||
380 | regoffset = regoffset_a[index]; | ||
381 | else | ||
382 | regoffset = regoffset_b[index]; | ||
383 | rtl_set_bbreg(hw, regoffset, MASKDWORD, writeVal); | ||
384 | |||
385 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
386 | ("Set 0x%x = %08x\n", regoffset, writeVal)); | ||
387 | |||
388 | if (((get_rf_type(rtlphy) == RF_2T2R) && | ||
389 | (regoffset == RTXAGC_A_MCS15_MCS12 || | ||
390 | regoffset == RTXAGC_B_MCS15_MCS12)) || | ||
391 | ((get_rf_type(rtlphy) != RF_2T2R) && | ||
392 | (regoffset == RTXAGC_A_MCS07_MCS04 || | ||
393 | regoffset == RTXAGC_B_MCS07_MCS04))) { | ||
394 | |||
395 | writeVal = pwr_val[3]; | ||
396 | if (regoffset == RTXAGC_A_MCS15_MCS12 || | ||
397 | regoffset == RTXAGC_A_MCS07_MCS04) | ||
398 | regoffset = 0xc90; | ||
399 | if (regoffset == RTXAGC_B_MCS15_MCS12 || | ||
400 | regoffset == RTXAGC_B_MCS07_MCS04) | ||
401 | regoffset = 0xc98; | ||
402 | |||
403 | for (i = 0; i < 3; i++) { | ||
404 | writeVal = (writeVal > 6) ? (writeVal - 6) : 0; | ||
405 | rtl_write_byte(rtlpriv, (u32) (regoffset + i), | ||
406 | (u8) writeVal); | ||
407 | } | ||
408 | } | ||
409 | } | ||
410 | } | ||
411 | |||
412 | void rtl92ce_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
413 | u8 *ppowerlevel, u8 channel) | ||
414 | { | ||
415 | u32 writeVal[2], powerBase0[2], powerBase1[2]; | ||
416 | u8 index; | ||
417 | |||
418 | rtl92c_phy_get_power_base(hw, ppowerlevel, | ||
419 | channel, &powerBase0[0], &powerBase1[0]); | ||
420 | |||
421 | for (index = 0; index < 6; index++) { | ||
422 | _rtl92c_get_txpower_writeval_by_regulatory(hw, | ||
423 | channel, index, | ||
424 | &powerBase0[0], | ||
425 | &powerBase1[0], | ||
426 | &writeVal[0]); | ||
427 | |||
428 | _rtl92c_write_ofdm_power_reg(hw, index, &writeVal[0]); | ||
429 | } | ||
430 | } | ||
431 | |||
432 | bool rtl92ce_phy_rf6052_config(struct ieee80211_hw *hw) | ||
433 | { | ||
434 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
435 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
436 | |||
437 | if (rtlphy->rf_type == RF_1T1R) | ||
438 | rtlphy->num_total_rfpath = 1; | ||
439 | else | ||
440 | rtlphy->num_total_rfpath = 2; | ||
441 | |||
442 | return _rtl92ce_phy_rf6052_config_parafile(hw); | ||
443 | |||
444 | } | ||
445 | |||
446 | static bool _rtl92ce_phy_rf6052_config_parafile(struct ieee80211_hw *hw) | ||
447 | { | ||
448 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
449 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
450 | u32 u4_regvalue = 0; | ||
451 | u8 rfpath; | ||
452 | bool rtstatus = true; | ||
453 | struct bb_reg_def *pphyreg; | ||
454 | |||
455 | for (rfpath = 0; rfpath < rtlphy->num_total_rfpath; rfpath++) { | ||
456 | |||
457 | pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
458 | |||
459 | switch (rfpath) { | ||
460 | case RF90_PATH_A: | ||
461 | case RF90_PATH_C: | ||
462 | u4_regvalue = rtl_get_bbreg(hw, pphyreg->rfintfs, | ||
463 | BRFSI_RFENV); | ||
464 | break; | ||
465 | case RF90_PATH_B: | ||
466 | case RF90_PATH_D: | ||
467 | u4_regvalue = rtl_get_bbreg(hw, pphyreg->rfintfs, | ||
468 | BRFSI_RFENV << 16); | ||
469 | break; | ||
470 | } | ||
471 | |||
472 | rtl_set_bbreg(hw, pphyreg->rfintfe, BRFSI_RFENV << 16, 0x1); | ||
473 | udelay(1); | ||
474 | |||
475 | rtl_set_bbreg(hw, pphyreg->rfintfo, BRFSI_RFENV, 0x1); | ||
476 | udelay(1); | ||
477 | |||
478 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, | ||
479 | B3WIREADDREAALENGTH, 0x0); | ||
480 | udelay(1); | ||
481 | |||
482 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, B3WIREDATALENGTH, 0x0); | ||
483 | udelay(1); | ||
484 | |||
485 | switch (rfpath) { | ||
486 | case RF90_PATH_A: | ||
487 | rtstatus = rtl92c_phy_config_rf_with_headerfile(hw, | ||
488 | (enum radio_path)rfpath); | ||
489 | break; | ||
490 | case RF90_PATH_B: | ||
491 | rtstatus = rtl92c_phy_config_rf_with_headerfile(hw, | ||
492 | (enum radio_path)rfpath); | ||
493 | break; | ||
494 | case RF90_PATH_C: | ||
495 | break; | ||
496 | case RF90_PATH_D: | ||
497 | break; | ||
498 | } | ||
499 | |||
500 | switch (rfpath) { | ||
501 | case RF90_PATH_A: | ||
502 | case RF90_PATH_C: | ||
503 | rtl_set_bbreg(hw, pphyreg->rfintfs, | ||
504 | BRFSI_RFENV, u4_regvalue); | ||
505 | break; | ||
506 | case RF90_PATH_B: | ||
507 | case RF90_PATH_D: | ||
508 | rtl_set_bbreg(hw, pphyreg->rfintfs, | ||
509 | BRFSI_RFENV << 16, u4_regvalue); | ||
510 | break; | ||
511 | } | ||
512 | |||
513 | if (rtstatus != true) { | ||
514 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
515 | ("Radio[%d] Fail!!", rfpath)); | ||
516 | return false; | ||
517 | } | ||
518 | |||
519 | } | ||
520 | |||
521 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("<---\n")); | ||
522 | return rtstatus; | ||
523 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/rf.h b/drivers/net/wireless/rtlwifi/rtl8192ce/rf.h new file mode 100644 index 000000000000..39ff03685986 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/rf.h | |||
@@ -0,0 +1,44 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_RF_H__ | ||
31 | #define __RTL92C_RF_H__ | ||
32 | |||
33 | #define RF6052_MAX_TX_PWR 0x3F | ||
34 | #define RF6052_MAX_REG 0x3F | ||
35 | #define RF6052_MAX_PATH 2 | ||
36 | |||
37 | extern void rtl92ce_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, | ||
38 | u8 bandwidth); | ||
39 | extern void rtl92ce_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
40 | u8 *ppowerlevel); | ||
41 | extern void rtl92ce_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
42 | u8 *ppowerlevel, u8 channel); | ||
43 | extern bool rtl92ce_phy_rf6052_config(struct ieee80211_hw *hw); | ||
44 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/sw.c b/drivers/net/wireless/rtlwifi/rtl8192ce/sw.c new file mode 100644 index 000000000000..373dc78af1dc --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/sw.c | |||
@@ -0,0 +1,394 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include <linux/vmalloc.h> | ||
31 | |||
32 | #include "../wifi.h" | ||
33 | #include "../core.h" | ||
34 | #include "../pci.h" | ||
35 | #include "reg.h" | ||
36 | #include "def.h" | ||
37 | #include "phy.h" | ||
38 | #include "dm.h" | ||
39 | #include "hw.h" | ||
40 | #include "rf.h" | ||
41 | #include "sw.h" | ||
42 | #include "trx.h" | ||
43 | #include "led.h" | ||
44 | |||
45 | static void rtl92c_init_aspm_vars(struct ieee80211_hw *hw) | ||
46 | { | ||
47 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
48 | |||
49 | /*close ASPM for AMD defaultly */ | ||
50 | rtlpci->const_amdpci_aspm = 0; | ||
51 | |||
52 | /* | ||
53 | * ASPM PS mode. | ||
54 | * 0 - Disable ASPM, | ||
55 | * 1 - Enable ASPM without Clock Req, | ||
56 | * 2 - Enable ASPM with Clock Req, | ||
57 | * 3 - Alwyas Enable ASPM with Clock Req, | ||
58 | * 4 - Always Enable ASPM without Clock Req. | ||
59 | * set defult to RTL8192CE:3 RTL8192E:2 | ||
60 | * */ | ||
61 | rtlpci->const_pci_aspm = 3; | ||
62 | |||
63 | /*Setting for PCI-E device */ | ||
64 | rtlpci->const_devicepci_aspm_setting = 0x03; | ||
65 | |||
66 | /*Setting for PCI-E bridge */ | ||
67 | rtlpci->const_hostpci_aspm_setting = 0x02; | ||
68 | |||
69 | /* | ||
70 | * In Hw/Sw Radio Off situation. | ||
71 | * 0 - Default, | ||
72 | * 1 - From ASPM setting without low Mac Pwr, | ||
73 | * 2 - From ASPM setting with low Mac Pwr, | ||
74 | * 3 - Bus D3 | ||
75 | * set default to RTL8192CE:0 RTL8192SE:2 | ||
76 | */ | ||
77 | rtlpci->const_hwsw_rfoff_d3 = 0; | ||
78 | |||
79 | /* | ||
80 | * This setting works for those device with | ||
81 | * backdoor ASPM setting such as EPHY setting. | ||
82 | * 0 - Not support ASPM, | ||
83 | * 1 - Support ASPM, | ||
84 | * 2 - According to chipset. | ||
85 | */ | ||
86 | rtlpci->const_support_pciaspm = 1; | ||
87 | } | ||
88 | |||
89 | int rtl92c_init_sw_vars(struct ieee80211_hw *hw) | ||
90 | { | ||
91 | int err; | ||
92 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
93 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
94 | const struct firmware *firmware; | ||
95 | |||
96 | rtl8192ce_bt_reg_init(hw); | ||
97 | |||
98 | rtlpriv->dm.dm_initialgain_enable = 1; | ||
99 | rtlpriv->dm.dm_flag = 0; | ||
100 | rtlpriv->dm.disable_framebursting = 0; | ||
101 | rtlpriv->dm.thermalvalue = 0; | ||
102 | rtlpci->transmit_config = CFENDFORM | BIT(12) | BIT(13); | ||
103 | |||
104 | /* compatible 5G band 88ce just 2.4G band & smsp */ | ||
105 | rtlpriv->rtlhal.current_bandtype = BAND_ON_2_4G; | ||
106 | rtlpriv->rtlhal.bandset = BAND_ON_2_4G; | ||
107 | rtlpriv->rtlhal.macphymode = SINGLEMAC_SINGLEPHY; | ||
108 | |||
109 | rtlpci->receive_config = (RCR_APPFCS | | ||
110 | RCR_AMF | | ||
111 | RCR_ADF | | ||
112 | RCR_APP_MIC | | ||
113 | RCR_APP_ICV | | ||
114 | RCR_AICV | | ||
115 | RCR_ACRC32 | | ||
116 | RCR_AB | | ||
117 | RCR_AM | | ||
118 | RCR_APM | | ||
119 | RCR_APP_PHYST_RXFF | RCR_HTC_LOC_CTRL | 0); | ||
120 | |||
121 | rtlpci->irq_mask[0] = | ||
122 | (u32) (IMR_ROK | | ||
123 | IMR_VODOK | | ||
124 | IMR_VIDOK | | ||
125 | IMR_BEDOK | | ||
126 | IMR_BKDOK | | ||
127 | IMR_MGNTDOK | | ||
128 | IMR_HIGHDOK | IMR_BDOK | IMR_RDU | IMR_RXFOVW | 0); | ||
129 | |||
130 | rtlpci->irq_mask[1] = (u32) (IMR_CPWM | IMR_C2HCMD | 0); | ||
131 | |||
132 | /* for LPS & IPS */ | ||
133 | rtlpriv->psc.inactiveps = rtlpriv->cfg->mod_params->inactiveps; | ||
134 | rtlpriv->psc.swctrl_lps = rtlpriv->cfg->mod_params->swctrl_lps; | ||
135 | rtlpriv->psc.fwctrl_lps = rtlpriv->cfg->mod_params->fwctrl_lps; | ||
136 | rtlpriv->psc.reg_fwctrl_lps = 3; | ||
137 | rtlpriv->psc.reg_max_lps_awakeintvl = 5; | ||
138 | /* for ASPM, you can close aspm through | ||
139 | * set const_support_pciaspm = 0 */ | ||
140 | rtl92c_init_aspm_vars(hw); | ||
141 | |||
142 | if (rtlpriv->psc.reg_fwctrl_lps == 1) | ||
143 | rtlpriv->psc.fwctrl_psmode = FW_PS_MIN_MODE; | ||
144 | else if (rtlpriv->psc.reg_fwctrl_lps == 2) | ||
145 | rtlpriv->psc.fwctrl_psmode = FW_PS_MAX_MODE; | ||
146 | else if (rtlpriv->psc.reg_fwctrl_lps == 3) | ||
147 | rtlpriv->psc.fwctrl_psmode = FW_PS_DTIM_MODE; | ||
148 | |||
149 | /* for firmware buf */ | ||
150 | rtlpriv->rtlhal.pfirmware = vzalloc(0x4000); | ||
151 | if (!rtlpriv->rtlhal.pfirmware) { | ||
152 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
153 | ("Can't alloc buffer for fw.\n")); | ||
154 | return 1; | ||
155 | } | ||
156 | |||
157 | /* request fw */ | ||
158 | err = request_firmware(&firmware, rtlpriv->cfg->fw_name, | ||
159 | rtlpriv->io.dev); | ||
160 | if (err) { | ||
161 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
162 | ("Failed to request firmware!\n")); | ||
163 | return 1; | ||
164 | } | ||
165 | if (firmware->size > 0x4000) { | ||
166 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
167 | ("Firmware is too big!\n")); | ||
168 | release_firmware(firmware); | ||
169 | return 1; | ||
170 | } | ||
171 | memcpy(rtlpriv->rtlhal.pfirmware, firmware->data, firmware->size); | ||
172 | rtlpriv->rtlhal.fwsize = firmware->size; | ||
173 | release_firmware(firmware); | ||
174 | |||
175 | return 0; | ||
176 | } | ||
177 | |||
178 | void rtl92c_deinit_sw_vars(struct ieee80211_hw *hw) | ||
179 | { | ||
180 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
181 | |||
182 | if (rtlpriv->rtlhal.pfirmware) { | ||
183 | vfree(rtlpriv->rtlhal.pfirmware); | ||
184 | rtlpriv->rtlhal.pfirmware = NULL; | ||
185 | } | ||
186 | } | ||
187 | |||
188 | static struct rtl_hal_ops rtl8192ce_hal_ops = { | ||
189 | .init_sw_vars = rtl92c_init_sw_vars, | ||
190 | .deinit_sw_vars = rtl92c_deinit_sw_vars, | ||
191 | .read_eeprom_info = rtl92ce_read_eeprom_info, | ||
192 | .interrupt_recognized = rtl92ce_interrupt_recognized, | ||
193 | .hw_init = rtl92ce_hw_init, | ||
194 | .hw_disable = rtl92ce_card_disable, | ||
195 | .hw_suspend = rtl92ce_suspend, | ||
196 | .hw_resume = rtl92ce_resume, | ||
197 | .enable_interrupt = rtl92ce_enable_interrupt, | ||
198 | .disable_interrupt = rtl92ce_disable_interrupt, | ||
199 | .set_network_type = rtl92ce_set_network_type, | ||
200 | .set_chk_bssid = rtl92ce_set_check_bssid, | ||
201 | .set_qos = rtl92ce_set_qos, | ||
202 | .set_bcn_reg = rtl92ce_set_beacon_related_registers, | ||
203 | .set_bcn_intv = rtl92ce_set_beacon_interval, | ||
204 | .update_interrupt_mask = rtl92ce_update_interrupt_mask, | ||
205 | .get_hw_reg = rtl92ce_get_hw_reg, | ||
206 | .set_hw_reg = rtl92ce_set_hw_reg, | ||
207 | .update_rate_tbl = rtl92ce_update_hal_rate_tbl, | ||
208 | .fill_tx_desc = rtl92ce_tx_fill_desc, | ||
209 | .fill_tx_cmddesc = rtl92ce_tx_fill_cmddesc, | ||
210 | .query_rx_desc = rtl92ce_rx_query_desc, | ||
211 | .set_channel_access = rtl92ce_update_channel_access_setting, | ||
212 | .radio_onoff_checking = rtl92ce_gpio_radio_on_off_checking, | ||
213 | .set_bw_mode = rtl92c_phy_set_bw_mode, | ||
214 | .switch_channel = rtl92c_phy_sw_chnl, | ||
215 | .dm_watchdog = rtl92c_dm_watchdog, | ||
216 | .scan_operation_backup = rtl92c_phy_scan_operation_backup, | ||
217 | .set_rf_power_state = rtl92c_phy_set_rf_power_state, | ||
218 | .led_control = rtl92ce_led_control, | ||
219 | .set_desc = rtl92ce_set_desc, | ||
220 | .get_desc = rtl92ce_get_desc, | ||
221 | .tx_polling = rtl92ce_tx_polling, | ||
222 | .enable_hw_sec = rtl92ce_enable_hw_security_config, | ||
223 | .set_key = rtl92ce_set_key, | ||
224 | .init_sw_leds = rtl92ce_init_sw_leds, | ||
225 | .get_bbreg = rtl92c_phy_query_bb_reg, | ||
226 | .set_bbreg = rtl92c_phy_set_bb_reg, | ||
227 | .set_rfreg = rtl92ce_phy_set_rf_reg, | ||
228 | .get_rfreg = rtl92c_phy_query_rf_reg, | ||
229 | .phy_rf6052_config = rtl92ce_phy_rf6052_config, | ||
230 | .phy_rf6052_set_cck_txpower = rtl92ce_phy_rf6052_set_cck_txpower, | ||
231 | .phy_rf6052_set_ofdm_txpower = rtl92ce_phy_rf6052_set_ofdm_txpower, | ||
232 | .config_bb_with_headerfile = _rtl92ce_phy_config_bb_with_headerfile, | ||
233 | .config_bb_with_pgheaderfile = _rtl92ce_phy_config_bb_with_pgheaderfile, | ||
234 | .phy_lc_calibrate = _rtl92ce_phy_lc_calibrate, | ||
235 | .phy_set_bw_mode_callback = rtl92ce_phy_set_bw_mode_callback, | ||
236 | .dm_dynamic_txpower = rtl92ce_dm_dynamic_txpower, | ||
237 | }; | ||
238 | |||
239 | static struct rtl_mod_params rtl92ce_mod_params = { | ||
240 | .sw_crypto = false, | ||
241 | .inactiveps = true, | ||
242 | .swctrl_lps = false, | ||
243 | .fwctrl_lps = true, | ||
244 | }; | ||
245 | |||
246 | static struct rtl_hal_cfg rtl92ce_hal_cfg = { | ||
247 | .bar_id = 2, | ||
248 | .write_readback = true, | ||
249 | .name = "rtl92c_pci", | ||
250 | .fw_name = "rtlwifi/rtl8192cfw.bin", | ||
251 | .ops = &rtl8192ce_hal_ops, | ||
252 | .mod_params = &rtl92ce_mod_params, | ||
253 | |||
254 | .maps[SYS_ISO_CTRL] = REG_SYS_ISO_CTRL, | ||
255 | .maps[SYS_FUNC_EN] = REG_SYS_FUNC_EN, | ||
256 | .maps[SYS_CLK] = REG_SYS_CLKR, | ||
257 | .maps[MAC_RCR_AM] = AM, | ||
258 | .maps[MAC_RCR_AB] = AB, | ||
259 | .maps[MAC_RCR_ACRC32] = ACRC32, | ||
260 | .maps[MAC_RCR_ACF] = ACF, | ||
261 | .maps[MAC_RCR_AAP] = AAP, | ||
262 | |||
263 | .maps[EFUSE_TEST] = REG_EFUSE_TEST, | ||
264 | .maps[EFUSE_CTRL] = REG_EFUSE_CTRL, | ||
265 | .maps[EFUSE_CLK] = 0, | ||
266 | .maps[EFUSE_CLK_CTRL] = REG_EFUSE_CTRL, | ||
267 | .maps[EFUSE_PWC_EV12V] = PWC_EV12V, | ||
268 | .maps[EFUSE_FEN_ELDR] = FEN_ELDR, | ||
269 | .maps[EFUSE_LOADER_CLK_EN] = LOADER_CLK_EN, | ||
270 | .maps[EFUSE_ANA8M] = EFUSE_ANA8M, | ||
271 | .maps[EFUSE_HWSET_MAX_SIZE] = HWSET_MAX_SIZE, | ||
272 | .maps[EFUSE_MAX_SECTION_MAP] = EFUSE_MAX_SECTION, | ||
273 | .maps[EFUSE_REAL_CONTENT_SIZE] = EFUSE_REAL_CONTENT_LEN, | ||
274 | |||
275 | .maps[RWCAM] = REG_CAMCMD, | ||
276 | .maps[WCAMI] = REG_CAMWRITE, | ||
277 | .maps[RCAMO] = REG_CAMREAD, | ||
278 | .maps[CAMDBG] = REG_CAMDBG, | ||
279 | .maps[SECR] = REG_SECCFG, | ||
280 | .maps[SEC_CAM_NONE] = CAM_NONE, | ||
281 | .maps[SEC_CAM_WEP40] = CAM_WEP40, | ||
282 | .maps[SEC_CAM_TKIP] = CAM_TKIP, | ||
283 | .maps[SEC_CAM_AES] = CAM_AES, | ||
284 | .maps[SEC_CAM_WEP104] = CAM_WEP104, | ||
285 | |||
286 | .maps[RTL_IMR_BCNDMAINT6] = IMR_BCNDMAINT6, | ||
287 | .maps[RTL_IMR_BCNDMAINT5] = IMR_BCNDMAINT5, | ||
288 | .maps[RTL_IMR_BCNDMAINT4] = IMR_BCNDMAINT4, | ||
289 | .maps[RTL_IMR_BCNDMAINT3] = IMR_BCNDMAINT3, | ||
290 | .maps[RTL_IMR_BCNDMAINT2] = IMR_BCNDMAINT2, | ||
291 | .maps[RTL_IMR_BCNDMAINT1] = IMR_BCNDMAINT1, | ||
292 | .maps[RTL_IMR_BCNDOK8] = IMR_BCNDOK8, | ||
293 | .maps[RTL_IMR_BCNDOK7] = IMR_BCNDOK7, | ||
294 | .maps[RTL_IMR_BCNDOK6] = IMR_BCNDOK6, | ||
295 | .maps[RTL_IMR_BCNDOK5] = IMR_BCNDOK5, | ||
296 | .maps[RTL_IMR_BCNDOK4] = IMR_BCNDOK4, | ||
297 | .maps[RTL_IMR_BCNDOK3] = IMR_BCNDOK3, | ||
298 | .maps[RTL_IMR_BCNDOK2] = IMR_BCNDOK2, | ||
299 | .maps[RTL_IMR_BCNDOK1] = IMR_BCNDOK1, | ||
300 | .maps[RTL_IMR_TIMEOUT2] = IMR_TIMEOUT2, | ||
301 | .maps[RTL_IMR_TIMEOUT1] = IMR_TIMEOUT1, | ||
302 | |||
303 | .maps[RTL_IMR_TXFOVW] = IMR_TXFOVW, | ||
304 | .maps[RTL_IMR_PSTIMEOUT] = IMR_PSTIMEOUT, | ||
305 | .maps[RTL_IMR_BcnInt] = IMR_BCNINT, | ||
306 | .maps[RTL_IMR_RXFOVW] = IMR_RXFOVW, | ||
307 | .maps[RTL_IMR_RDU] = IMR_RDU, | ||
308 | .maps[RTL_IMR_ATIMEND] = IMR_ATIMEND, | ||
309 | .maps[RTL_IMR_BDOK] = IMR_BDOK, | ||
310 | .maps[RTL_IMR_MGNTDOK] = IMR_MGNTDOK, | ||
311 | .maps[RTL_IMR_TBDER] = IMR_TBDER, | ||
312 | .maps[RTL_IMR_HIGHDOK] = IMR_HIGHDOK, | ||
313 | .maps[RTL_IMR_TBDOK] = IMR_TBDOK, | ||
314 | .maps[RTL_IMR_BKDOK] = IMR_BKDOK, | ||
315 | .maps[RTL_IMR_BEDOK] = IMR_BEDOK, | ||
316 | .maps[RTL_IMR_VIDOK] = IMR_VIDOK, | ||
317 | .maps[RTL_IMR_VODOK] = IMR_VODOK, | ||
318 | .maps[RTL_IMR_ROK] = IMR_ROK, | ||
319 | .maps[RTL_IBSS_INT_MASKS] = (IMR_BCNINT | IMR_TBDOK | IMR_TBDER), | ||
320 | |||
321 | .maps[RTL_RC_CCK_RATE1M] = DESC92C_RATE1M, | ||
322 | .maps[RTL_RC_CCK_RATE2M] = DESC92C_RATE2M, | ||
323 | .maps[RTL_RC_CCK_RATE5_5M] = DESC92C_RATE5_5M, | ||
324 | .maps[RTL_RC_CCK_RATE11M] = DESC92C_RATE11M, | ||
325 | .maps[RTL_RC_OFDM_RATE6M] = DESC92C_RATE6M, | ||
326 | .maps[RTL_RC_OFDM_RATE9M] = DESC92C_RATE9M, | ||
327 | .maps[RTL_RC_OFDM_RATE12M] = DESC92C_RATE12M, | ||
328 | .maps[RTL_RC_OFDM_RATE18M] = DESC92C_RATE18M, | ||
329 | .maps[RTL_RC_OFDM_RATE24M] = DESC92C_RATE24M, | ||
330 | .maps[RTL_RC_OFDM_RATE36M] = DESC92C_RATE36M, | ||
331 | .maps[RTL_RC_OFDM_RATE48M] = DESC92C_RATE48M, | ||
332 | .maps[RTL_RC_OFDM_RATE54M] = DESC92C_RATE54M, | ||
333 | |||
334 | .maps[RTL_RC_HT_RATEMCS7] = DESC92C_RATEMCS7, | ||
335 | .maps[RTL_RC_HT_RATEMCS15] = DESC92C_RATEMCS15, | ||
336 | }; | ||
337 | |||
338 | DEFINE_PCI_DEVICE_TABLE(rtl92ce_pci_ids) = { | ||
339 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8191, rtl92ce_hal_cfg)}, | ||
340 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8178, rtl92ce_hal_cfg)}, | ||
341 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8177, rtl92ce_hal_cfg)}, | ||
342 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8176, rtl92ce_hal_cfg)}, | ||
343 | {}, | ||
344 | }; | ||
345 | |||
346 | MODULE_DEVICE_TABLE(pci, rtl92ce_pci_ids); | ||
347 | |||
348 | MODULE_AUTHOR("lizhaoming <chaoming_li@realsil.com.cn>"); | ||
349 | MODULE_AUTHOR("Realtek WlanFAE <wlanfae@realtek.com>"); | ||
350 | MODULE_AUTHOR("Larry Finger <Larry.Finger@lwfinger.net>"); | ||
351 | MODULE_LICENSE("GPL"); | ||
352 | MODULE_DESCRIPTION("Realtek 8192C/8188C 802.11n PCI wireless"); | ||
353 | MODULE_FIRMWARE("rtlwifi/rtl8192cfw.bin"); | ||
354 | |||
355 | module_param_named(swenc, rtl92ce_mod_params.sw_crypto, bool, 0444); | ||
356 | module_param_named(ips, rtl92ce_mod_params.inactiveps, bool, 0444); | ||
357 | module_param_named(swlps, rtl92ce_mod_params.swctrl_lps, bool, 0444); | ||
358 | module_param_named(fwlps, rtl92ce_mod_params.fwctrl_lps, bool, 0444); | ||
359 | MODULE_PARM_DESC(swenc, "using hardware crypto (default 0 [hardware])\n"); | ||
360 | MODULE_PARM_DESC(ips, "using no link power save (default 1 is open)\n"); | ||
361 | MODULE_PARM_DESC(fwlps, "using linked fw control power save " | ||
362 | "(default 1 is open)\n"); | ||
363 | |||
364 | static struct pci_driver rtl92ce_driver = { | ||
365 | .name = KBUILD_MODNAME, | ||
366 | .id_table = rtl92ce_pci_ids, | ||
367 | .probe = rtl_pci_probe, | ||
368 | .remove = rtl_pci_disconnect, | ||
369 | |||
370 | #ifdef CONFIG_PM | ||
371 | .suspend = rtl_pci_suspend, | ||
372 | .resume = rtl_pci_resume, | ||
373 | #endif | ||
374 | |||
375 | }; | ||
376 | |||
377 | static int __init rtl92ce_module_init(void) | ||
378 | { | ||
379 | int ret; | ||
380 | |||
381 | ret = pci_register_driver(&rtl92ce_driver); | ||
382 | if (ret) | ||
383 | RT_ASSERT(false, (": No device found\n")); | ||
384 | |||
385 | return ret; | ||
386 | } | ||
387 | |||
388 | static void __exit rtl92ce_module_exit(void) | ||
389 | { | ||
390 | pci_unregister_driver(&rtl92ce_driver); | ||
391 | } | ||
392 | |||
393 | module_init(rtl92ce_module_init); | ||
394 | module_exit(rtl92ce_module_exit); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/sw.h b/drivers/net/wireless/rtlwifi/rtl8192ce/sw.h new file mode 100644 index 000000000000..b7dc3263e433 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/sw.h | |||
@@ -0,0 +1,41 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CE_SW_H__ | ||
31 | #define __RTL92CE_SW_H__ | ||
32 | |||
33 | int rtl92c_init_sw_vars(struct ieee80211_hw *hw); | ||
34 | void rtl92c_deinit_sw_vars(struct ieee80211_hw *hw); | ||
35 | void rtl92c_init_var_map(struct ieee80211_hw *hw); | ||
36 | bool _rtl92ce_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
37 | u8 configtype); | ||
38 | bool _rtl92ce_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
39 | u8 configtype); | ||
40 | |||
41 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/table.c b/drivers/net/wireless/rtlwifi/rtl8192ce/table.c new file mode 100644 index 000000000000..ba938b91aa6f --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/table.c | |||
@@ -0,0 +1,1224 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Created on 2010/ 5/18, 1:41 | ||
27 | * | ||
28 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
29 | * | ||
30 | *****************************************************************************/ | ||
31 | |||
32 | #include "table.h" | ||
33 | |||
34 | |||
35 | u32 RTL8192CEPHY_REG_2TARRAY[PHY_REG_2TARRAY_LENGTH] = { | ||
36 | 0x024, 0x0011800f, | ||
37 | 0x028, 0x00ffdb83, | ||
38 | 0x800, 0x80040002, | ||
39 | 0x804, 0x00000003, | ||
40 | 0x808, 0x0000fc00, | ||
41 | 0x80c, 0x0000000a, | ||
42 | 0x810, 0x10005388, | ||
43 | 0x814, 0x020c3d10, | ||
44 | 0x818, 0x02200385, | ||
45 | 0x81c, 0x00000000, | ||
46 | 0x820, 0x01000100, | ||
47 | 0x824, 0x00390004, | ||
48 | 0x828, 0x01000100, | ||
49 | 0x82c, 0x00390004, | ||
50 | 0x830, 0x27272727, | ||
51 | 0x834, 0x27272727, | ||
52 | 0x838, 0x27272727, | ||
53 | 0x83c, 0x27272727, | ||
54 | 0x840, 0x00010000, | ||
55 | 0x844, 0x00010000, | ||
56 | 0x848, 0x27272727, | ||
57 | 0x84c, 0x27272727, | ||
58 | 0x850, 0x00000000, | ||
59 | 0x854, 0x00000000, | ||
60 | 0x858, 0x569a569a, | ||
61 | 0x85c, 0x0c1b25a4, | ||
62 | 0x860, 0x66e60230, | ||
63 | 0x864, 0x061f0130, | ||
64 | 0x868, 0x27272727, | ||
65 | 0x86c, 0x2b2b2b27, | ||
66 | 0x870, 0x07000700, | ||
67 | 0x874, 0x22184000, | ||
68 | 0x878, 0x08080808, | ||
69 | 0x87c, 0x00000000, | ||
70 | 0x880, 0xc0083070, | ||
71 | 0x884, 0x000004d5, | ||
72 | 0x888, 0x00000000, | ||
73 | 0x88c, 0xcc0000c0, | ||
74 | 0x890, 0x00000800, | ||
75 | 0x894, 0xfffffffe, | ||
76 | 0x898, 0x40302010, | ||
77 | 0x89c, 0x00706050, | ||
78 | 0x900, 0x00000000, | ||
79 | 0x904, 0x00000023, | ||
80 | 0x908, 0x00000000, | ||
81 | 0x90c, 0x81121313, | ||
82 | 0xa00, 0x00d047c8, | ||
83 | 0xa04, 0x80ff000c, | ||
84 | 0xa08, 0x8c838300, | ||
85 | 0xa0c, 0x2e68120f, | ||
86 | 0xa10, 0x9500bb78, | ||
87 | 0xa14, 0x11144028, | ||
88 | 0xa18, 0x00881117, | ||
89 | 0xa1c, 0x89140f00, | ||
90 | 0xa20, 0x1a1b0000, | ||
91 | 0xa24, 0x090e1317, | ||
92 | 0xa28, 0x00000204, | ||
93 | 0xa2c, 0x00d30000, | ||
94 | 0xa70, 0x101fbf00, | ||
95 | 0xa74, 0x00000007, | ||
96 | 0xc00, 0x48071d40, | ||
97 | 0xc04, 0x03a05633, | ||
98 | 0xc08, 0x000000e4, | ||
99 | 0xc0c, 0x6c6c6c6c, | ||
100 | 0xc10, 0x08800000, | ||
101 | 0xc14, 0x40000100, | ||
102 | 0xc18, 0x08800000, | ||
103 | 0xc1c, 0x40000100, | ||
104 | 0xc20, 0x00000000, | ||
105 | 0xc24, 0x00000000, | ||
106 | 0xc28, 0x00000000, | ||
107 | 0xc2c, 0x00000000, | ||
108 | 0xc30, 0x69e9ac44, | ||
109 | 0xc34, 0x469652cf, | ||
110 | 0xc38, 0x49795994, | ||
111 | 0xc3c, 0x0a97971c, | ||
112 | 0xc40, 0x1f7c403f, | ||
113 | 0xc44, 0x000100b7, | ||
114 | 0xc48, 0xec020107, | ||
115 | 0xc4c, 0x007f037f, | ||
116 | 0xc50, 0x69543420, | ||
117 | 0xc54, 0x43bc0094, | ||
118 | 0xc58, 0x69543420, | ||
119 | 0xc5c, 0x433c0094, | ||
120 | 0xc60, 0x00000000, | ||
121 | 0xc64, 0x5116848b, | ||
122 | 0xc68, 0x47c00bff, | ||
123 | 0xc6c, 0x00000036, | ||
124 | 0xc70, 0x2c7f000d, | ||
125 | 0xc74, 0x018610db, | ||
126 | 0xc78, 0x0000001f, | ||
127 | 0xc7c, 0x00b91612, | ||
128 | 0xc80, 0x40000100, | ||
129 | 0xc84, 0x20f60000, | ||
130 | 0xc88, 0x40000100, | ||
131 | 0xc8c, 0x20200000, | ||
132 | 0xc90, 0x00121820, | ||
133 | 0xc94, 0x00000000, | ||
134 | 0xc98, 0x00121820, | ||
135 | 0xc9c, 0x00007f7f, | ||
136 | 0xca0, 0x00000000, | ||
137 | 0xca4, 0x00000080, | ||
138 | 0xca8, 0x00000000, | ||
139 | 0xcac, 0x00000000, | ||
140 | 0xcb0, 0x00000000, | ||
141 | 0xcb4, 0x00000000, | ||
142 | 0xcb8, 0x00000000, | ||
143 | 0xcbc, 0x28000000, | ||
144 | 0xcc0, 0x00000000, | ||
145 | 0xcc4, 0x00000000, | ||
146 | 0xcc8, 0x00000000, | ||
147 | 0xccc, 0x00000000, | ||
148 | 0xcd0, 0x00000000, | ||
149 | 0xcd4, 0x00000000, | ||
150 | 0xcd8, 0x64b22427, | ||
151 | 0xcdc, 0x00766932, | ||
152 | 0xce0, 0x00222222, | ||
153 | 0xce4, 0x00000000, | ||
154 | 0xce8, 0x37644302, | ||
155 | 0xcec, 0x2f97d40c, | ||
156 | 0xd00, 0x00080740, | ||
157 | 0xd04, 0x00020403, | ||
158 | 0xd08, 0x0000907f, | ||
159 | 0xd0c, 0x20010201, | ||
160 | 0xd10, 0xa0633333, | ||
161 | 0xd14, 0x3333bc43, | ||
162 | 0xd18, 0x7a8f5b6b, | ||
163 | 0xd2c, 0xcc979975, | ||
164 | 0xd30, 0x00000000, | ||
165 | 0xd34, 0x80608000, | ||
166 | 0xd38, 0x00000000, | ||
167 | 0xd3c, 0x00027293, | ||
168 | 0xd40, 0x00000000, | ||
169 | 0xd44, 0x00000000, | ||
170 | 0xd48, 0x00000000, | ||
171 | 0xd4c, 0x00000000, | ||
172 | 0xd50, 0x6437140a, | ||
173 | 0xd54, 0x00000000, | ||
174 | 0xd58, 0x00000000, | ||
175 | 0xd5c, 0x30032064, | ||
176 | 0xd60, 0x4653de68, | ||
177 | 0xd64, 0x04518a3c, | ||
178 | 0xd68, 0x00002101, | ||
179 | 0xd6c, 0x2a201c16, | ||
180 | 0xd70, 0x1812362e, | ||
181 | 0xd74, 0x322c2220, | ||
182 | 0xd78, 0x000e3c24, | ||
183 | 0xe00, 0x2a2a2a2a, | ||
184 | 0xe04, 0x2a2a2a2a, | ||
185 | 0xe08, 0x03902a2a, | ||
186 | 0xe10, 0x2a2a2a2a, | ||
187 | 0xe14, 0x2a2a2a2a, | ||
188 | 0xe18, 0x2a2a2a2a, | ||
189 | 0xe1c, 0x2a2a2a2a, | ||
190 | 0xe28, 0x00000000, | ||
191 | 0xe30, 0x1000dc1f, | ||
192 | 0xe34, 0x10008c1f, | ||
193 | 0xe38, 0x02140102, | ||
194 | 0xe3c, 0x681604c2, | ||
195 | 0xe40, 0x01007c00, | ||
196 | 0xe44, 0x01004800, | ||
197 | 0xe48, 0xfb000000, | ||
198 | 0xe4c, 0x000028d1, | ||
199 | 0xe50, 0x1000dc1f, | ||
200 | 0xe54, 0x10008c1f, | ||
201 | 0xe58, 0x02140102, | ||
202 | 0xe5c, 0x28160d05, | ||
203 | 0xe60, 0x00000010, | ||
204 | 0xe68, 0x001b25a4, | ||
205 | 0xe6c, 0x63db25a4, | ||
206 | 0xe70, 0x63db25a4, | ||
207 | 0xe74, 0x0c1b25a4, | ||
208 | 0xe78, 0x0c1b25a4, | ||
209 | 0xe7c, 0x0c1b25a4, | ||
210 | 0xe80, 0x0c1b25a4, | ||
211 | 0xe84, 0x63db25a4, | ||
212 | 0xe88, 0x0c1b25a4, | ||
213 | 0xe8c, 0x63db25a4, | ||
214 | 0xed0, 0x63db25a4, | ||
215 | 0xed4, 0x63db25a4, | ||
216 | 0xed8, 0x63db25a4, | ||
217 | 0xedc, 0x001b25a4, | ||
218 | 0xee0, 0x001b25a4, | ||
219 | 0xeec, 0x6fdb25a4, | ||
220 | 0xf14, 0x00000003, | ||
221 | 0xf4c, 0x00000000, | ||
222 | 0xf00, 0x00000300, | ||
223 | }; | ||
224 | |||
225 | u32 RTL8192CEPHY_REG_1TARRAY[PHY_REG_1TARRAY_LENGTH] = { | ||
226 | 0x024, 0x0011800f, | ||
227 | 0x028, 0x00ffdb83, | ||
228 | 0x800, 0x80040000, | ||
229 | 0x804, 0x00000001, | ||
230 | 0x808, 0x0000fc00, | ||
231 | 0x80c, 0x0000000a, | ||
232 | 0x810, 0x10005388, | ||
233 | 0x814, 0x020c3d10, | ||
234 | 0x818, 0x02200385, | ||
235 | 0x81c, 0x00000000, | ||
236 | 0x820, 0x01000100, | ||
237 | 0x824, 0x00390004, | ||
238 | 0x828, 0x00000000, | ||
239 | 0x82c, 0x00000000, | ||
240 | 0x830, 0x00000000, | ||
241 | 0x834, 0x00000000, | ||
242 | 0x838, 0x00000000, | ||
243 | 0x83c, 0x00000000, | ||
244 | 0x840, 0x00010000, | ||
245 | 0x844, 0x00000000, | ||
246 | 0x848, 0x00000000, | ||
247 | 0x84c, 0x00000000, | ||
248 | 0x850, 0x00000000, | ||
249 | 0x854, 0x00000000, | ||
250 | 0x858, 0x569a569a, | ||
251 | 0x85c, 0x001b25a4, | ||
252 | 0x860, 0x66e60230, | ||
253 | 0x864, 0x061f0130, | ||
254 | 0x868, 0x00000000, | ||
255 | 0x86c, 0x32323200, | ||
256 | 0x870, 0x07000700, | ||
257 | 0x874, 0x22004000, | ||
258 | 0x878, 0x00000808, | ||
259 | 0x87c, 0x00000000, | ||
260 | 0x880, 0xc0083070, | ||
261 | 0x884, 0x000004d5, | ||
262 | 0x888, 0x00000000, | ||
263 | 0x88c, 0xccc000c0, | ||
264 | 0x890, 0x00000800, | ||
265 | 0x894, 0xfffffffe, | ||
266 | 0x898, 0x40302010, | ||
267 | 0x89c, 0x00706050, | ||
268 | 0x900, 0x00000000, | ||
269 | 0x904, 0x00000023, | ||
270 | 0x908, 0x00000000, | ||
271 | 0x90c, 0x81121111, | ||
272 | 0xa00, 0x00d047c8, | ||
273 | 0xa04, 0x80ff000c, | ||
274 | 0xa08, 0x8c838300, | ||
275 | 0xa0c, 0x2e68120f, | ||
276 | 0xa10, 0x9500bb78, | ||
277 | 0xa14, 0x11144028, | ||
278 | 0xa18, 0x00881117, | ||
279 | 0xa1c, 0x89140f00, | ||
280 | 0xa20, 0x1a1b0000, | ||
281 | 0xa24, 0x090e1317, | ||
282 | 0xa28, 0x00000204, | ||
283 | 0xa2c, 0x00d30000, | ||
284 | 0xa70, 0x101fbf00, | ||
285 | 0xa74, 0x00000007, | ||
286 | 0xc00, 0x48071d40, | ||
287 | 0xc04, 0x03a05611, | ||
288 | 0xc08, 0x000000e4, | ||
289 | 0xc0c, 0x6c6c6c6c, | ||
290 | 0xc10, 0x08800000, | ||
291 | 0xc14, 0x40000100, | ||
292 | 0xc18, 0x08800000, | ||
293 | 0xc1c, 0x40000100, | ||
294 | 0xc20, 0x00000000, | ||
295 | 0xc24, 0x00000000, | ||
296 | 0xc28, 0x00000000, | ||
297 | 0xc2c, 0x00000000, | ||
298 | 0xc30, 0x69e9ac44, | ||
299 | 0xc34, 0x469652cf, | ||
300 | 0xc38, 0x49795994, | ||
301 | 0xc3c, 0x0a97971c, | ||
302 | 0xc40, 0x1f7c403f, | ||
303 | 0xc44, 0x000100b7, | ||
304 | 0xc48, 0xec020107, | ||
305 | 0xc4c, 0x007f037f, | ||
306 | 0xc50, 0x69543420, | ||
307 | 0xc54, 0x43bc0094, | ||
308 | 0xc58, 0x69543420, | ||
309 | 0xc5c, 0x433c0094, | ||
310 | 0xc60, 0x00000000, | ||
311 | 0xc64, 0x5116848b, | ||
312 | 0xc68, 0x47c00bff, | ||
313 | 0xc6c, 0x00000036, | ||
314 | 0xc70, 0x2c7f000d, | ||
315 | 0xc74, 0x018610db, | ||
316 | 0xc78, 0x0000001f, | ||
317 | 0xc7c, 0x00b91612, | ||
318 | 0xc80, 0x40000100, | ||
319 | 0xc84, 0x20f60000, | ||
320 | 0xc88, 0x40000100, | ||
321 | 0xc8c, 0x20200000, | ||
322 | 0xc90, 0x00121820, | ||
323 | 0xc94, 0x00000000, | ||
324 | 0xc98, 0x00121820, | ||
325 | 0xc9c, 0x00007f7f, | ||
326 | 0xca0, 0x00000000, | ||
327 | 0xca4, 0x00000080, | ||
328 | 0xca8, 0x00000000, | ||
329 | 0xcac, 0x00000000, | ||
330 | 0xcb0, 0x00000000, | ||
331 | 0xcb4, 0x00000000, | ||
332 | 0xcb8, 0x00000000, | ||
333 | 0xcbc, 0x28000000, | ||
334 | 0xcc0, 0x00000000, | ||
335 | 0xcc4, 0x00000000, | ||
336 | 0xcc8, 0x00000000, | ||
337 | 0xccc, 0x00000000, | ||
338 | 0xcd0, 0x00000000, | ||
339 | 0xcd4, 0x00000000, | ||
340 | 0xcd8, 0x64b22427, | ||
341 | 0xcdc, 0x00766932, | ||
342 | 0xce0, 0x00222222, | ||
343 | 0xce4, 0x00000000, | ||
344 | 0xce8, 0x37644302, | ||
345 | 0xcec, 0x2f97d40c, | ||
346 | 0xd00, 0x00080740, | ||
347 | 0xd04, 0x00020401, | ||
348 | 0xd08, 0x0000907f, | ||
349 | 0xd0c, 0x20010201, | ||
350 | 0xd10, 0xa0633333, | ||
351 | 0xd14, 0x3333bc43, | ||
352 | 0xd18, 0x7a8f5b6b, | ||
353 | 0xd2c, 0xcc979975, | ||
354 | 0xd30, 0x00000000, | ||
355 | 0xd34, 0x80608000, | ||
356 | 0xd38, 0x00000000, | ||
357 | 0xd3c, 0x00027293, | ||
358 | 0xd40, 0x00000000, | ||
359 | 0xd44, 0x00000000, | ||
360 | 0xd48, 0x00000000, | ||
361 | 0xd4c, 0x00000000, | ||
362 | 0xd50, 0x6437140a, | ||
363 | 0xd54, 0x00000000, | ||
364 | 0xd58, 0x00000000, | ||
365 | 0xd5c, 0x30032064, | ||
366 | 0xd60, 0x4653de68, | ||
367 | 0xd64, 0x04518a3c, | ||
368 | 0xd68, 0x00002101, | ||
369 | 0xd6c, 0x2a201c16, | ||
370 | 0xd70, 0x1812362e, | ||
371 | 0xd74, 0x322c2220, | ||
372 | 0xd78, 0x000e3c24, | ||
373 | 0xe00, 0x2a2a2a2a, | ||
374 | 0xe04, 0x2a2a2a2a, | ||
375 | 0xe08, 0x03902a2a, | ||
376 | 0xe10, 0x2a2a2a2a, | ||
377 | 0xe14, 0x2a2a2a2a, | ||
378 | 0xe18, 0x2a2a2a2a, | ||
379 | 0xe1c, 0x2a2a2a2a, | ||
380 | 0xe28, 0x00000000, | ||
381 | 0xe30, 0x1000dc1f, | ||
382 | 0xe34, 0x10008c1f, | ||
383 | 0xe38, 0x02140102, | ||
384 | 0xe3c, 0x681604c2, | ||
385 | 0xe40, 0x01007c00, | ||
386 | 0xe44, 0x01004800, | ||
387 | 0xe48, 0xfb000000, | ||
388 | 0xe4c, 0x000028d1, | ||
389 | 0xe50, 0x1000dc1f, | ||
390 | 0xe54, 0x10008c1f, | ||
391 | 0xe58, 0x02140102, | ||
392 | 0xe5c, 0x28160d05, | ||
393 | 0xe60, 0x00000010, | ||
394 | 0xe68, 0x001b25a4, | ||
395 | 0xe6c, 0x631b25a0, | ||
396 | 0xe70, 0x631b25a0, | ||
397 | 0xe74, 0x081b25a0, | ||
398 | 0xe78, 0x081b25a0, | ||
399 | 0xe7c, 0x081b25a0, | ||
400 | 0xe80, 0x081b25a0, | ||
401 | 0xe84, 0x631b25a0, | ||
402 | 0xe88, 0x081b25a0, | ||
403 | 0xe8c, 0x631b25a0, | ||
404 | 0xed0, 0x631b25a0, | ||
405 | 0xed4, 0x631b25a0, | ||
406 | 0xed8, 0x631b25a0, | ||
407 | 0xedc, 0x001b25a0, | ||
408 | 0xee0, 0x001b25a0, | ||
409 | 0xeec, 0x6b1b25a0, | ||
410 | 0xf14, 0x00000003, | ||
411 | 0xf4c, 0x00000000, | ||
412 | 0xf00, 0x00000300, | ||
413 | }; | ||
414 | |||
415 | u32 RTL8192CEPHY_REG_ARRAY_PG[PHY_REG_ARRAY_PGLENGTH] = { | ||
416 | 0xe00, 0xffffffff, 0x0a0c0c0c, | ||
417 | 0xe04, 0xffffffff, 0x02040608, | ||
418 | 0xe08, 0x0000ff00, 0x00000000, | ||
419 | 0x86c, 0xffffff00, 0x00000000, | ||
420 | 0xe10, 0xffffffff, 0x0a0c0d0e, | ||
421 | 0xe14, 0xffffffff, 0x02040608, | ||
422 | 0xe18, 0xffffffff, 0x0a0c0d0e, | ||
423 | 0xe1c, 0xffffffff, 0x02040608, | ||
424 | 0x830, 0xffffffff, 0x0a0c0c0c, | ||
425 | 0x834, 0xffffffff, 0x02040608, | ||
426 | 0x838, 0xffffff00, 0x00000000, | ||
427 | 0x86c, 0x000000ff, 0x00000000, | ||
428 | 0x83c, 0xffffffff, 0x0a0c0d0e, | ||
429 | 0x848, 0xffffffff, 0x02040608, | ||
430 | 0x84c, 0xffffffff, 0x0a0c0d0e, | ||
431 | 0x868, 0xffffffff, 0x02040608, | ||
432 | 0xe00, 0xffffffff, 0x00000000, | ||
433 | 0xe04, 0xffffffff, 0x00000000, | ||
434 | 0xe08, 0x0000ff00, 0x00000000, | ||
435 | 0x86c, 0xffffff00, 0x00000000, | ||
436 | 0xe10, 0xffffffff, 0x00000000, | ||
437 | 0xe14, 0xffffffff, 0x00000000, | ||
438 | 0xe18, 0xffffffff, 0x00000000, | ||
439 | 0xe1c, 0xffffffff, 0x00000000, | ||
440 | 0x830, 0xffffffff, 0x00000000, | ||
441 | 0x834, 0xffffffff, 0x00000000, | ||
442 | 0x838, 0xffffff00, 0x00000000, | ||
443 | 0x86c, 0x000000ff, 0x00000000, | ||
444 | 0x83c, 0xffffffff, 0x00000000, | ||
445 | 0x848, 0xffffffff, 0x00000000, | ||
446 | 0x84c, 0xffffffff, 0x00000000, | ||
447 | 0x868, 0xffffffff, 0x00000000, | ||
448 | 0xe00, 0xffffffff, 0x04040404, | ||
449 | 0xe04, 0xffffffff, 0x00020204, | ||
450 | 0xe08, 0x0000ff00, 0x00000000, | ||
451 | 0x86c, 0xffffff00, 0x00000000, | ||
452 | 0xe10, 0xffffffff, 0x06060606, | ||
453 | 0xe14, 0xffffffff, 0x00020406, | ||
454 | 0xe18, 0xffffffff, 0x06060606, | ||
455 | 0xe1c, 0xffffffff, 0x00020406, | ||
456 | 0x830, 0xffffffff, 0x04040404, | ||
457 | 0x834, 0xffffffff, 0x00020204, | ||
458 | 0x838, 0xffffff00, 0x00000000, | ||
459 | 0x86c, 0x000000ff, 0x00000000, | ||
460 | 0x83c, 0xffffffff, 0x06060606, | ||
461 | 0x848, 0xffffffff, 0x00020406, | ||
462 | 0x84c, 0xffffffff, 0x06060606, | ||
463 | 0x868, 0xffffffff, 0x00020406, | ||
464 | 0xe00, 0xffffffff, 0x00000000, | ||
465 | 0xe04, 0xffffffff, 0x00000000, | ||
466 | 0xe08, 0x0000ff00, 0x00000000, | ||
467 | 0x86c, 0xffffff00, 0x00000000, | ||
468 | 0xe10, 0xffffffff, 0x00000000, | ||
469 | 0xe14, 0xffffffff, 0x00000000, | ||
470 | 0xe18, 0xffffffff, 0x00000000, | ||
471 | 0xe1c, 0xffffffff, 0x00000000, | ||
472 | 0x830, 0xffffffff, 0x00000000, | ||
473 | 0x834, 0xffffffff, 0x00000000, | ||
474 | 0x838, 0xffffff00, 0x00000000, | ||
475 | 0x86c, 0x000000ff, 0x00000000, | ||
476 | 0x83c, 0xffffffff, 0x00000000, | ||
477 | 0x848, 0xffffffff, 0x00000000, | ||
478 | 0x84c, 0xffffffff, 0x00000000, | ||
479 | 0x868, 0xffffffff, 0x00000000, | ||
480 | }; | ||
481 | |||
482 | u32 RTL8192CERADIOA_2TARRAY[RADIOA_2TARRAYLENGTH] = { | ||
483 | 0x000, 0x00030159, | ||
484 | 0x001, 0x00031284, | ||
485 | 0x002, 0x00098000, | ||
486 | 0x003, 0x00018c63, | ||
487 | 0x004, 0x000210e7, | ||
488 | 0x009, 0x0002044f, | ||
489 | 0x00a, 0x0001adb0, | ||
490 | 0x00b, 0x00054867, | ||
491 | 0x00c, 0x0008992e, | ||
492 | 0x00d, 0x0000e52c, | ||
493 | 0x00e, 0x00039ce7, | ||
494 | 0x00f, 0x00000451, | ||
495 | 0x019, 0x00000000, | ||
496 | 0x01a, 0x00010255, | ||
497 | 0x01b, 0x00060a00, | ||
498 | 0x01c, 0x000fc378, | ||
499 | 0x01d, 0x000a1250, | ||
500 | 0x01e, 0x0004445f, | ||
501 | 0x01f, 0x00080001, | ||
502 | 0x020, 0x0000b614, | ||
503 | 0x021, 0x0006c000, | ||
504 | 0x022, 0x00000000, | ||
505 | 0x023, 0x00001558, | ||
506 | 0x024, 0x00000060, | ||
507 | 0x025, 0x00000483, | ||
508 | 0x026, 0x0004f000, | ||
509 | 0x027, 0x000ec7d9, | ||
510 | 0x028, 0x000977c0, | ||
511 | 0x029, 0x00004783, | ||
512 | 0x02a, 0x00000001, | ||
513 | 0x02b, 0x00021334, | ||
514 | 0x02a, 0x00000000, | ||
515 | 0x02b, 0x00000054, | ||
516 | 0x02a, 0x00000001, | ||
517 | 0x02b, 0x00000808, | ||
518 | 0x02b, 0x00053333, | ||
519 | 0x02c, 0x0000000c, | ||
520 | 0x02a, 0x00000002, | ||
521 | 0x02b, 0x00000808, | ||
522 | 0x02b, 0x0005b333, | ||
523 | 0x02c, 0x0000000d, | ||
524 | 0x02a, 0x00000003, | ||
525 | 0x02b, 0x00000808, | ||
526 | 0x02b, 0x00063333, | ||
527 | 0x02c, 0x0000000d, | ||
528 | 0x02a, 0x00000004, | ||
529 | 0x02b, 0x00000808, | ||
530 | 0x02b, 0x0006b333, | ||
531 | 0x02c, 0x0000000d, | ||
532 | 0x02a, 0x00000005, | ||
533 | 0x02b, 0x00000808, | ||
534 | 0x02b, 0x00073333, | ||
535 | 0x02c, 0x0000000d, | ||
536 | 0x02a, 0x00000006, | ||
537 | 0x02b, 0x00000709, | ||
538 | 0x02b, 0x0005b333, | ||
539 | 0x02c, 0x0000000d, | ||
540 | 0x02a, 0x00000007, | ||
541 | 0x02b, 0x00000709, | ||
542 | 0x02b, 0x00063333, | ||
543 | 0x02c, 0x0000000d, | ||
544 | 0x02a, 0x00000008, | ||
545 | 0x02b, 0x0000060a, | ||
546 | 0x02b, 0x0004b333, | ||
547 | 0x02c, 0x0000000d, | ||
548 | 0x02a, 0x00000009, | ||
549 | 0x02b, 0x0000060a, | ||
550 | 0x02b, 0x00053333, | ||
551 | 0x02c, 0x0000000d, | ||
552 | 0x02a, 0x0000000a, | ||
553 | 0x02b, 0x0000060a, | ||
554 | 0x02b, 0x0005b333, | ||
555 | 0x02c, 0x0000000d, | ||
556 | 0x02a, 0x0000000b, | ||
557 | 0x02b, 0x0000060a, | ||
558 | 0x02b, 0x00063333, | ||
559 | 0x02c, 0x0000000d, | ||
560 | 0x02a, 0x0000000c, | ||
561 | 0x02b, 0x0000060a, | ||
562 | 0x02b, 0x0006b333, | ||
563 | 0x02c, 0x0000000d, | ||
564 | 0x02a, 0x0000000d, | ||
565 | 0x02b, 0x0000060a, | ||
566 | 0x02b, 0x00073333, | ||
567 | 0x02c, 0x0000000d, | ||
568 | 0x02a, 0x0000000e, | ||
569 | 0x02b, 0x0000050b, | ||
570 | 0x02b, 0x00066666, | ||
571 | 0x02c, 0x0000001a, | ||
572 | 0x02a, 0x000e0000, | ||
573 | 0x010, 0x0004000f, | ||
574 | 0x011, 0x000e31fc, | ||
575 | 0x010, 0x0006000f, | ||
576 | 0x011, 0x000ff9f8, | ||
577 | 0x010, 0x0002000f, | ||
578 | 0x011, 0x000203f9, | ||
579 | 0x010, 0x0003000f, | ||
580 | 0x011, 0x000ff500, | ||
581 | 0x010, 0x00000000, | ||
582 | 0x011, 0x00000000, | ||
583 | 0x010, 0x0008000f, | ||
584 | 0x011, 0x0003f100, | ||
585 | 0x010, 0x0009000f, | ||
586 | 0x011, 0x00023100, | ||
587 | 0x012, 0x00032000, | ||
588 | 0x012, 0x00071000, | ||
589 | 0x012, 0x000b0000, | ||
590 | 0x012, 0x000fc000, | ||
591 | 0x013, 0x000287af, | ||
592 | 0x013, 0x000244b7, | ||
593 | 0x013, 0x000204ab, | ||
594 | 0x013, 0x0001c49f, | ||
595 | 0x013, 0x00018493, | ||
596 | 0x013, 0x00014297, | ||
597 | 0x013, 0x00010295, | ||
598 | 0x013, 0x0000c298, | ||
599 | 0x013, 0x0000819c, | ||
600 | 0x013, 0x000040a8, | ||
601 | 0x013, 0x0000001c, | ||
602 | 0x014, 0x0001944c, | ||
603 | 0x014, 0x00059444, | ||
604 | 0x014, 0x0009944c, | ||
605 | 0x014, 0x000d9444, | ||
606 | 0x015, 0x0000f424, | ||
607 | 0x015, 0x0004f424, | ||
608 | 0x015, 0x0008f424, | ||
609 | 0x015, 0x000cf424, | ||
610 | 0x016, 0x000e0330, | ||
611 | 0x016, 0x000a0330, | ||
612 | 0x016, 0x00060330, | ||
613 | 0x016, 0x00020330, | ||
614 | 0x000, 0x00010159, | ||
615 | 0x018, 0x0000f401, | ||
616 | 0x0fe, 0x00000000, | ||
617 | 0x0fe, 0x00000000, | ||
618 | 0x01f, 0x00080003, | ||
619 | 0x0fe, 0x00000000, | ||
620 | 0x0fe, 0x00000000, | ||
621 | 0x01e, 0x00044457, | ||
622 | 0x01f, 0x00080000, | ||
623 | 0x000, 0x00030159, | ||
624 | }; | ||
625 | |||
626 | u32 RTL8192CE_RADIOB_2TARRAY[RADIOB_2TARRAYLENGTH] = { | ||
627 | 0x000, 0x00030159, | ||
628 | 0x001, 0x00031284, | ||
629 | 0x002, 0x00098000, | ||
630 | 0x003, 0x00018c63, | ||
631 | 0x004, 0x000210e7, | ||
632 | 0x009, 0x0002044f, | ||
633 | 0x00a, 0x0001adb0, | ||
634 | 0x00b, 0x00054867, | ||
635 | 0x00c, 0x0008992e, | ||
636 | 0x00d, 0x0000e52c, | ||
637 | 0x00e, 0x00039ce7, | ||
638 | 0x00f, 0x00000451, | ||
639 | 0x012, 0x00032000, | ||
640 | 0x012, 0x00071000, | ||
641 | 0x012, 0x000b0000, | ||
642 | 0x012, 0x000fc000, | ||
643 | 0x013, 0x000287af, | ||
644 | 0x013, 0x000244b7, | ||
645 | 0x013, 0x000204ab, | ||
646 | 0x013, 0x0001c49f, | ||
647 | 0x013, 0x00018493, | ||
648 | 0x013, 0x00014297, | ||
649 | 0x013, 0x00010295, | ||
650 | 0x013, 0x0000c298, | ||
651 | 0x013, 0x0000819c, | ||
652 | 0x013, 0x000040a8, | ||
653 | 0x013, 0x0000001c, | ||
654 | 0x014, 0x0001944c, | ||
655 | 0x014, 0x00059444, | ||
656 | 0x014, 0x0009944c, | ||
657 | 0x014, 0x000d9444, | ||
658 | 0x015, 0x0000f424, | ||
659 | 0x015, 0x0004f424, | ||
660 | 0x015, 0x0008f424, | ||
661 | 0x015, 0x000cf424, | ||
662 | 0x016, 0x000e0330, | ||
663 | 0x016, 0x000a0330, | ||
664 | 0x016, 0x00060330, | ||
665 | 0x016, 0x00020330, | ||
666 | }; | ||
667 | |||
668 | u32 RTL8192CE_RADIOA_1TARRAY[RADIOA_1TARRAYLENGTH] = { | ||
669 | 0x000, 0x00030159, | ||
670 | 0x001, 0x00031284, | ||
671 | 0x002, 0x00098000, | ||
672 | 0x003, 0x00018c63, | ||
673 | 0x004, 0x000210e7, | ||
674 | 0x009, 0x0002044f, | ||
675 | 0x00a, 0x0001adb0, | ||
676 | 0x00b, 0x00054867, | ||
677 | 0x00c, 0x0008992e, | ||
678 | 0x00d, 0x0000e52c, | ||
679 | 0x00e, 0x00039ce7, | ||
680 | 0x00f, 0x00000451, | ||
681 | 0x019, 0x00000000, | ||
682 | 0x01a, 0x00010255, | ||
683 | 0x01b, 0x00060a00, | ||
684 | 0x01c, 0x000fc378, | ||
685 | 0x01d, 0x000a1250, | ||
686 | 0x01e, 0x0004445f, | ||
687 | 0x01f, 0x00080001, | ||
688 | 0x020, 0x0000b614, | ||
689 | 0x021, 0x0006c000, | ||
690 | 0x022, 0x00000000, | ||
691 | 0x023, 0x00001558, | ||
692 | 0x024, 0x00000060, | ||
693 | 0x025, 0x00000483, | ||
694 | 0x026, 0x0004f000, | ||
695 | 0x027, 0x000ec7d9, | ||
696 | 0x028, 0x000977c0, | ||
697 | 0x029, 0x00004783, | ||
698 | 0x02a, 0x00000001, | ||
699 | 0x02b, 0x00021334, | ||
700 | 0x02a, 0x00000000, | ||
701 | 0x02b, 0x00000054, | ||
702 | 0x02a, 0x00000001, | ||
703 | 0x02b, 0x00000808, | ||
704 | 0x02b, 0x00053333, | ||
705 | 0x02c, 0x0000000c, | ||
706 | 0x02a, 0x00000002, | ||
707 | 0x02b, 0x00000808, | ||
708 | 0x02b, 0x0005b333, | ||
709 | 0x02c, 0x0000000d, | ||
710 | 0x02a, 0x00000003, | ||
711 | 0x02b, 0x00000808, | ||
712 | 0x02b, 0x00063333, | ||
713 | 0x02c, 0x0000000d, | ||
714 | 0x02a, 0x00000004, | ||
715 | 0x02b, 0x00000808, | ||
716 | 0x02b, 0x0006b333, | ||
717 | 0x02c, 0x0000000d, | ||
718 | 0x02a, 0x00000005, | ||
719 | 0x02b, 0x00000808, | ||
720 | 0x02b, 0x00073333, | ||
721 | 0x02c, 0x0000000d, | ||
722 | 0x02a, 0x00000006, | ||
723 | 0x02b, 0x00000709, | ||
724 | 0x02b, 0x0005b333, | ||
725 | 0x02c, 0x0000000d, | ||
726 | 0x02a, 0x00000007, | ||
727 | 0x02b, 0x00000709, | ||
728 | 0x02b, 0x00063333, | ||
729 | 0x02c, 0x0000000d, | ||
730 | 0x02a, 0x00000008, | ||
731 | 0x02b, 0x0000060a, | ||
732 | 0x02b, 0x0004b333, | ||
733 | 0x02c, 0x0000000d, | ||
734 | 0x02a, 0x00000009, | ||
735 | 0x02b, 0x0000060a, | ||
736 | 0x02b, 0x00053333, | ||
737 | 0x02c, 0x0000000d, | ||
738 | 0x02a, 0x0000000a, | ||
739 | 0x02b, 0x0000060a, | ||
740 | 0x02b, 0x0005b333, | ||
741 | 0x02c, 0x0000000d, | ||
742 | 0x02a, 0x0000000b, | ||
743 | 0x02b, 0x0000060a, | ||
744 | 0x02b, 0x00063333, | ||
745 | 0x02c, 0x0000000d, | ||
746 | 0x02a, 0x0000000c, | ||
747 | 0x02b, 0x0000060a, | ||
748 | 0x02b, 0x0006b333, | ||
749 | 0x02c, 0x0000000d, | ||
750 | 0x02a, 0x0000000d, | ||
751 | 0x02b, 0x0000060a, | ||
752 | 0x02b, 0x00073333, | ||
753 | 0x02c, 0x0000000d, | ||
754 | 0x02a, 0x0000000e, | ||
755 | 0x02b, 0x0000050b, | ||
756 | 0x02b, 0x00066666, | ||
757 | 0x02c, 0x0000001a, | ||
758 | 0x02a, 0x000e0000, | ||
759 | 0x010, 0x0004000f, | ||
760 | 0x011, 0x000e31fc, | ||
761 | 0x010, 0x0006000f, | ||
762 | 0x011, 0x000ff9f8, | ||
763 | 0x010, 0x0002000f, | ||
764 | 0x011, 0x000203f9, | ||
765 | 0x010, 0x0003000f, | ||
766 | 0x011, 0x000ff500, | ||
767 | 0x010, 0x00000000, | ||
768 | 0x011, 0x00000000, | ||
769 | 0x010, 0x0008000f, | ||
770 | 0x011, 0x0003f100, | ||
771 | 0x010, 0x0009000f, | ||
772 | 0x011, 0x00023100, | ||
773 | 0x012, 0x00032000, | ||
774 | 0x012, 0x00071000, | ||
775 | 0x012, 0x000b0000, | ||
776 | 0x012, 0x000fc000, | ||
777 | 0x013, 0x000287af, | ||
778 | 0x013, 0x000244b7, | ||
779 | 0x013, 0x000204ab, | ||
780 | 0x013, 0x0001c49f, | ||
781 | 0x013, 0x00018493, | ||
782 | 0x013, 0x00014297, | ||
783 | 0x013, 0x00010295, | ||
784 | 0x013, 0x0000c298, | ||
785 | 0x013, 0x0000819c, | ||
786 | 0x013, 0x000040a8, | ||
787 | 0x013, 0x0000001c, | ||
788 | 0x014, 0x0001944c, | ||
789 | 0x014, 0x00059444, | ||
790 | 0x014, 0x0009944c, | ||
791 | 0x014, 0x000d9444, | ||
792 | 0x015, 0x0000f424, | ||
793 | 0x015, 0x0004f424, | ||
794 | 0x015, 0x0008f424, | ||
795 | 0x015, 0x000cf424, | ||
796 | 0x016, 0x000e0330, | ||
797 | 0x016, 0x000a0330, | ||
798 | 0x016, 0x00060330, | ||
799 | 0x016, 0x00020330, | ||
800 | 0x000, 0x00010159, | ||
801 | 0x018, 0x0000f401, | ||
802 | 0x0fe, 0x00000000, | ||
803 | 0x0fe, 0x00000000, | ||
804 | 0x01f, 0x00080003, | ||
805 | 0x0fe, 0x00000000, | ||
806 | 0x0fe, 0x00000000, | ||
807 | 0x01e, 0x00044457, | ||
808 | 0x01f, 0x00080000, | ||
809 | 0x000, 0x00030159, | ||
810 | }; | ||
811 | |||
812 | u32 RTL8192CE_RADIOB_1TARRAY[RADIOB_1TARRAYLENGTH] = { | ||
813 | 0x0, | ||
814 | }; | ||
815 | |||
816 | u32 RTL8192CEMAC_2T_ARRAY[MAC_2T_ARRAYLENGTH] = { | ||
817 | 0x420, 0x00000080, | ||
818 | 0x423, 0x00000000, | ||
819 | 0x430, 0x00000000, | ||
820 | 0x431, 0x00000000, | ||
821 | 0x432, 0x00000000, | ||
822 | 0x433, 0x00000001, | ||
823 | 0x434, 0x00000004, | ||
824 | 0x435, 0x00000005, | ||
825 | 0x436, 0x00000006, | ||
826 | 0x437, 0x00000007, | ||
827 | 0x438, 0x00000000, | ||
828 | 0x439, 0x00000000, | ||
829 | 0x43a, 0x00000000, | ||
830 | 0x43b, 0x00000001, | ||
831 | 0x43c, 0x00000004, | ||
832 | 0x43d, 0x00000005, | ||
833 | 0x43e, 0x00000006, | ||
834 | 0x43f, 0x00000007, | ||
835 | 0x440, 0x0000005d, | ||
836 | 0x441, 0x00000001, | ||
837 | 0x442, 0x00000000, | ||
838 | 0x444, 0x00000015, | ||
839 | 0x445, 0x000000f0, | ||
840 | 0x446, 0x0000000f, | ||
841 | 0x447, 0x00000000, | ||
842 | 0x458, 0x00000041, | ||
843 | 0x459, 0x000000a8, | ||
844 | 0x45a, 0x00000072, | ||
845 | 0x45b, 0x000000b9, | ||
846 | 0x460, 0x00000088, | ||
847 | 0x461, 0x00000088, | ||
848 | 0x462, 0x00000006, | ||
849 | 0x463, 0x00000003, | ||
850 | 0x4c8, 0x00000004, | ||
851 | 0x4c9, 0x00000008, | ||
852 | 0x4cc, 0x00000002, | ||
853 | 0x4cd, 0x00000028, | ||
854 | 0x4ce, 0x00000001, | ||
855 | 0x500, 0x00000026, | ||
856 | 0x501, 0x000000a2, | ||
857 | 0x502, 0x0000002f, | ||
858 | 0x503, 0x00000000, | ||
859 | 0x504, 0x00000028, | ||
860 | 0x505, 0x000000a3, | ||
861 | 0x506, 0x0000005e, | ||
862 | 0x507, 0x00000000, | ||
863 | 0x508, 0x0000002b, | ||
864 | 0x509, 0x000000a4, | ||
865 | 0x50a, 0x0000005e, | ||
866 | 0x50b, 0x00000000, | ||
867 | 0x50c, 0x0000004f, | ||
868 | 0x50d, 0x000000a4, | ||
869 | 0x50e, 0x00000000, | ||
870 | 0x50f, 0x00000000, | ||
871 | 0x512, 0x0000001c, | ||
872 | 0x514, 0x0000000a, | ||
873 | 0x515, 0x00000010, | ||
874 | 0x516, 0x0000000a, | ||
875 | 0x517, 0x00000010, | ||
876 | 0x51a, 0x00000016, | ||
877 | 0x524, 0x0000000f, | ||
878 | 0x525, 0x0000004f, | ||
879 | 0x546, 0x00000020, | ||
880 | 0x547, 0x00000000, | ||
881 | 0x559, 0x00000002, | ||
882 | 0x55a, 0x00000002, | ||
883 | 0x55d, 0x000000ff, | ||
884 | 0x605, 0x00000030, | ||
885 | 0x608, 0x0000000e, | ||
886 | 0x609, 0x0000002a, | ||
887 | 0x652, 0x00000020, | ||
888 | 0x63c, 0x0000000a, | ||
889 | 0x63d, 0x0000000a, | ||
890 | 0x700, 0x00000021, | ||
891 | 0x701, 0x00000043, | ||
892 | 0x702, 0x00000065, | ||
893 | 0x703, 0x00000087, | ||
894 | 0x708, 0x00000021, | ||
895 | 0x709, 0x00000043, | ||
896 | 0x70a, 0x00000065, | ||
897 | 0x70b, 0x00000087, | ||
898 | }; | ||
899 | |||
900 | u32 RTL8192CEAGCTAB_2TARRAY[AGCTAB_2TARRAYLENGTH] = { | ||
901 | 0xc78, 0x7b000001, | ||
902 | 0xc78, 0x7b010001, | ||
903 | 0xc78, 0x7b020001, | ||
904 | 0xc78, 0x7b030001, | ||
905 | 0xc78, 0x7b040001, | ||
906 | 0xc78, 0x7b050001, | ||
907 | 0xc78, 0x7a060001, | ||
908 | 0xc78, 0x79070001, | ||
909 | 0xc78, 0x78080001, | ||
910 | 0xc78, 0x77090001, | ||
911 | 0xc78, 0x760a0001, | ||
912 | 0xc78, 0x750b0001, | ||
913 | 0xc78, 0x740c0001, | ||
914 | 0xc78, 0x730d0001, | ||
915 | 0xc78, 0x720e0001, | ||
916 | 0xc78, 0x710f0001, | ||
917 | 0xc78, 0x70100001, | ||
918 | 0xc78, 0x6f110001, | ||
919 | 0xc78, 0x6e120001, | ||
920 | 0xc78, 0x6d130001, | ||
921 | 0xc78, 0x6c140001, | ||
922 | 0xc78, 0x6b150001, | ||
923 | 0xc78, 0x6a160001, | ||
924 | 0xc78, 0x69170001, | ||
925 | 0xc78, 0x68180001, | ||
926 | 0xc78, 0x67190001, | ||
927 | 0xc78, 0x661a0001, | ||
928 | 0xc78, 0x651b0001, | ||
929 | 0xc78, 0x641c0001, | ||
930 | 0xc78, 0x631d0001, | ||
931 | 0xc78, 0x621e0001, | ||
932 | 0xc78, 0x611f0001, | ||
933 | 0xc78, 0x60200001, | ||
934 | 0xc78, 0x49210001, | ||
935 | 0xc78, 0x48220001, | ||
936 | 0xc78, 0x47230001, | ||
937 | 0xc78, 0x46240001, | ||
938 | 0xc78, 0x45250001, | ||
939 | 0xc78, 0x44260001, | ||
940 | 0xc78, 0x43270001, | ||
941 | 0xc78, 0x42280001, | ||
942 | 0xc78, 0x41290001, | ||
943 | 0xc78, 0x402a0001, | ||
944 | 0xc78, 0x262b0001, | ||
945 | 0xc78, 0x252c0001, | ||
946 | 0xc78, 0x242d0001, | ||
947 | 0xc78, 0x232e0001, | ||
948 | 0xc78, 0x222f0001, | ||
949 | 0xc78, 0x21300001, | ||
950 | 0xc78, 0x20310001, | ||
951 | 0xc78, 0x06320001, | ||
952 | 0xc78, 0x05330001, | ||
953 | 0xc78, 0x04340001, | ||
954 | 0xc78, 0x03350001, | ||
955 | 0xc78, 0x02360001, | ||
956 | 0xc78, 0x01370001, | ||
957 | 0xc78, 0x00380001, | ||
958 | 0xc78, 0x00390001, | ||
959 | 0xc78, 0x003a0001, | ||
960 | 0xc78, 0x003b0001, | ||
961 | 0xc78, 0x003c0001, | ||
962 | 0xc78, 0x003d0001, | ||
963 | 0xc78, 0x003e0001, | ||
964 | 0xc78, 0x003f0001, | ||
965 | 0xc78, 0x7b400001, | ||
966 | 0xc78, 0x7b410001, | ||
967 | 0xc78, 0x7b420001, | ||
968 | 0xc78, 0x7b430001, | ||
969 | 0xc78, 0x7b440001, | ||
970 | 0xc78, 0x7b450001, | ||
971 | 0xc78, 0x7a460001, | ||
972 | 0xc78, 0x79470001, | ||
973 | 0xc78, 0x78480001, | ||
974 | 0xc78, 0x77490001, | ||
975 | 0xc78, 0x764a0001, | ||
976 | 0xc78, 0x754b0001, | ||
977 | 0xc78, 0x744c0001, | ||
978 | 0xc78, 0x734d0001, | ||
979 | 0xc78, 0x724e0001, | ||
980 | 0xc78, 0x714f0001, | ||
981 | 0xc78, 0x70500001, | ||
982 | 0xc78, 0x6f510001, | ||
983 | 0xc78, 0x6e520001, | ||
984 | 0xc78, 0x6d530001, | ||
985 | 0xc78, 0x6c540001, | ||
986 | 0xc78, 0x6b550001, | ||
987 | 0xc78, 0x6a560001, | ||
988 | 0xc78, 0x69570001, | ||
989 | 0xc78, 0x68580001, | ||
990 | 0xc78, 0x67590001, | ||
991 | 0xc78, 0x665a0001, | ||
992 | 0xc78, 0x655b0001, | ||
993 | 0xc78, 0x645c0001, | ||
994 | 0xc78, 0x635d0001, | ||
995 | 0xc78, 0x625e0001, | ||
996 | 0xc78, 0x615f0001, | ||
997 | 0xc78, 0x60600001, | ||
998 | 0xc78, 0x49610001, | ||
999 | 0xc78, 0x48620001, | ||
1000 | 0xc78, 0x47630001, | ||
1001 | 0xc78, 0x46640001, | ||
1002 | 0xc78, 0x45650001, | ||
1003 | 0xc78, 0x44660001, | ||
1004 | 0xc78, 0x43670001, | ||
1005 | 0xc78, 0x42680001, | ||
1006 | 0xc78, 0x41690001, | ||
1007 | 0xc78, 0x406a0001, | ||
1008 | 0xc78, 0x266b0001, | ||
1009 | 0xc78, 0x256c0001, | ||
1010 | 0xc78, 0x246d0001, | ||
1011 | 0xc78, 0x236e0001, | ||
1012 | 0xc78, 0x226f0001, | ||
1013 | 0xc78, 0x21700001, | ||
1014 | 0xc78, 0x20710001, | ||
1015 | 0xc78, 0x06720001, | ||
1016 | 0xc78, 0x05730001, | ||
1017 | 0xc78, 0x04740001, | ||
1018 | 0xc78, 0x03750001, | ||
1019 | 0xc78, 0x02760001, | ||
1020 | 0xc78, 0x01770001, | ||
1021 | 0xc78, 0x00780001, | ||
1022 | 0xc78, 0x00790001, | ||
1023 | 0xc78, 0x007a0001, | ||
1024 | 0xc78, 0x007b0001, | ||
1025 | 0xc78, 0x007c0001, | ||
1026 | 0xc78, 0x007d0001, | ||
1027 | 0xc78, 0x007e0001, | ||
1028 | 0xc78, 0x007f0001, | ||
1029 | 0xc78, 0x3800001e, | ||
1030 | 0xc78, 0x3801001e, | ||
1031 | 0xc78, 0x3802001e, | ||
1032 | 0xc78, 0x3803001e, | ||
1033 | 0xc78, 0x3804001e, | ||
1034 | 0xc78, 0x3805001e, | ||
1035 | 0xc78, 0x3806001e, | ||
1036 | 0xc78, 0x3807001e, | ||
1037 | 0xc78, 0x3808001e, | ||
1038 | 0xc78, 0x3c09001e, | ||
1039 | 0xc78, 0x3e0a001e, | ||
1040 | 0xc78, 0x400b001e, | ||
1041 | 0xc78, 0x440c001e, | ||
1042 | 0xc78, 0x480d001e, | ||
1043 | 0xc78, 0x4c0e001e, | ||
1044 | 0xc78, 0x500f001e, | ||
1045 | 0xc78, 0x5210001e, | ||
1046 | 0xc78, 0x5611001e, | ||
1047 | 0xc78, 0x5a12001e, | ||
1048 | 0xc78, 0x5e13001e, | ||
1049 | 0xc78, 0x6014001e, | ||
1050 | 0xc78, 0x6015001e, | ||
1051 | 0xc78, 0x6016001e, | ||
1052 | 0xc78, 0x6217001e, | ||
1053 | 0xc78, 0x6218001e, | ||
1054 | 0xc78, 0x6219001e, | ||
1055 | 0xc78, 0x621a001e, | ||
1056 | 0xc78, 0x621b001e, | ||
1057 | 0xc78, 0x621c001e, | ||
1058 | 0xc78, 0x621d001e, | ||
1059 | 0xc78, 0x621e001e, | ||
1060 | 0xc78, 0x621f001e, | ||
1061 | }; | ||
1062 | |||
1063 | u32 RTL8192CEAGCTAB_1TARRAY[AGCTAB_1TARRAYLENGTH] = { | ||
1064 | 0xc78, 0x7b000001, | ||
1065 | 0xc78, 0x7b010001, | ||
1066 | 0xc78, 0x7b020001, | ||
1067 | 0xc78, 0x7b030001, | ||
1068 | 0xc78, 0x7b040001, | ||
1069 | 0xc78, 0x7b050001, | ||
1070 | 0xc78, 0x7a060001, | ||
1071 | 0xc78, 0x79070001, | ||
1072 | 0xc78, 0x78080001, | ||
1073 | 0xc78, 0x77090001, | ||
1074 | 0xc78, 0x760a0001, | ||
1075 | 0xc78, 0x750b0001, | ||
1076 | 0xc78, 0x740c0001, | ||
1077 | 0xc78, 0x730d0001, | ||
1078 | 0xc78, 0x720e0001, | ||
1079 | 0xc78, 0x710f0001, | ||
1080 | 0xc78, 0x70100001, | ||
1081 | 0xc78, 0x6f110001, | ||
1082 | 0xc78, 0x6e120001, | ||
1083 | 0xc78, 0x6d130001, | ||
1084 | 0xc78, 0x6c140001, | ||
1085 | 0xc78, 0x6b150001, | ||
1086 | 0xc78, 0x6a160001, | ||
1087 | 0xc78, 0x69170001, | ||
1088 | 0xc78, 0x68180001, | ||
1089 | 0xc78, 0x67190001, | ||
1090 | 0xc78, 0x661a0001, | ||
1091 | 0xc78, 0x651b0001, | ||
1092 | 0xc78, 0x641c0001, | ||
1093 | 0xc78, 0x631d0001, | ||
1094 | 0xc78, 0x621e0001, | ||
1095 | 0xc78, 0x611f0001, | ||
1096 | 0xc78, 0x60200001, | ||
1097 | 0xc78, 0x49210001, | ||
1098 | 0xc78, 0x48220001, | ||
1099 | 0xc78, 0x47230001, | ||
1100 | 0xc78, 0x46240001, | ||
1101 | 0xc78, 0x45250001, | ||
1102 | 0xc78, 0x44260001, | ||
1103 | 0xc78, 0x43270001, | ||
1104 | 0xc78, 0x42280001, | ||
1105 | 0xc78, 0x41290001, | ||
1106 | 0xc78, 0x402a0001, | ||
1107 | 0xc78, 0x262b0001, | ||
1108 | 0xc78, 0x252c0001, | ||
1109 | 0xc78, 0x242d0001, | ||
1110 | 0xc78, 0x232e0001, | ||
1111 | 0xc78, 0x222f0001, | ||
1112 | 0xc78, 0x21300001, | ||
1113 | 0xc78, 0x20310001, | ||
1114 | 0xc78, 0x06320001, | ||
1115 | 0xc78, 0x05330001, | ||
1116 | 0xc78, 0x04340001, | ||
1117 | 0xc78, 0x03350001, | ||
1118 | 0xc78, 0x02360001, | ||
1119 | 0xc78, 0x01370001, | ||
1120 | 0xc78, 0x00380001, | ||
1121 | 0xc78, 0x00390001, | ||
1122 | 0xc78, 0x003a0001, | ||
1123 | 0xc78, 0x003b0001, | ||
1124 | 0xc78, 0x003c0001, | ||
1125 | 0xc78, 0x003d0001, | ||
1126 | 0xc78, 0x003e0001, | ||
1127 | 0xc78, 0x003f0001, | ||
1128 | 0xc78, 0x7b400001, | ||
1129 | 0xc78, 0x7b410001, | ||
1130 | 0xc78, 0x7b420001, | ||
1131 | 0xc78, 0x7b430001, | ||
1132 | 0xc78, 0x7b440001, | ||
1133 | 0xc78, 0x7b450001, | ||
1134 | 0xc78, 0x7a460001, | ||
1135 | 0xc78, 0x79470001, | ||
1136 | 0xc78, 0x78480001, | ||
1137 | 0xc78, 0x77490001, | ||
1138 | 0xc78, 0x764a0001, | ||
1139 | 0xc78, 0x754b0001, | ||
1140 | 0xc78, 0x744c0001, | ||
1141 | 0xc78, 0x734d0001, | ||
1142 | 0xc78, 0x724e0001, | ||
1143 | 0xc78, 0x714f0001, | ||
1144 | 0xc78, 0x70500001, | ||
1145 | 0xc78, 0x6f510001, | ||
1146 | 0xc78, 0x6e520001, | ||
1147 | 0xc78, 0x6d530001, | ||
1148 | 0xc78, 0x6c540001, | ||
1149 | 0xc78, 0x6b550001, | ||
1150 | 0xc78, 0x6a560001, | ||
1151 | 0xc78, 0x69570001, | ||
1152 | 0xc78, 0x68580001, | ||
1153 | 0xc78, 0x67590001, | ||
1154 | 0xc78, 0x665a0001, | ||
1155 | 0xc78, 0x655b0001, | ||
1156 | 0xc78, 0x645c0001, | ||
1157 | 0xc78, 0x635d0001, | ||
1158 | 0xc78, 0x625e0001, | ||
1159 | 0xc78, 0x615f0001, | ||
1160 | 0xc78, 0x60600001, | ||
1161 | 0xc78, 0x49610001, | ||
1162 | 0xc78, 0x48620001, | ||
1163 | 0xc78, 0x47630001, | ||
1164 | 0xc78, 0x46640001, | ||
1165 | 0xc78, 0x45650001, | ||
1166 | 0xc78, 0x44660001, | ||
1167 | 0xc78, 0x43670001, | ||
1168 | 0xc78, 0x42680001, | ||
1169 | 0xc78, 0x41690001, | ||
1170 | 0xc78, 0x406a0001, | ||
1171 | 0xc78, 0x266b0001, | ||
1172 | 0xc78, 0x256c0001, | ||
1173 | 0xc78, 0x246d0001, | ||
1174 | 0xc78, 0x236e0001, | ||
1175 | 0xc78, 0x226f0001, | ||
1176 | 0xc78, 0x21700001, | ||
1177 | 0xc78, 0x20710001, | ||
1178 | 0xc78, 0x06720001, | ||
1179 | 0xc78, 0x05730001, | ||
1180 | 0xc78, 0x04740001, | ||
1181 | 0xc78, 0x03750001, | ||
1182 | 0xc78, 0x02760001, | ||
1183 | 0xc78, 0x01770001, | ||
1184 | 0xc78, 0x00780001, | ||
1185 | 0xc78, 0x00790001, | ||
1186 | 0xc78, 0x007a0001, | ||
1187 | 0xc78, 0x007b0001, | ||
1188 | 0xc78, 0x007c0001, | ||
1189 | 0xc78, 0x007d0001, | ||
1190 | 0xc78, 0x007e0001, | ||
1191 | 0xc78, 0x007f0001, | ||
1192 | 0xc78, 0x3800001e, | ||
1193 | 0xc78, 0x3801001e, | ||
1194 | 0xc78, 0x3802001e, | ||
1195 | 0xc78, 0x3803001e, | ||
1196 | 0xc78, 0x3804001e, | ||
1197 | 0xc78, 0x3805001e, | ||
1198 | 0xc78, 0x3806001e, | ||
1199 | 0xc78, 0x3807001e, | ||
1200 | 0xc78, 0x3808001e, | ||
1201 | 0xc78, 0x3c09001e, | ||
1202 | 0xc78, 0x3e0a001e, | ||
1203 | 0xc78, 0x400b001e, | ||
1204 | 0xc78, 0x440c001e, | ||
1205 | 0xc78, 0x480d001e, | ||
1206 | 0xc78, 0x4c0e001e, | ||
1207 | 0xc78, 0x500f001e, | ||
1208 | 0xc78, 0x5210001e, | ||
1209 | 0xc78, 0x5611001e, | ||
1210 | 0xc78, 0x5a12001e, | ||
1211 | 0xc78, 0x5e13001e, | ||
1212 | 0xc78, 0x6014001e, | ||
1213 | 0xc78, 0x6015001e, | ||
1214 | 0xc78, 0x6016001e, | ||
1215 | 0xc78, 0x6217001e, | ||
1216 | 0xc78, 0x6218001e, | ||
1217 | 0xc78, 0x6219001e, | ||
1218 | 0xc78, 0x621a001e, | ||
1219 | 0xc78, 0x621b001e, | ||
1220 | 0xc78, 0x621c001e, | ||
1221 | 0xc78, 0x621d001e, | ||
1222 | 0xc78, 0x621e001e, | ||
1223 | 0xc78, 0x621f001e, | ||
1224 | }; | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/table.h b/drivers/net/wireless/rtlwifi/rtl8192ce/table.h new file mode 100644 index 000000000000..3a6e8b6aeee0 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/table.h | |||
@@ -0,0 +1,58 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Created on 2010/ 5/18, 1:41 | ||
27 | * | ||
28 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
29 | * | ||
30 | *****************************************************************************/ | ||
31 | |||
32 | #ifndef __RTL92CE_TABLE__H_ | ||
33 | #define __RTL92CE_TABLE__H_ | ||
34 | |||
35 | #include <linux/types.h> | ||
36 | |||
37 | #define PHY_REG_2TARRAY_LENGTH 374 | ||
38 | extern u32 RTL8192CEPHY_REG_2TARRAY[PHY_REG_2TARRAY_LENGTH]; | ||
39 | #define PHY_REG_1TARRAY_LENGTH 374 | ||
40 | extern u32 RTL8192CEPHY_REG_1TARRAY[PHY_REG_1TARRAY_LENGTH]; | ||
41 | #define PHY_REG_ARRAY_PGLENGTH 192 | ||
42 | extern u32 RTL8192CEPHY_REG_ARRAY_PG[PHY_REG_ARRAY_PGLENGTH]; | ||
43 | #define RADIOA_2TARRAYLENGTH 282 | ||
44 | extern u32 RTL8192CERADIOA_2TARRAY[RADIOA_2TARRAYLENGTH]; | ||
45 | #define RADIOB_2TARRAYLENGTH 78 | ||
46 | extern u32 RTL8192CE_RADIOB_2TARRAY[RADIOB_2TARRAYLENGTH]; | ||
47 | #define RADIOA_1TARRAYLENGTH 282 | ||
48 | extern u32 RTL8192CE_RADIOA_1TARRAY[RADIOA_1TARRAYLENGTH]; | ||
49 | #define RADIOB_1TARRAYLENGTH 1 | ||
50 | extern u32 RTL8192CE_RADIOB_1TARRAY[RADIOB_1TARRAYLENGTH]; | ||
51 | #define MAC_2T_ARRAYLENGTH 162 | ||
52 | extern u32 RTL8192CEMAC_2T_ARRAY[MAC_2T_ARRAYLENGTH]; | ||
53 | #define AGCTAB_2TARRAYLENGTH 320 | ||
54 | extern u32 RTL8192CEAGCTAB_2TARRAY[AGCTAB_2TARRAYLENGTH]; | ||
55 | #define AGCTAB_1TARRAYLENGTH 320 | ||
56 | extern u32 RTL8192CEAGCTAB_1TARRAY[AGCTAB_1TARRAYLENGTH]; | ||
57 | |||
58 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/trx.c b/drivers/net/wireless/rtlwifi/rtl8192ce/trx.c new file mode 100644 index 000000000000..54b2bd53d36a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/trx.c | |||
@@ -0,0 +1,1012 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../base.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "phy.h" | ||
36 | #include "trx.h" | ||
37 | #include "led.h" | ||
38 | |||
39 | static u8 _rtl92ce_map_hwqueue_to_fwqueue(struct sk_buff *skb, u8 hw_queue) | ||
40 | { | ||
41 | __le16 fc = rtl_get_fc(skb); | ||
42 | |||
43 | if (unlikely(ieee80211_is_beacon(fc))) | ||
44 | return QSLT_BEACON; | ||
45 | if (ieee80211_is_mgmt(fc)) | ||
46 | return QSLT_MGNT; | ||
47 | |||
48 | return skb->priority; | ||
49 | } | ||
50 | |||
51 | static int _rtl92ce_rate_mapping(bool isht, u8 desc_rate, bool first_ampdu) | ||
52 | { | ||
53 | int rate_idx; | ||
54 | |||
55 | if (first_ampdu) { | ||
56 | if (false == isht) { | ||
57 | switch (desc_rate) { | ||
58 | case DESC92C_RATE1M: | ||
59 | rate_idx = 0; | ||
60 | break; | ||
61 | case DESC92C_RATE2M: | ||
62 | rate_idx = 1; | ||
63 | break; | ||
64 | case DESC92C_RATE5_5M: | ||
65 | rate_idx = 2; | ||
66 | break; | ||
67 | case DESC92C_RATE11M: | ||
68 | rate_idx = 3; | ||
69 | break; | ||
70 | case DESC92C_RATE6M: | ||
71 | rate_idx = 4; | ||
72 | break; | ||
73 | case DESC92C_RATE9M: | ||
74 | rate_idx = 5; | ||
75 | break; | ||
76 | case DESC92C_RATE12M: | ||
77 | rate_idx = 6; | ||
78 | break; | ||
79 | case DESC92C_RATE18M: | ||
80 | rate_idx = 7; | ||
81 | break; | ||
82 | case DESC92C_RATE24M: | ||
83 | rate_idx = 8; | ||
84 | break; | ||
85 | case DESC92C_RATE36M: | ||
86 | rate_idx = 9; | ||
87 | break; | ||
88 | case DESC92C_RATE48M: | ||
89 | rate_idx = 10; | ||
90 | break; | ||
91 | case DESC92C_RATE54M: | ||
92 | rate_idx = 11; | ||
93 | break; | ||
94 | default: | ||
95 | rate_idx = 0; | ||
96 | break; | ||
97 | } | ||
98 | } else { | ||
99 | rate_idx = 11; | ||
100 | } | ||
101 | |||
102 | return rate_idx; | ||
103 | } | ||
104 | |||
105 | switch (desc_rate) { | ||
106 | case DESC92C_RATE1M: | ||
107 | rate_idx = 0; | ||
108 | break; | ||
109 | case DESC92C_RATE2M: | ||
110 | rate_idx = 1; | ||
111 | break; | ||
112 | case DESC92C_RATE5_5M: | ||
113 | rate_idx = 2; | ||
114 | break; | ||
115 | case DESC92C_RATE11M: | ||
116 | rate_idx = 3; | ||
117 | break; | ||
118 | case DESC92C_RATE6M: | ||
119 | rate_idx = 4; | ||
120 | break; | ||
121 | case DESC92C_RATE9M: | ||
122 | rate_idx = 5; | ||
123 | break; | ||
124 | case DESC92C_RATE12M: | ||
125 | rate_idx = 6; | ||
126 | break; | ||
127 | case DESC92C_RATE18M: | ||
128 | rate_idx = 7; | ||
129 | break; | ||
130 | case DESC92C_RATE24M: | ||
131 | rate_idx = 8; | ||
132 | break; | ||
133 | case DESC92C_RATE36M: | ||
134 | rate_idx = 9; | ||
135 | break; | ||
136 | case DESC92C_RATE48M: | ||
137 | rate_idx = 10; | ||
138 | break; | ||
139 | case DESC92C_RATE54M: | ||
140 | rate_idx = 11; | ||
141 | break; | ||
142 | default: | ||
143 | rate_idx = 11; | ||
144 | break; | ||
145 | } | ||
146 | return rate_idx; | ||
147 | } | ||
148 | |||
149 | static u8 _rtl92c_query_rxpwrpercentage(char antpower) | ||
150 | { | ||
151 | if ((antpower <= -100) || (antpower >= 20)) | ||
152 | return 0; | ||
153 | else if (antpower >= 0) | ||
154 | return 100; | ||
155 | else | ||
156 | return 100 + antpower; | ||
157 | } | ||
158 | |||
159 | static u8 _rtl92c_evm_db_to_percentage(char value) | ||
160 | { | ||
161 | char ret_val; | ||
162 | ret_val = value; | ||
163 | |||
164 | if (ret_val >= 0) | ||
165 | ret_val = 0; | ||
166 | |||
167 | if (ret_val <= -33) | ||
168 | ret_val = -33; | ||
169 | |||
170 | ret_val = 0 - ret_val; | ||
171 | ret_val *= 3; | ||
172 | |||
173 | if (ret_val == 99) | ||
174 | ret_val = 100; | ||
175 | |||
176 | return ret_val; | ||
177 | } | ||
178 | |||
179 | static long _rtl92ce_translate_todbm(struct ieee80211_hw *hw, | ||
180 | u8 signal_strength_index) | ||
181 | { | ||
182 | long signal_power; | ||
183 | |||
184 | signal_power = (long)((signal_strength_index + 1) >> 1); | ||
185 | signal_power -= 95; | ||
186 | return signal_power; | ||
187 | } | ||
188 | |||
189 | static long _rtl92ce_signal_scale_mapping(struct ieee80211_hw *hw, | ||
190 | long currsig) | ||
191 | { | ||
192 | long retsig; | ||
193 | |||
194 | if (currsig >= 61 && currsig <= 100) | ||
195 | retsig = 90 + ((currsig - 60) / 4); | ||
196 | else if (currsig >= 41 && currsig <= 60) | ||
197 | retsig = 78 + ((currsig - 40) / 2); | ||
198 | else if (currsig >= 31 && currsig <= 40) | ||
199 | retsig = 66 + (currsig - 30); | ||
200 | else if (currsig >= 21 && currsig <= 30) | ||
201 | retsig = 54 + (currsig - 20); | ||
202 | else if (currsig >= 5 && currsig <= 20) | ||
203 | retsig = 42 + (((currsig - 5) * 2) / 3); | ||
204 | else if (currsig == 4) | ||
205 | retsig = 36; | ||
206 | else if (currsig == 3) | ||
207 | retsig = 27; | ||
208 | else if (currsig == 2) | ||
209 | retsig = 18; | ||
210 | else if (currsig == 1) | ||
211 | retsig = 9; | ||
212 | else | ||
213 | retsig = currsig; | ||
214 | |||
215 | return retsig; | ||
216 | } | ||
217 | |||
218 | static void _rtl92ce_query_rxphystatus(struct ieee80211_hw *hw, | ||
219 | struct rtl_stats *pstats, | ||
220 | struct rx_desc_92c *pdesc, | ||
221 | struct rx_fwinfo_92c *p_drvinfo, | ||
222 | bool packet_match_bssid, | ||
223 | bool packet_toself, | ||
224 | bool packet_beacon) | ||
225 | { | ||
226 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
227 | struct phy_sts_cck_8192s_t *cck_buf; | ||
228 | s8 rx_pwr_all, rx_pwr[4]; | ||
229 | u8 evm, pwdb_all, rf_rx_num = 0; | ||
230 | u8 i, max_spatial_stream; | ||
231 | u32 rssi, total_rssi = 0; | ||
232 | bool in_powersavemode = false; | ||
233 | bool is_cck_rate; | ||
234 | |||
235 | is_cck_rate = RX_HAL_IS_CCK_RATE(pdesc); | ||
236 | pstats->packet_matchbssid = packet_match_bssid; | ||
237 | pstats->packet_toself = packet_toself; | ||
238 | pstats->is_cck = is_cck_rate; | ||
239 | pstats->packet_beacon = packet_beacon; | ||
240 | pstats->is_cck = is_cck_rate; | ||
241 | pstats->rx_mimo_signalquality[0] = -1; | ||
242 | pstats->rx_mimo_signalquality[1] = -1; | ||
243 | |||
244 | if (is_cck_rate) { | ||
245 | u8 report, cck_highpwr; | ||
246 | cck_buf = (struct phy_sts_cck_8192s_t *)p_drvinfo; | ||
247 | |||
248 | if (!in_powersavemode) | ||
249 | cck_highpwr = (u8) rtl_get_bbreg(hw, | ||
250 | RFPGA0_XA_HSSIPARAMETER2, | ||
251 | BIT(9)); | ||
252 | else | ||
253 | cck_highpwr = false; | ||
254 | |||
255 | if (!cck_highpwr) { | ||
256 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
257 | report = cck_buf->cck_agc_rpt & 0xc0; | ||
258 | report = report >> 6; | ||
259 | switch (report) { | ||
260 | case 0x3: | ||
261 | rx_pwr_all = -46 - (cck_agc_rpt & 0x3e); | ||
262 | break; | ||
263 | case 0x2: | ||
264 | rx_pwr_all = -26 - (cck_agc_rpt & 0x3e); | ||
265 | break; | ||
266 | case 0x1: | ||
267 | rx_pwr_all = -12 - (cck_agc_rpt & 0x3e); | ||
268 | break; | ||
269 | case 0x0: | ||
270 | rx_pwr_all = 16 - (cck_agc_rpt & 0x3e); | ||
271 | break; | ||
272 | } | ||
273 | } else { | ||
274 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
275 | report = p_drvinfo->cfosho[0] & 0x60; | ||
276 | report = report >> 5; | ||
277 | switch (report) { | ||
278 | case 0x3: | ||
279 | rx_pwr_all = -46 - ((cck_agc_rpt & 0x1f) << 1); | ||
280 | break; | ||
281 | case 0x2: | ||
282 | rx_pwr_all = -26 - ((cck_agc_rpt & 0x1f) << 1); | ||
283 | break; | ||
284 | case 0x1: | ||
285 | rx_pwr_all = -12 - ((cck_agc_rpt & 0x1f) << 1); | ||
286 | break; | ||
287 | case 0x0: | ||
288 | rx_pwr_all = 16 - ((cck_agc_rpt & 0x1f) << 1); | ||
289 | break; | ||
290 | } | ||
291 | } | ||
292 | |||
293 | pwdb_all = _rtl92c_query_rxpwrpercentage(rx_pwr_all); | ||
294 | pstats->rx_pwdb_all = pwdb_all; | ||
295 | pstats->recvsignalpower = rx_pwr_all; | ||
296 | |||
297 | if (packet_match_bssid) { | ||
298 | u8 sq; | ||
299 | if (pstats->rx_pwdb_all > 40) | ||
300 | sq = 100; | ||
301 | else { | ||
302 | sq = cck_buf->sq_rpt; | ||
303 | if (sq > 64) | ||
304 | sq = 0; | ||
305 | else if (sq < 20) | ||
306 | sq = 100; | ||
307 | else | ||
308 | sq = ((64 - sq) * 100) / 44; | ||
309 | } | ||
310 | |||
311 | pstats->signalquality = sq; | ||
312 | pstats->rx_mimo_signalquality[0] = sq; | ||
313 | pstats->rx_mimo_signalquality[1] = -1; | ||
314 | } | ||
315 | } else { | ||
316 | rtlpriv->dm.rfpath_rxenable[0] = | ||
317 | rtlpriv->dm.rfpath_rxenable[1] = true; | ||
318 | for (i = RF90_PATH_A; i < RF90_PATH_MAX; i++) { | ||
319 | if (rtlpriv->dm.rfpath_rxenable[i]) | ||
320 | rf_rx_num++; | ||
321 | |||
322 | rx_pwr[i] = | ||
323 | ((p_drvinfo->gain_trsw[i] & 0x3f) * 2) - 110; | ||
324 | rssi = _rtl92c_query_rxpwrpercentage(rx_pwr[i]); | ||
325 | total_rssi += rssi; | ||
326 | rtlpriv->stats.rx_snr_db[i] = | ||
327 | (long)(p_drvinfo->rxsnr[i] / 2); | ||
328 | |||
329 | if (packet_match_bssid) | ||
330 | pstats->rx_mimo_signalstrength[i] = (u8) rssi; | ||
331 | } | ||
332 | |||
333 | rx_pwr_all = ((p_drvinfo->pwdb_all >> 1) & 0x7f) - 110; | ||
334 | pwdb_all = _rtl92c_query_rxpwrpercentage(rx_pwr_all); | ||
335 | pstats->rx_pwdb_all = pwdb_all; | ||
336 | pstats->rxpower = rx_pwr_all; | ||
337 | pstats->recvsignalpower = rx_pwr_all; | ||
338 | |||
339 | if (pdesc->rxht && pdesc->rxmcs >= DESC92C_RATEMCS8 && | ||
340 | pdesc->rxmcs <= DESC92C_RATEMCS15) | ||
341 | max_spatial_stream = 2; | ||
342 | else | ||
343 | max_spatial_stream = 1; | ||
344 | |||
345 | for (i = 0; i < max_spatial_stream; i++) { | ||
346 | evm = _rtl92c_evm_db_to_percentage(p_drvinfo->rxevm[i]); | ||
347 | |||
348 | if (packet_match_bssid) { | ||
349 | if (i == 0) | ||
350 | pstats->signalquality = | ||
351 | (u8) (evm & 0xff); | ||
352 | pstats->rx_mimo_signalquality[i] = | ||
353 | (u8) (evm & 0xff); | ||
354 | } | ||
355 | } | ||
356 | } | ||
357 | |||
358 | if (is_cck_rate) | ||
359 | pstats->signalstrength = | ||
360 | (u8) (_rtl92ce_signal_scale_mapping(hw, pwdb_all)); | ||
361 | else if (rf_rx_num != 0) | ||
362 | pstats->signalstrength = | ||
363 | (u8) (_rtl92ce_signal_scale_mapping | ||
364 | (hw, total_rssi /= rf_rx_num)); | ||
365 | } | ||
366 | |||
367 | static void _rtl92ce_process_ui_rssi(struct ieee80211_hw *hw, | ||
368 | struct rtl_stats *pstats) | ||
369 | { | ||
370 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
371 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
372 | u8 rfpath; | ||
373 | u32 last_rssi, tmpval; | ||
374 | |||
375 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
376 | rtlpriv->stats.rssi_calculate_cnt++; | ||
377 | |||
378 | if (rtlpriv->stats.ui_rssi.total_num++ >= | ||
379 | PHY_RSSI_SLID_WIN_MAX) { | ||
380 | |||
381 | rtlpriv->stats.ui_rssi.total_num = | ||
382 | PHY_RSSI_SLID_WIN_MAX; | ||
383 | last_rssi = | ||
384 | rtlpriv->stats.ui_rssi.elements[rtlpriv-> | ||
385 | stats.ui_rssi.index]; | ||
386 | rtlpriv->stats.ui_rssi.total_val -= last_rssi; | ||
387 | } | ||
388 | |||
389 | rtlpriv->stats.ui_rssi.total_val += pstats->signalstrength; | ||
390 | rtlpriv->stats.ui_rssi.elements[rtlpriv->stats.ui_rssi. | ||
391 | index++] = | ||
392 | pstats->signalstrength; | ||
393 | |||
394 | if (rtlpriv->stats.ui_rssi.index >= PHY_RSSI_SLID_WIN_MAX) | ||
395 | rtlpriv->stats.ui_rssi.index = 0; | ||
396 | |||
397 | tmpval = rtlpriv->stats.ui_rssi.total_val / | ||
398 | rtlpriv->stats.ui_rssi.total_num; | ||
399 | rtlpriv->stats.signal_strength = | ||
400 | _rtl92ce_translate_todbm(hw, (u8) tmpval); | ||
401 | pstats->rssi = rtlpriv->stats.signal_strength; | ||
402 | } | ||
403 | |||
404 | if (!pstats->is_cck && pstats->packet_toself) { | ||
405 | for (rfpath = RF90_PATH_A; rfpath < rtlphy->num_total_rfpath; | ||
406 | rfpath++) { | ||
407 | if (rtlpriv->stats.rx_rssi_percentage[rfpath] == 0) { | ||
408 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
409 | pstats->rx_mimo_signalstrength[rfpath]; | ||
410 | |||
411 | } | ||
412 | |||
413 | if (pstats->rx_mimo_signalstrength[rfpath] > | ||
414 | rtlpriv->stats.rx_rssi_percentage[rfpath]) { | ||
415 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
416 | ((rtlpriv->stats. | ||
417 | rx_rssi_percentage[rfpath] * | ||
418 | (RX_SMOOTH_FACTOR - 1)) + | ||
419 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
420 | (RX_SMOOTH_FACTOR); | ||
421 | |||
422 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
423 | rtlpriv->stats.rx_rssi_percentage[rfpath] + | ||
424 | 1; | ||
425 | } else { | ||
426 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
427 | ((rtlpriv->stats. | ||
428 | rx_rssi_percentage[rfpath] * | ||
429 | (RX_SMOOTH_FACTOR - 1)) + | ||
430 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
431 | (RX_SMOOTH_FACTOR); | ||
432 | } | ||
433 | |||
434 | } | ||
435 | } | ||
436 | } | ||
437 | |||
438 | static void _rtl92ce_update_rxsignalstatistics(struct ieee80211_hw *hw, | ||
439 | struct rtl_stats *pstats) | ||
440 | { | ||
441 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
442 | int weighting = 0; | ||
443 | |||
444 | if (rtlpriv->stats.recv_signal_power == 0) | ||
445 | rtlpriv->stats.recv_signal_power = pstats->recvsignalpower; | ||
446 | |||
447 | if (pstats->recvsignalpower > rtlpriv->stats.recv_signal_power) | ||
448 | weighting = 5; | ||
449 | |||
450 | else if (pstats->recvsignalpower < rtlpriv->stats.recv_signal_power) | ||
451 | weighting = (-5); | ||
452 | |||
453 | rtlpriv->stats.recv_signal_power = | ||
454 | (rtlpriv->stats.recv_signal_power * 5 + | ||
455 | pstats->recvsignalpower + weighting) / 6; | ||
456 | } | ||
457 | |||
458 | static void _rtl92ce_process_pwdb(struct ieee80211_hw *hw, | ||
459 | struct rtl_stats *pstats) | ||
460 | { | ||
461 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
462 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
463 | long undecorated_smoothed_pwdb; | ||
464 | |||
465 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
466 | return; | ||
467 | } else { | ||
468 | undecorated_smoothed_pwdb = | ||
469 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
470 | } | ||
471 | |||
472 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
473 | if (undecorated_smoothed_pwdb < 0) | ||
474 | undecorated_smoothed_pwdb = pstats->rx_pwdb_all; | ||
475 | |||
476 | if (pstats->rx_pwdb_all > (u32) undecorated_smoothed_pwdb) { | ||
477 | undecorated_smoothed_pwdb = | ||
478 | (((undecorated_smoothed_pwdb) * | ||
479 | (RX_SMOOTH_FACTOR - 1)) + | ||
480 | (pstats->rx_pwdb_all)) / (RX_SMOOTH_FACTOR); | ||
481 | |||
482 | undecorated_smoothed_pwdb = undecorated_smoothed_pwdb | ||
483 | + 1; | ||
484 | } else { | ||
485 | undecorated_smoothed_pwdb = | ||
486 | (((undecorated_smoothed_pwdb) * | ||
487 | (RX_SMOOTH_FACTOR - 1)) + | ||
488 | (pstats->rx_pwdb_all)) / (RX_SMOOTH_FACTOR); | ||
489 | } | ||
490 | |||
491 | rtlpriv->dm.undecorated_smoothed_pwdb = | ||
492 | undecorated_smoothed_pwdb; | ||
493 | _rtl92ce_update_rxsignalstatistics(hw, pstats); | ||
494 | } | ||
495 | } | ||
496 | |||
497 | static void _rtl92ce_process_ui_link_quality(struct ieee80211_hw *hw, | ||
498 | struct rtl_stats *pstats) | ||
499 | { | ||
500 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
501 | u32 last_evm, n_spatialstream, tmpval; | ||
502 | |||
503 | if (pstats->signalquality != 0) { | ||
504 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
505 | |||
506 | if (rtlpriv->stats.ui_link_quality.total_num++ >= | ||
507 | PHY_LINKQUALITY_SLID_WIN_MAX) { | ||
508 | rtlpriv->stats.ui_link_quality.total_num = | ||
509 | PHY_LINKQUALITY_SLID_WIN_MAX; | ||
510 | last_evm = | ||
511 | rtlpriv->stats. | ||
512 | ui_link_quality.elements[rtlpriv-> | ||
513 | stats.ui_link_quality. | ||
514 | index]; | ||
515 | rtlpriv->stats.ui_link_quality.total_val -= | ||
516 | last_evm; | ||
517 | } | ||
518 | |||
519 | rtlpriv->stats.ui_link_quality.total_val += | ||
520 | pstats->signalquality; | ||
521 | rtlpriv->stats.ui_link_quality.elements[rtlpriv->stats. | ||
522 | ui_link_quality. | ||
523 | index++] = | ||
524 | pstats->signalquality; | ||
525 | |||
526 | if (rtlpriv->stats.ui_link_quality.index >= | ||
527 | PHY_LINKQUALITY_SLID_WIN_MAX) | ||
528 | rtlpriv->stats.ui_link_quality.index = 0; | ||
529 | |||
530 | tmpval = rtlpriv->stats.ui_link_quality.total_val / | ||
531 | rtlpriv->stats.ui_link_quality.total_num; | ||
532 | rtlpriv->stats.signal_quality = tmpval; | ||
533 | |||
534 | rtlpriv->stats.last_sigstrength_inpercent = tmpval; | ||
535 | |||
536 | for (n_spatialstream = 0; n_spatialstream < 2; | ||
537 | n_spatialstream++) { | ||
538 | if (pstats-> | ||
539 | rx_mimo_signalquality[n_spatialstream] != | ||
540 | -1) { | ||
541 | if (rtlpriv->stats. | ||
542 | rx_evm_percentage[n_spatialstream] | ||
543 | == 0) { | ||
544 | rtlpriv->stats. | ||
545 | rx_evm_percentage | ||
546 | [n_spatialstream] = | ||
547 | pstats->rx_mimo_signalquality | ||
548 | [n_spatialstream]; | ||
549 | } | ||
550 | |||
551 | rtlpriv->stats. | ||
552 | rx_evm_percentage[n_spatialstream] = | ||
553 | ((rtlpriv-> | ||
554 | stats.rx_evm_percentage | ||
555 | [n_spatialstream] * | ||
556 | (RX_SMOOTH_FACTOR - 1)) + | ||
557 | (pstats-> | ||
558 | rx_mimo_signalquality | ||
559 | [n_spatialstream] * 1)) / | ||
560 | (RX_SMOOTH_FACTOR); | ||
561 | } | ||
562 | } | ||
563 | } | ||
564 | } else { | ||
565 | ; | ||
566 | } | ||
567 | } | ||
568 | |||
569 | static void _rtl92ce_process_phyinfo(struct ieee80211_hw *hw, | ||
570 | u8 *buffer, | ||
571 | struct rtl_stats *pcurrent_stats) | ||
572 | { | ||
573 | |||
574 | if (!pcurrent_stats->packet_matchbssid && | ||
575 | !pcurrent_stats->packet_beacon) | ||
576 | return; | ||
577 | |||
578 | _rtl92ce_process_ui_rssi(hw, pcurrent_stats); | ||
579 | _rtl92ce_process_pwdb(hw, pcurrent_stats); | ||
580 | _rtl92ce_process_ui_link_quality(hw, pcurrent_stats); | ||
581 | } | ||
582 | |||
583 | static void _rtl92ce_translate_rx_signal_stuff(struct ieee80211_hw *hw, | ||
584 | struct sk_buff *skb, | ||
585 | struct rtl_stats *pstats, | ||
586 | struct rx_desc_92c *pdesc, | ||
587 | struct rx_fwinfo_92c *p_drvinfo) | ||
588 | { | ||
589 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
590 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
591 | |||
592 | struct ieee80211_hdr *hdr; | ||
593 | u8 *tmp_buf; | ||
594 | u8 *praddr; | ||
595 | u8 *psaddr; | ||
596 | __le16 fc; | ||
597 | u16 type, c_fc; | ||
598 | bool packet_matchbssid, packet_toself, packet_beacon; | ||
599 | |||
600 | tmp_buf = skb->data + pstats->rx_drvinfo_size + pstats->rx_bufshift; | ||
601 | |||
602 | hdr = (struct ieee80211_hdr *)tmp_buf; | ||
603 | fc = hdr->frame_control; | ||
604 | c_fc = le16_to_cpu(fc); | ||
605 | type = WLAN_FC_GET_TYPE(fc); | ||
606 | praddr = hdr->addr1; | ||
607 | psaddr = hdr->addr2; | ||
608 | |||
609 | packet_matchbssid = | ||
610 | ((IEEE80211_FTYPE_CTL != type) && | ||
611 | (!compare_ether_addr(mac->bssid, | ||
612 | (c_fc & IEEE80211_FCTL_TODS) ? | ||
613 | hdr->addr1 : (c_fc & IEEE80211_FCTL_FROMDS) ? | ||
614 | hdr->addr2 : hdr->addr3)) && | ||
615 | (!pstats->hwerror) && (!pstats->crc) && (!pstats->icv)); | ||
616 | |||
617 | packet_toself = packet_matchbssid && | ||
618 | (!compare_ether_addr(praddr, rtlefuse->dev_addr)); | ||
619 | |||
620 | if (ieee80211_is_beacon(fc)) | ||
621 | packet_beacon = true; | ||
622 | |||
623 | _rtl92ce_query_rxphystatus(hw, pstats, pdesc, p_drvinfo, | ||
624 | packet_matchbssid, packet_toself, | ||
625 | packet_beacon); | ||
626 | |||
627 | _rtl92ce_process_phyinfo(hw, tmp_buf, pstats); | ||
628 | } | ||
629 | |||
630 | bool rtl92ce_rx_query_desc(struct ieee80211_hw *hw, | ||
631 | struct rtl_stats *stats, | ||
632 | struct ieee80211_rx_status *rx_status, | ||
633 | u8 *p_desc, struct sk_buff *skb) | ||
634 | { | ||
635 | struct rx_fwinfo_92c *p_drvinfo; | ||
636 | struct rx_desc_92c *pdesc = (struct rx_desc_92c *)p_desc; | ||
637 | |||
638 | u32 phystatus = GET_RX_DESC_PHYST(pdesc); | ||
639 | stats->length = (u16) GET_RX_DESC_PKT_LEN(pdesc); | ||
640 | stats->rx_drvinfo_size = (u8) GET_RX_DESC_DRV_INFO_SIZE(pdesc) * | ||
641 | RX_DRV_INFO_SIZE_UNIT; | ||
642 | stats->rx_bufshift = (u8) (GET_RX_DESC_SHIFT(pdesc) & 0x03); | ||
643 | stats->icv = (u16) GET_RX_DESC_ICV(pdesc); | ||
644 | stats->crc = (u16) GET_RX_DESC_CRC32(pdesc); | ||
645 | stats->hwerror = (stats->crc | stats->icv); | ||
646 | stats->decrypted = !GET_RX_DESC_SWDEC(pdesc); | ||
647 | stats->rate = (u8) GET_RX_DESC_RXMCS(pdesc); | ||
648 | stats->shortpreamble = (u16) GET_RX_DESC_SPLCP(pdesc); | ||
649 | stats->isampdu = (bool) (GET_RX_DESC_PAGGR(pdesc) == 1); | ||
650 | stats->isampdu = (bool) ((GET_RX_DESC_PAGGR(pdesc) == 1) | ||
651 | && (GET_RX_DESC_FAGGR(pdesc) == 1)); | ||
652 | stats->timestamp_low = GET_RX_DESC_TSFL(pdesc); | ||
653 | stats->rx_is40Mhzpacket = (bool) GET_RX_DESC_BW(pdesc); | ||
654 | |||
655 | rx_status->freq = hw->conf.channel->center_freq; | ||
656 | rx_status->band = hw->conf.channel->band; | ||
657 | |||
658 | if (GET_RX_DESC_CRC32(pdesc)) | ||
659 | rx_status->flag |= RX_FLAG_FAILED_FCS_CRC; | ||
660 | |||
661 | if (!GET_RX_DESC_SWDEC(pdesc)) | ||
662 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
663 | |||
664 | if (GET_RX_DESC_BW(pdesc)) | ||
665 | rx_status->flag |= RX_FLAG_40MHZ; | ||
666 | |||
667 | if (GET_RX_DESC_RXHT(pdesc)) | ||
668 | rx_status->flag |= RX_FLAG_HT; | ||
669 | |||
670 | rx_status->flag |= RX_FLAG_MACTIME_MPDU; | ||
671 | |||
672 | if (stats->decrypted) | ||
673 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
674 | |||
675 | rx_status->rate_idx = _rtl92ce_rate_mapping((bool) | ||
676 | GET_RX_DESC_RXHT(pdesc), | ||
677 | (u8) | ||
678 | GET_RX_DESC_RXMCS(pdesc), | ||
679 | (bool) | ||
680 | GET_RX_DESC_PAGGR(pdesc)); | ||
681 | |||
682 | rx_status->mactime = GET_RX_DESC_TSFL(pdesc); | ||
683 | if (phystatus == true) { | ||
684 | p_drvinfo = (struct rx_fwinfo_92c *)(skb->data + | ||
685 | stats->rx_bufshift); | ||
686 | |||
687 | _rtl92ce_translate_rx_signal_stuff(hw, | ||
688 | skb, stats, pdesc, | ||
689 | p_drvinfo); | ||
690 | } | ||
691 | |||
692 | /*rx_status->qual = stats->signal; */ | ||
693 | rx_status->signal = stats->rssi + 10; | ||
694 | /*rx_status->noise = -stats->noise; */ | ||
695 | |||
696 | return true; | ||
697 | } | ||
698 | |||
699 | void rtl92ce_tx_fill_desc(struct ieee80211_hw *hw, | ||
700 | struct ieee80211_hdr *hdr, u8 *pdesc_tx, | ||
701 | struct ieee80211_tx_info *info, struct sk_buff *skb, | ||
702 | u8 hw_queue, struct rtl_tcb_desc *tcb_desc) | ||
703 | { | ||
704 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
705 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
706 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
707 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
708 | bool defaultadapter = true; | ||
709 | struct ieee80211_sta *sta; | ||
710 | u8 *pdesc = (u8 *) pdesc_tx; | ||
711 | u16 seq_number; | ||
712 | __le16 fc = hdr->frame_control; | ||
713 | u8 fw_qsel = _rtl92ce_map_hwqueue_to_fwqueue(skb, hw_queue); | ||
714 | bool firstseg = ((hdr->seq_ctrl & | ||
715 | cpu_to_le16(IEEE80211_SCTL_FRAG)) == 0); | ||
716 | |||
717 | bool lastseg = ((hdr->frame_control & | ||
718 | cpu_to_le16(IEEE80211_FCTL_MOREFRAGS)) == 0); | ||
719 | |||
720 | dma_addr_t mapping = pci_map_single(rtlpci->pdev, | ||
721 | skb->data, skb->len, | ||
722 | PCI_DMA_TODEVICE); | ||
723 | u8 bw_40 = 0; | ||
724 | |||
725 | rcu_read_lock(); | ||
726 | sta = get_sta(hw, mac->vif, mac->bssid); | ||
727 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
728 | bw_40 = mac->bw_40; | ||
729 | } else if (mac->opmode == NL80211_IFTYPE_AP || | ||
730 | mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
731 | if (sta) | ||
732 | bw_40 = sta->ht_cap.cap & | ||
733 | IEEE80211_HT_CAP_SUP_WIDTH_20_40; | ||
734 | } | ||
735 | |||
736 | seq_number = (le16_to_cpu(hdr->seq_ctrl) & IEEE80211_SCTL_SEQ) >> 4; | ||
737 | |||
738 | rtl_get_tcb_desc(hw, info, sta, skb, tcb_desc); | ||
739 | |||
740 | CLEAR_PCI_TX_DESC_CONTENT(pdesc, sizeof(struct tx_desc_92c)); | ||
741 | |||
742 | if (ieee80211_is_nullfunc(fc) || ieee80211_is_ctl(fc)) { | ||
743 | firstseg = true; | ||
744 | lastseg = true; | ||
745 | } | ||
746 | if (firstseg) { | ||
747 | SET_TX_DESC_OFFSET(pdesc, USB_HWDESC_HEADER_LEN); | ||
748 | |||
749 | SET_TX_DESC_TX_RATE(pdesc, tcb_desc->hw_rate); | ||
750 | |||
751 | if (tcb_desc->use_shortgi || tcb_desc->use_shortpreamble) | ||
752 | SET_TX_DESC_DATA_SHORTGI(pdesc, 1); | ||
753 | |||
754 | if (info->flags & IEEE80211_TX_CTL_AMPDU) { | ||
755 | SET_TX_DESC_AGG_BREAK(pdesc, 1); | ||
756 | SET_TX_DESC_MAX_AGG_NUM(pdesc, 0x14); | ||
757 | } | ||
758 | SET_TX_DESC_SEQ(pdesc, seq_number); | ||
759 | |||
760 | SET_TX_DESC_RTS_ENABLE(pdesc, ((tcb_desc->rts_enable && | ||
761 | !tcb_desc-> | ||
762 | cts_enable) ? 1 : 0)); | ||
763 | SET_TX_DESC_HW_RTS_ENABLE(pdesc, | ||
764 | ((tcb_desc->rts_enable | ||
765 | || tcb_desc->cts_enable) ? 1 : 0)); | ||
766 | SET_TX_DESC_CTS2SELF(pdesc, ((tcb_desc->cts_enable) ? 1 : 0)); | ||
767 | SET_TX_DESC_RTS_STBC(pdesc, ((tcb_desc->rts_stbc) ? 1 : 0)); | ||
768 | |||
769 | SET_TX_DESC_RTS_RATE(pdesc, tcb_desc->rts_rate); | ||
770 | SET_TX_DESC_RTS_BW(pdesc, 0); | ||
771 | SET_TX_DESC_RTS_SC(pdesc, tcb_desc->rts_sc); | ||
772 | SET_TX_DESC_RTS_SHORT(pdesc, | ||
773 | ((tcb_desc->rts_rate <= DESC92C_RATE54M) ? | ||
774 | (tcb_desc->rts_use_shortpreamble ? 1 : 0) | ||
775 | : (tcb_desc->rts_use_shortgi ? 1 : 0))); | ||
776 | |||
777 | if (bw_40) { | ||
778 | if (tcb_desc->packet_bw) { | ||
779 | SET_TX_DESC_DATA_BW(pdesc, 1); | ||
780 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, 3); | ||
781 | } else { | ||
782 | SET_TX_DESC_DATA_BW(pdesc, 0); | ||
783 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, | ||
784 | mac->cur_40_prime_sc); | ||
785 | } | ||
786 | } else { | ||
787 | SET_TX_DESC_DATA_BW(pdesc, 0); | ||
788 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, 0); | ||
789 | } | ||
790 | |||
791 | SET_TX_DESC_LINIP(pdesc, 0); | ||
792 | SET_TX_DESC_PKT_SIZE(pdesc, (u16) skb->len); | ||
793 | |||
794 | if (sta) { | ||
795 | u8 ampdu_density = sta->ht_cap.ampdu_density; | ||
796 | SET_TX_DESC_AMPDU_DENSITY(pdesc, ampdu_density); | ||
797 | } | ||
798 | |||
799 | if (info->control.hw_key) { | ||
800 | struct ieee80211_key_conf *keyconf = | ||
801 | info->control.hw_key; | ||
802 | |||
803 | switch (keyconf->cipher) { | ||
804 | case WLAN_CIPHER_SUITE_WEP40: | ||
805 | case WLAN_CIPHER_SUITE_WEP104: | ||
806 | case WLAN_CIPHER_SUITE_TKIP: | ||
807 | SET_TX_DESC_SEC_TYPE(pdesc, 0x1); | ||
808 | break; | ||
809 | case WLAN_CIPHER_SUITE_CCMP: | ||
810 | SET_TX_DESC_SEC_TYPE(pdesc, 0x3); | ||
811 | break; | ||
812 | default: | ||
813 | SET_TX_DESC_SEC_TYPE(pdesc, 0x0); | ||
814 | break; | ||
815 | |||
816 | } | ||
817 | } | ||
818 | |||
819 | SET_TX_DESC_PKT_ID(pdesc, 0); | ||
820 | SET_TX_DESC_QUEUE_SEL(pdesc, fw_qsel); | ||
821 | |||
822 | SET_TX_DESC_DATA_RATE_FB_LIMIT(pdesc, 0x1F); | ||
823 | SET_TX_DESC_RTS_RATE_FB_LIMIT(pdesc, 0xF); | ||
824 | SET_TX_DESC_DISABLE_FB(pdesc, 0); | ||
825 | SET_TX_DESC_USE_RATE(pdesc, tcb_desc->use_driver_rate ? 1 : 0); | ||
826 | |||
827 | if (ieee80211_is_data_qos(fc)) { | ||
828 | if (mac->rdg_en) { | ||
829 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, | ||
830 | ("Enable RDG function.\n")); | ||
831 | SET_TX_DESC_RDG_ENABLE(pdesc, 1); | ||
832 | SET_TX_DESC_HTC(pdesc, 1); | ||
833 | } | ||
834 | } | ||
835 | } | ||
836 | rcu_read_unlock(); | ||
837 | |||
838 | SET_TX_DESC_FIRST_SEG(pdesc, (firstseg ? 1 : 0)); | ||
839 | SET_TX_DESC_LAST_SEG(pdesc, (lastseg ? 1 : 0)); | ||
840 | |||
841 | SET_TX_DESC_TX_BUFFER_SIZE(pdesc, (u16) skb->len); | ||
842 | |||
843 | SET_TX_DESC_TX_BUFFER_ADDRESS(pdesc, mapping); | ||
844 | |||
845 | if (rtlpriv->dm.useramask) { | ||
846 | SET_TX_DESC_RATE_ID(pdesc, tcb_desc->ratr_index); | ||
847 | SET_TX_DESC_MACID(pdesc, tcb_desc->mac_id); | ||
848 | } else { | ||
849 | SET_TX_DESC_RATE_ID(pdesc, 0xC + tcb_desc->ratr_index); | ||
850 | SET_TX_DESC_MACID(pdesc, tcb_desc->ratr_index); | ||
851 | } | ||
852 | |||
853 | if ((!ieee80211_is_data_qos(fc)) && ppsc->fwctrl_lps) { | ||
854 | SET_TX_DESC_HWSEQ_EN(pdesc, 1); | ||
855 | SET_TX_DESC_PKT_ID(pdesc, 8); | ||
856 | |||
857 | if (!defaultadapter) | ||
858 | SET_TX_DESC_QOS(pdesc, 1); | ||
859 | } | ||
860 | |||
861 | SET_TX_DESC_MORE_FRAG(pdesc, (lastseg ? 0 : 1)); | ||
862 | |||
863 | if (is_multicast_ether_addr(ieee80211_get_DA(hdr)) || | ||
864 | is_broadcast_ether_addr(ieee80211_get_DA(hdr))) { | ||
865 | SET_TX_DESC_BMC(pdesc, 1); | ||
866 | } | ||
867 | |||
868 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, ("\n")); | ||
869 | } | ||
870 | |||
871 | void rtl92ce_tx_fill_cmddesc(struct ieee80211_hw *hw, | ||
872 | u8 *pdesc, bool firstseg, | ||
873 | bool lastseg, struct sk_buff *skb) | ||
874 | { | ||
875 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
876 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
877 | u8 fw_queue = QSLT_BEACON; | ||
878 | |||
879 | dma_addr_t mapping = pci_map_single(rtlpci->pdev, | ||
880 | skb->data, skb->len, | ||
881 | PCI_DMA_TODEVICE); | ||
882 | |||
883 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)(skb->data); | ||
884 | __le16 fc = hdr->frame_control; | ||
885 | |||
886 | CLEAR_PCI_TX_DESC_CONTENT(pdesc, TX_DESC_SIZE); | ||
887 | |||
888 | if (firstseg) | ||
889 | SET_TX_DESC_OFFSET(pdesc, USB_HWDESC_HEADER_LEN); | ||
890 | |||
891 | SET_TX_DESC_TX_RATE(pdesc, DESC92C_RATE1M); | ||
892 | |||
893 | SET_TX_DESC_SEQ(pdesc, 0); | ||
894 | |||
895 | SET_TX_DESC_LINIP(pdesc, 0); | ||
896 | |||
897 | SET_TX_DESC_QUEUE_SEL(pdesc, fw_queue); | ||
898 | |||
899 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
900 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
901 | |||
902 | SET_TX_DESC_TX_BUFFER_SIZE(pdesc, (u16) (skb->len)); | ||
903 | |||
904 | SET_TX_DESC_TX_BUFFER_ADDRESS(pdesc, mapping); | ||
905 | |||
906 | SET_TX_DESC_RATE_ID(pdesc, 7); | ||
907 | SET_TX_DESC_MACID(pdesc, 0); | ||
908 | |||
909 | SET_TX_DESC_OWN(pdesc, 1); | ||
910 | |||
911 | SET_TX_DESC_PKT_SIZE((u8 *) pdesc, (u16) (skb->len)); | ||
912 | |||
913 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
914 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
915 | |||
916 | SET_TX_DESC_OFFSET(pdesc, 0x20); | ||
917 | |||
918 | SET_TX_DESC_USE_RATE(pdesc, 1); | ||
919 | |||
920 | if (!ieee80211_is_data_qos(fc)) { | ||
921 | SET_TX_DESC_HWSEQ_EN(pdesc, 1); | ||
922 | SET_TX_DESC_PKT_ID(pdesc, 8); | ||
923 | } | ||
924 | |||
925 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_LOUD, | ||
926 | "H2C Tx Cmd Content\n", | ||
927 | pdesc, TX_DESC_SIZE); | ||
928 | } | ||
929 | |||
930 | void rtl92ce_set_desc(u8 *pdesc, bool istx, u8 desc_name, u8 *val) | ||
931 | { | ||
932 | if (istx == true) { | ||
933 | switch (desc_name) { | ||
934 | case HW_DESC_OWN: | ||
935 | SET_TX_DESC_OWN(pdesc, 1); | ||
936 | break; | ||
937 | case HW_DESC_TX_NEXTDESC_ADDR: | ||
938 | SET_TX_DESC_NEXT_DESC_ADDRESS(pdesc, *(u32 *) val); | ||
939 | break; | ||
940 | default: | ||
941 | RT_ASSERT(false, ("ERR txdesc :%d" | ||
942 | " not process\n", desc_name)); | ||
943 | break; | ||
944 | } | ||
945 | } else { | ||
946 | switch (desc_name) { | ||
947 | case HW_DESC_RXOWN: | ||
948 | SET_RX_DESC_OWN(pdesc, 1); | ||
949 | break; | ||
950 | case HW_DESC_RXBUFF_ADDR: | ||
951 | SET_RX_DESC_BUFF_ADDR(pdesc, *(u32 *) val); | ||
952 | break; | ||
953 | case HW_DESC_RXPKT_LEN: | ||
954 | SET_RX_DESC_PKT_LEN(pdesc, *(u32 *) val); | ||
955 | break; | ||
956 | case HW_DESC_RXERO: | ||
957 | SET_RX_DESC_EOR(pdesc, 1); | ||
958 | break; | ||
959 | default: | ||
960 | RT_ASSERT(false, ("ERR rxdesc :%d " | ||
961 | "not process\n", desc_name)); | ||
962 | break; | ||
963 | } | ||
964 | } | ||
965 | } | ||
966 | |||
967 | u32 rtl92ce_get_desc(u8 *p_desc, bool istx, u8 desc_name) | ||
968 | { | ||
969 | u32 ret = 0; | ||
970 | |||
971 | if (istx == true) { | ||
972 | switch (desc_name) { | ||
973 | case HW_DESC_OWN: | ||
974 | ret = GET_TX_DESC_OWN(p_desc); | ||
975 | break; | ||
976 | case HW_DESC_TXBUFF_ADDR: | ||
977 | ret = GET_TX_DESC_TX_BUFFER_ADDRESS(p_desc); | ||
978 | break; | ||
979 | default: | ||
980 | RT_ASSERT(false, ("ERR txdesc :%d " | ||
981 | "not process\n", desc_name)); | ||
982 | break; | ||
983 | } | ||
984 | } else { | ||
985 | struct rx_desc_92c *pdesc = (struct rx_desc_92c *)p_desc; | ||
986 | switch (desc_name) { | ||
987 | case HW_DESC_OWN: | ||
988 | ret = GET_RX_DESC_OWN(pdesc); | ||
989 | break; | ||
990 | case HW_DESC_RXPKT_LEN: | ||
991 | ret = GET_RX_DESC_PKT_LEN(pdesc); | ||
992 | break; | ||
993 | default: | ||
994 | RT_ASSERT(false, ("ERR rxdesc :%d " | ||
995 | "not process\n", desc_name)); | ||
996 | break; | ||
997 | } | ||
998 | } | ||
999 | return ret; | ||
1000 | } | ||
1001 | |||
1002 | void rtl92ce_tx_polling(struct ieee80211_hw *hw, u8 hw_queue) | ||
1003 | { | ||
1004 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1005 | if (hw_queue == BEACON_QUEUE) { | ||
1006 | rtl_write_word(rtlpriv, REG_PCIE_CTRL_REG, BIT(4)); | ||
1007 | } else { | ||
1008 | rtl_write_word(rtlpriv, REG_PCIE_CTRL_REG, | ||
1009 | BIT(0) << (hw_queue)); | ||
1010 | } | ||
1011 | } | ||
1012 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192ce/trx.h b/drivers/net/wireless/rtlwifi/rtl8192ce/trx.h new file mode 100644 index 000000000000..0f1177137501 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192ce/trx.h | |||
@@ -0,0 +1,739 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CE_TRX_H__ | ||
31 | #define __RTL92CE_TRX_H__ | ||
32 | |||
33 | #define TX_DESC_SIZE 64 | ||
34 | #define TX_DESC_AGGR_SUBFRAME_SIZE 32 | ||
35 | |||
36 | #define RX_DESC_SIZE 32 | ||
37 | #define RX_DRV_INFO_SIZE_UNIT 8 | ||
38 | |||
39 | #define TX_DESC_NEXT_DESC_OFFSET 40 | ||
40 | #define USB_HWDESC_HEADER_LEN 32 | ||
41 | #define CRCLENGTH 4 | ||
42 | |||
43 | /* Define a macro that takes a le32 word, converts it to host ordering, | ||
44 | * right shifts by a specified count, creates a mask of the specified | ||
45 | * bit count, and extracts that number of bits. | ||
46 | */ | ||
47 | |||
48 | #define SHIFT_AND_MASK_LE(__pdesc, __shift, __mask) \ | ||
49 | ((le32_to_cpu(*(((__le32 *)(__pdesc)))) >> (__shift)) & \ | ||
50 | BIT_LEN_MASK_32(__mask)) | ||
51 | |||
52 | /* Define a macro that clears a bit field in an le32 word and | ||
53 | * sets the specified value into that bit field. The resulting | ||
54 | * value remains in le32 ordering; however, it is properly converted | ||
55 | * to host ordering for the clear and set operations before conversion | ||
56 | * back to le32. | ||
57 | */ | ||
58 | |||
59 | #define SET_BITS_OFFSET_LE(__pdesc, __shift, __len, __val) \ | ||
60 | (*(__le32 *)(__pdesc) = \ | ||
61 | (cpu_to_le32((le32_to_cpu(*((__le32 *)(__pdesc))) & \ | ||
62 | (~(BIT_OFFSET_LEN_MASK_32((__shift), __len)))) | \ | ||
63 | (((u32)(__val) & BIT_LEN_MASK_32(__len)) << (__shift))))); | ||
64 | |||
65 | /* macros to read/write various fields in RX or TX descriptors */ | ||
66 | |||
67 | #define SET_TX_DESC_PKT_SIZE(__pdesc, __val) \ | ||
68 | SET_BITS_OFFSET_LE(__pdesc, 0, 16, __val) | ||
69 | #define SET_TX_DESC_OFFSET(__pdesc, __val) \ | ||
70 | SET_BITS_OFFSET_LE(__pdesc, 16, 8, __val) | ||
71 | #define SET_TX_DESC_BMC(__pdesc, __val) \ | ||
72 | SET_BITS_OFFSET_LE(__pdesc, 24, 1, __val) | ||
73 | #define SET_TX_DESC_HTC(__pdesc, __val) \ | ||
74 | SET_BITS_OFFSET_LE(__pdesc, 25, 1, __val) | ||
75 | #define SET_TX_DESC_LAST_SEG(__pdesc, __val) \ | ||
76 | SET_BITS_OFFSET_LE(__pdesc, 26, 1, __val) | ||
77 | #define SET_TX_DESC_FIRST_SEG(__pdesc, __val) \ | ||
78 | SET_BITS_OFFSET_LE(__pdesc, 27, 1, __val) | ||
79 | #define SET_TX_DESC_LINIP(__pdesc, __val) \ | ||
80 | SET_BITS_OFFSET_LE(__pdesc, 28, 1, __val) | ||
81 | #define SET_TX_DESC_NO_ACM(__pdesc, __val) \ | ||
82 | SET_BITS_OFFSET_LE(__pdesc, 29, 1, __val) | ||
83 | #define SET_TX_DESC_GF(__pdesc, __val) \ | ||
84 | SET_BITS_OFFSET_LE(__pdesc, 30, 1, __val) | ||
85 | #define SET_TX_DESC_OWN(__pdesc, __val) \ | ||
86 | SET_BITS_OFFSET_LE(__pdesc, 31, 1, __val) | ||
87 | |||
88 | #define GET_TX_DESC_PKT_SIZE(__pdesc) \ | ||
89 | SHIFT_AND_MASK_LE(__pdesc, 0, 16) | ||
90 | #define GET_TX_DESC_OFFSET(__pdesc) \ | ||
91 | SHIFT_AND_MASK_LE(__pdesc, 16, 8) | ||
92 | #define GET_TX_DESC_BMC(__pdesc) \ | ||
93 | SHIFT_AND_MASK_LE(__pdesc, 24, 1) | ||
94 | #define GET_TX_DESC_HTC(__pdesc) \ | ||
95 | SHIFT_AND_MASK_LE(__pdesc, 25, 1) | ||
96 | #define GET_TX_DESC_LAST_SEG(__pdesc) \ | ||
97 | SHIFT_AND_MASK_LE(__pdesc, 26, 1) | ||
98 | #define GET_TX_DESC_FIRST_SEG(__pdesc) \ | ||
99 | SHIFT_AND_MASK_LE(__pdesc, 27, 1) | ||
100 | #define GET_TX_DESC_LINIP(__pdesc) \ | ||
101 | SHIFT_AND_MASK_LE(__pdesc, 28, 1) | ||
102 | #define GET_TX_DESC_NO_ACM(__pdesc) \ | ||
103 | SHIFT_AND_MASK_LE(__pdesc, 29, 1) | ||
104 | #define GET_TX_DESC_GF(__pdesc) \ | ||
105 | SHIFT_AND_MASK_LE(__pdesc, 30, 1) | ||
106 | #define GET_TX_DESC_OWN(__pdesc) \ | ||
107 | SHIFT_AND_MASK_LE(__pdesc, 31, 1) | ||
108 | |||
109 | #define SET_TX_DESC_MACID(__pdesc, __val) \ | ||
110 | SET_BITS_OFFSET_LE(__pdesc+4, 0, 5, __val) | ||
111 | #define SET_TX_DESC_AGG_BREAK(__pdesc, __val) \ | ||
112 | SET_BITS_OFFSET_LE(__pdesc+4, 5, 1, __val) | ||
113 | #define SET_TX_DESC_BK(__pdesc, __val) \ | ||
114 | SET_BITS_OFFSET_LE(__pdesc+4, 6, 1, __val) | ||
115 | #define SET_TX_DESC_RDG_ENABLE(__pdesc, __val) \ | ||
116 | SET_BITS_OFFSET_LE(__pdesc+4, 7, 1, __val) | ||
117 | #define SET_TX_DESC_QUEUE_SEL(__pdesc, __val) \ | ||
118 | SET_BITS_OFFSET_LE(__pdesc+4, 8, 5, __val) | ||
119 | #define SET_TX_DESC_RDG_NAV_EXT(__pdesc, __val) \ | ||
120 | SET_BITS_OFFSET_LE(__pdesc+4, 13, 1, __val) | ||
121 | #define SET_TX_DESC_LSIG_TXOP_EN(__pdesc, __val) \ | ||
122 | SET_BITS_OFFSET_LE(__pdesc+4, 14, 1, __val) | ||
123 | #define SET_TX_DESC_PIFS(__pdesc, __val) \ | ||
124 | SET_BITS_OFFSET_LE(__pdesc+4, 15, 1, __val) | ||
125 | #define SET_TX_DESC_RATE_ID(__pdesc, __val) \ | ||
126 | SET_BITS_OFFSET_LE(__pdesc+4, 16, 4, __val) | ||
127 | #define SET_TX_DESC_NAV_USE_HDR(__pdesc, __val) \ | ||
128 | SET_BITS_OFFSET_LE(__pdesc+4, 20, 1, __val) | ||
129 | #define SET_TX_DESC_EN_DESC_ID(__pdesc, __val) \ | ||
130 | SET_BITS_OFFSET_LE(__pdesc+4, 21, 1, __val) | ||
131 | #define SET_TX_DESC_SEC_TYPE(__pdesc, __val) \ | ||
132 | SET_BITS_OFFSET_LE(__pdesc+4, 22, 2, __val) | ||
133 | #define SET_TX_DESC_PKT_OFFSET(__pdesc, __val) \ | ||
134 | SET_BITS_OFFSET_LE(__pdesc+4, 24, 8, __val) | ||
135 | |||
136 | #define GET_TX_DESC_MACID(__pdesc) \ | ||
137 | SHIFT_AND_MASK_LE(__pdesc+4, 0, 5) | ||
138 | #define GET_TX_DESC_AGG_ENABLE(__pdesc) \ | ||
139 | SHIFT_AND_MASK_LE(__pdesc+4, 5, 1) | ||
140 | #define GET_TX_DESC_AGG_BREAK(__pdesc) \ | ||
141 | SHIFT_AND_MASK_LE(__pdesc+4, 6, 1) | ||
142 | #define GET_TX_DESC_RDG_ENABLE(__pdesc) \ | ||
143 | SHIFT_AND_MASK_LE(__pdesc+4, 7, 1) | ||
144 | #define GET_TX_DESC_QUEUE_SEL(__pdesc) \ | ||
145 | SHIFT_AND_MASK_LE(__pdesc+4, 8, 5) | ||
146 | #define GET_TX_DESC_RDG_NAV_EXT(__pdesc) \ | ||
147 | SHIFT_AND_MASK_LE(__pdesc+4, 13, 1) | ||
148 | #define GET_TX_DESC_LSIG_TXOP_EN(__pdesc) \ | ||
149 | SHIFT_AND_MASK_LE(__pdesc+4, 14, 1) | ||
150 | #define GET_TX_DESC_PIFS(__pdesc) \ | ||
151 | SHIFT_AND_MASK_LE(__pdesc+4, 15, 1) | ||
152 | #define GET_TX_DESC_RATE_ID(__pdesc) \ | ||
153 | SHIFT_AND_MASK_LE(__pdesc+4, 16, 4) | ||
154 | #define GET_TX_DESC_NAV_USE_HDR(__pdesc) \ | ||
155 | SHIFT_AND_MASK_LE(__pdesc+4, 20, 1) | ||
156 | #define GET_TX_DESC_EN_DESC_ID(__pdesc) \ | ||
157 | SHIFT_AND_MASK_LE(__pdesc+4, 21, 1) | ||
158 | #define GET_TX_DESC_SEC_TYPE(__pdesc) \ | ||
159 | SHIFT_AND_MASK_LE(__pdesc+4, 22, 2) | ||
160 | #define GET_TX_DESC_PKT_OFFSET(__pdesc) \ | ||
161 | SHIFT_AND_MASK_LE(__pdesc+4, 24, 8) | ||
162 | |||
163 | #define SET_TX_DESC_RTS_RC(__pdesc, __val) \ | ||
164 | SET_BITS_OFFSET_LE(__pdesc+8, 0, 6, __val) | ||
165 | #define SET_TX_DESC_DATA_RC(__pdesc, __val) \ | ||
166 | SET_BITS_OFFSET_LE(__pdesc+8, 6, 6, __val) | ||
167 | #define SET_TX_DESC_BAR_RTY_TH(__pdesc, __val) \ | ||
168 | SET_BITS_OFFSET_LE(__pdesc+8, 14, 2, __val) | ||
169 | #define SET_TX_DESC_MORE_FRAG(__pdesc, __val) \ | ||
170 | SET_BITS_OFFSET_LE(__pdesc+8, 17, 1, __val) | ||
171 | #define SET_TX_DESC_RAW(__pdesc, __val) \ | ||
172 | SET_BITS_OFFSET_LE(__pdesc+8, 18, 1, __val) | ||
173 | #define SET_TX_DESC_CCX(__pdesc, __val) \ | ||
174 | SET_BITS_OFFSET_LE(__pdesc+8, 19, 1, __val) | ||
175 | #define SET_TX_DESC_AMPDU_DENSITY(__pdesc, __val) \ | ||
176 | SET_BITS_OFFSET_LE(__pdesc+8, 20, 3, __val) | ||
177 | #define SET_TX_DESC_ANTSEL_A(__pdesc, __val) \ | ||
178 | SET_BITS_OFFSET_LE(__pdesc+8, 24, 1, __val) | ||
179 | #define SET_TX_DESC_ANTSEL_B(__pdesc, __val) \ | ||
180 | SET_BITS_OFFSET_LE(__pdesc+8, 25, 1, __val) | ||
181 | #define SET_TX_DESC_TX_ANT_CCK(__pdesc, __val) \ | ||
182 | SET_BITS_OFFSET_LE(__pdesc+8, 26, 2, __val) | ||
183 | #define SET_TX_DESC_TX_ANTL(__pdesc, __val) \ | ||
184 | SET_BITS_OFFSET_LE(__pdesc+8, 28, 2, __val) | ||
185 | #define SET_TX_DESC_TX_ANT_HT(__pdesc, __val) \ | ||
186 | SET_BITS_OFFSET_LE(__pdesc+8, 30, 2, __val) | ||
187 | |||
188 | #define GET_TX_DESC_RTS_RC(__pdesc) \ | ||
189 | SHIFT_AND_MASK_LE(__pdesc+8, 0, 6) | ||
190 | #define GET_TX_DESC_DATA_RC(__pdesc) \ | ||
191 | SHIFT_AND_MASK_LE(__pdesc+8, 6, 6) | ||
192 | #define GET_TX_DESC_BAR_RTY_TH(__pdesc) \ | ||
193 | SHIFT_AND_MASK_LE(__pdesc+8, 14, 2) | ||
194 | #define GET_TX_DESC_MORE_FRAG(__pdesc) \ | ||
195 | SHIFT_AND_MASK_LE(__pdesc+8, 17, 1) | ||
196 | #define GET_TX_DESC_RAW(__pdesc) \ | ||
197 | SHIFT_AND_MASK_LE(__pdesc+8, 18, 1) | ||
198 | #define GET_TX_DESC_CCX(__pdesc) \ | ||
199 | SHIFT_AND_MASK_LE(__pdesc+8, 19, 1) | ||
200 | #define GET_TX_DESC_AMPDU_DENSITY(__pdesc) \ | ||
201 | SHIFT_AND_MASK_LE(__pdesc+8, 20, 3) | ||
202 | #define GET_TX_DESC_ANTSEL_A(__pdesc) \ | ||
203 | SHIFT_AND_MASK_LE(__pdesc+8, 24, 1) | ||
204 | #define GET_TX_DESC_ANTSEL_B(__pdesc) \ | ||
205 | SHIFT_AND_MASK_LE(__pdesc+8, 25, 1) | ||
206 | #define GET_TX_DESC_TX_ANT_CCK(__pdesc) \ | ||
207 | SHIFT_AND_MASK_LE(__pdesc+8, 26, 2) | ||
208 | #define GET_TX_DESC_TX_ANTL(__pdesc) \ | ||
209 | SHIFT_AND_MASK_LE(__pdesc+8, 28, 2) | ||
210 | #define GET_TX_DESC_TX_ANT_HT(__pdesc) \ | ||
211 | SHIFT_AND_MASK_LE(__pdesc+8, 30, 2) | ||
212 | |||
213 | #define SET_TX_DESC_NEXT_HEAP_PAGE(__pdesc, __val) \ | ||
214 | SET_BITS_OFFSET_LE(__pdesc+12, 0, 8, __val) | ||
215 | #define SET_TX_DESC_TAIL_PAGE(__pdesc, __val) \ | ||
216 | SET_BITS_OFFSET_LE(__pdesc+12, 8, 8, __val) | ||
217 | #define SET_TX_DESC_SEQ(__pdesc, __val) \ | ||
218 | SET_BITS_OFFSET_LE(__pdesc+12, 16, 12, __val) | ||
219 | #define SET_TX_DESC_PKT_ID(__pdesc, __val) \ | ||
220 | SET_BITS_OFFSET_LE(__pdesc+12, 28, 4, __val) | ||
221 | |||
222 | #define GET_TX_DESC_NEXT_HEAP_PAGE(__pdesc) \ | ||
223 | SHIFT_AND_MASK_LE(__pdesc+12, 0, 8) | ||
224 | #define GET_TX_DESC_TAIL_PAGE(__pdesc) \ | ||
225 | SHIFT_AND_MASK_LE(__pdesc+12, 8, 8) | ||
226 | #define GET_TX_DESC_SEQ(__pdesc) \ | ||
227 | SHIFT_AND_MASK_LE(__pdesc+12, 16, 12) | ||
228 | #define GET_TX_DESC_PKT_ID(__pdesc) \ | ||
229 | SHIFT_AND_MASK_LE(__pdesc+12, 28, 4) | ||
230 | |||
231 | #define SET_TX_DESC_RTS_RATE(__pdesc, __val) \ | ||
232 | SET_BITS_OFFSET_LE(__pdesc+16, 0, 5, __val) | ||
233 | #define SET_TX_DESC_AP_DCFE(__pdesc, __val) \ | ||
234 | SET_BITS_OFFSET_LE(__pdesc+16, 5, 1, __val) | ||
235 | #define SET_TX_DESC_QOS(__pdesc, __val) \ | ||
236 | SET_BITS_OFFSET_LE(__pdesc+16, 6, 1, __val) | ||
237 | #define SET_TX_DESC_HWSEQ_EN(__pdesc, __val) \ | ||
238 | SET_BITS_OFFSET_LE(__pdesc+16, 7, 1, __val) | ||
239 | #define SET_TX_DESC_USE_RATE(__pdesc, __val) \ | ||
240 | SET_BITS_OFFSET_LE(__pdesc+16, 8, 1, __val) | ||
241 | #define SET_TX_DESC_DISABLE_RTS_FB(__pdesc, __val) \ | ||
242 | SET_BITS_OFFSET_LE(__pdesc+16, 9, 1, __val) | ||
243 | #define SET_TX_DESC_DISABLE_FB(__pdesc, __val) \ | ||
244 | SET_BITS_OFFSET_LE(__pdesc+16, 10, 1, __val) | ||
245 | #define SET_TX_DESC_CTS2SELF(__pdesc, __val) \ | ||
246 | SET_BITS_OFFSET_LE(__pdesc+16, 11, 1, __val) | ||
247 | #define SET_TX_DESC_RTS_ENABLE(__pdesc, __val) \ | ||
248 | SET_BITS_OFFSET_LE(__pdesc+16, 12, 1, __val) | ||
249 | #define SET_TX_DESC_HW_RTS_ENABLE(__pdesc, __val) \ | ||
250 | SET_BITS_OFFSET_LE(__pdesc+16, 13, 1, __val) | ||
251 | #define SET_TX_DESC_PORT_ID(__pdesc, __val) \ | ||
252 | SET_BITS_OFFSET_LE(__pdesc+16, 14, 1, __val) | ||
253 | #define SET_TX_DESC_WAIT_DCTS(__pdesc, __val) \ | ||
254 | SET_BITS_OFFSET_LE(__pdesc+16, 18, 1, __val) | ||
255 | #define SET_TX_DESC_CTS2AP_EN(__pdesc, __val) \ | ||
256 | SET_BITS_OFFSET_LE(__pdesc+16, 19, 1, __val) | ||
257 | #define SET_TX_DESC_TX_SUB_CARRIER(__pdesc, __val) \ | ||
258 | SET_BITS_OFFSET_LE(__pdesc+16, 20, 2, __val) | ||
259 | #define SET_TX_DESC_TX_STBC(__pdesc, __val) \ | ||
260 | SET_BITS_OFFSET_LE(__pdesc+16, 22, 2, __val) | ||
261 | #define SET_TX_DESC_DATA_SHORT(__pdesc, __val) \ | ||
262 | SET_BITS_OFFSET_LE(__pdesc+16, 24, 1, __val) | ||
263 | #define SET_TX_DESC_DATA_BW(__pdesc, __val) \ | ||
264 | SET_BITS_OFFSET_LE(__pdesc+16, 25, 1, __val) | ||
265 | #define SET_TX_DESC_RTS_SHORT(__pdesc, __val) \ | ||
266 | SET_BITS_OFFSET_LE(__pdesc+16, 26, 1, __val) | ||
267 | #define SET_TX_DESC_RTS_BW(__pdesc, __val) \ | ||
268 | SET_BITS_OFFSET_LE(__pdesc+16, 27, 1, __val) | ||
269 | #define SET_TX_DESC_RTS_SC(__pdesc, __val) \ | ||
270 | SET_BITS_OFFSET_LE(__pdesc+16, 28, 2, __val) | ||
271 | #define SET_TX_DESC_RTS_STBC(__pdesc, __val) \ | ||
272 | SET_BITS_OFFSET_LE(__pdesc+16, 30, 2, __val) | ||
273 | |||
274 | #define GET_TX_DESC_RTS_RATE(__pdesc) \ | ||
275 | SHIFT_AND_MASK_LE(__pdesc+16, 0, 5) | ||
276 | #define GET_TX_DESC_AP_DCFE(__pdesc) \ | ||
277 | SHIFT_AND_MASK_LE(__pdesc+16, 5, 1) | ||
278 | #define GET_TX_DESC_QOS(__pdesc) \ | ||
279 | SHIFT_AND_MASK_LE(__pdesc+16, 6, 1) | ||
280 | #define GET_TX_DESC_HWSEQ_EN(__pdesc) \ | ||
281 | SHIFT_AND_MASK_LE(__pdesc+16, 7, 1) | ||
282 | #define GET_TX_DESC_USE_RATE(__pdesc) \ | ||
283 | SHIFT_AND_MASK_LE(__pdesc+16, 8, 1) | ||
284 | #define GET_TX_DESC_DISABLE_RTS_FB(__pdesc) \ | ||
285 | SHIFT_AND_MASK_LE(__pdesc+16, 9, 1) | ||
286 | #define GET_TX_DESC_DISABLE_FB(__pdesc) \ | ||
287 | SHIFT_AND_MASK_LE(__pdesc+16, 10, 1) | ||
288 | #define GET_TX_DESC_CTS2SELF(__pdesc) \ | ||
289 | SHIFT_AND_MASK_LE(__pdesc+16, 11, 1) | ||
290 | #define GET_TX_DESC_RTS_ENABLE(__pdesc) \ | ||
291 | SHIFT_AND_MASK_LE(__pdesc+16, 12, 1) | ||
292 | #define GET_TX_DESC_HW_RTS_ENABLE(__pdesc) \ | ||
293 | SHIFT_AND_MASK_LE(__pdesc+16, 13, 1) | ||
294 | #define GET_TX_DESC_PORT_ID(__pdesc) \ | ||
295 | SHIFT_AND_MASK_LE(__pdesc+16, 14, 1) | ||
296 | #define GET_TX_DESC_WAIT_DCTS(__pdesc) \ | ||
297 | SHIFT_AND_MASK_LE(__pdesc+16, 18, 1) | ||
298 | #define GET_TX_DESC_CTS2AP_EN(__pdesc) \ | ||
299 | SHIFT_AND_MASK_LE(__pdesc+16, 19, 1) | ||
300 | #define GET_TX_DESC_TX_SUB_CARRIER(__pdesc) \ | ||
301 | SHIFT_AND_MASK_LE(__pdesc+16, 20, 2) | ||
302 | #define GET_TX_DESC_TX_STBC(__pdesc) \ | ||
303 | SHIFT_AND_MASK_LE(__pdesc+16, 22, 2) | ||
304 | #define GET_TX_DESC_DATA_SHORT(__pdesc) \ | ||
305 | SHIFT_AND_MASK_LE(__pdesc+16, 24, 1) | ||
306 | #define GET_TX_DESC_DATA_BW(__pdesc) \ | ||
307 | SHIFT_AND_MASK_LE(__pdesc+16, 25, 1) | ||
308 | #define GET_TX_DESC_RTS_SHORT(__pdesc) \ | ||
309 | SHIFT_AND_MASK_LE(__pdesc+16, 26, 1) | ||
310 | #define GET_TX_DESC_RTS_BW(__pdesc) \ | ||
311 | SHIFT_AND_MASK_LE(__pdesc+16, 27, 1) | ||
312 | #define GET_TX_DESC_RTS_SC(__pdesc) \ | ||
313 | SHIFT_AND_MASK_LE(__pdesc+16, 28, 2) | ||
314 | #define GET_TX_DESC_RTS_STBC(__pdesc) \ | ||
315 | SHIFT_AND_MASK_LE(__pdesc+16, 30, 2) | ||
316 | |||
317 | #define SET_TX_DESC_TX_RATE(__pdesc, __val) \ | ||
318 | SET_BITS_OFFSET_LE(__pdesc+20, 0, 6, __val) | ||
319 | #define SET_TX_DESC_DATA_SHORTGI(__pdesc, __val) \ | ||
320 | SET_BITS_OFFSET_LE(__pdesc+20, 6, 1, __val) | ||
321 | #define SET_TX_DESC_CCX_TAG(__pdesc, __val) \ | ||
322 | SET_BITS_OFFSET_LE(__pdesc+20, 7, 1, __val) | ||
323 | #define SET_TX_DESC_DATA_RATE_FB_LIMIT(__pdesc, __val) \ | ||
324 | SET_BITS_OFFSET_LE(__pdesc+20, 8, 5, __val) | ||
325 | #define SET_TX_DESC_RTS_RATE_FB_LIMIT(__pdesc, __val) \ | ||
326 | SET_BITS_OFFSET_LE(__pdesc+20, 13, 4, __val) | ||
327 | #define SET_TX_DESC_RETRY_LIMIT_ENABLE(__pdesc, __val) \ | ||
328 | SET_BITS_OFFSET_LE(__pdesc+20, 17, 1, __val) | ||
329 | #define SET_TX_DESC_DATA_RETRY_LIMIT(__pdesc, __val) \ | ||
330 | SET_BITS_OFFSET_LE(__pdesc+20, 18, 6, __val) | ||
331 | #define SET_TX_DESC_USB_TXAGG_NUM(__pdesc, __val) \ | ||
332 | SET_BITS_OFFSET_LE(__pdesc+20, 24, 8, __val) | ||
333 | |||
334 | #define GET_TX_DESC_TX_RATE(__pdesc) \ | ||
335 | SHIFT_AND_MASK_LE(__pdesc+20, 0, 6) | ||
336 | #define GET_TX_DESC_DATA_SHORTGI(__pdesc) \ | ||
337 | SHIFT_AND_MASK_LE(__pdesc+20, 6, 1) | ||
338 | #define GET_TX_DESC_CCX_TAG(__pdesc) \ | ||
339 | SHIFT_AND_MASK_LE(__pdesc+20, 7, 1) | ||
340 | #define GET_TX_DESC_DATA_RATE_FB_LIMIT(__pdesc) \ | ||
341 | SHIFT_AND_MASK_LE(__pdesc+20, 8, 5) | ||
342 | #define GET_TX_DESC_RTS_RATE_FB_LIMIT(__pdesc) \ | ||
343 | SHIFT_AND_MASK_LE(__pdesc+20, 13, 4) | ||
344 | #define GET_TX_DESC_RETRY_LIMIT_ENABLE(__pdesc) \ | ||
345 | SHIFT_AND_MASK_LE(__pdesc+20, 17, 1) | ||
346 | #define GET_TX_DESC_DATA_RETRY_LIMIT(__pdesc) \ | ||
347 | SHIFT_AND_MASK_LE(__pdesc+20, 18, 6) | ||
348 | #define GET_TX_DESC_USB_TXAGG_NUM(__pdesc) \ | ||
349 | SHIFT_AND_MASK_LE(__pdesc+20, 24, 8) | ||
350 | |||
351 | #define SET_TX_DESC_TXAGC_A(__pdesc, __val) \ | ||
352 | SET_BITS_OFFSET_LE(__pdesc+24, 0, 5, __val) | ||
353 | #define SET_TX_DESC_TXAGC_B(__pdesc, __val) \ | ||
354 | SET_BITS_OFFSET_LE(__pdesc+24, 5, 5, __val) | ||
355 | #define SET_TX_DESC_USE_MAX_LEN(__pdesc, __val) \ | ||
356 | SET_BITS_OFFSET_LE(__pdesc+24, 10, 1, __val) | ||
357 | #define SET_TX_DESC_MAX_AGG_NUM(__pdesc, __val) \ | ||
358 | SET_BITS_OFFSET_LE(__pdesc+24, 11, 5, __val) | ||
359 | #define SET_TX_DESC_MCSG1_MAX_LEN(__pdesc, __val) \ | ||
360 | SET_BITS_OFFSET_LE(__pdesc+24, 16, 4, __val) | ||
361 | #define SET_TX_DESC_MCSG2_MAX_LEN(__pdesc, __val) \ | ||
362 | SET_BITS_OFFSET_LE(__pdesc+24, 20, 4, __val) | ||
363 | #define SET_TX_DESC_MCSG3_MAX_LEN(__pdesc, __val) \ | ||
364 | SET_BITS_OFFSET_LE(__pdesc+24, 24, 4, __val) | ||
365 | #define SET_TX_DESC_MCS7_SGI_MAX_LEN(__pdesc, __val) \ | ||
366 | SET_BITS_OFFSET_LE(__pdesc+24, 28, 4, __val) | ||
367 | |||
368 | #define GET_TX_DESC_TXAGC_A(__pdesc) \ | ||
369 | SHIFT_AND_MASK_LE(__pdesc+24, 0, 5) | ||
370 | #define GET_TX_DESC_TXAGC_B(__pdesc) \ | ||
371 | SHIFT_AND_MASK_LE(__pdesc+24, 5, 5) | ||
372 | #define GET_TX_DESC_USE_MAX_LEN(__pdesc) \ | ||
373 | SHIFT_AND_MASK_LE(__pdesc+24, 10, 1) | ||
374 | #define GET_TX_DESC_MAX_AGG_NUM(__pdesc) \ | ||
375 | SHIFT_AND_MASK_LE(__pdesc+24, 11, 5) | ||
376 | #define GET_TX_DESC_MCSG1_MAX_LEN(__pdesc) \ | ||
377 | SHIFT_AND_MASK_LE(__pdesc+24, 16, 4) | ||
378 | #define GET_TX_DESC_MCSG2_MAX_LEN(__pdesc) \ | ||
379 | SHIFT_AND_MASK_LE(__pdesc+24, 20, 4) | ||
380 | #define GET_TX_DESC_MCSG3_MAX_LEN(__pdesc) \ | ||
381 | SHIFT_AND_MASK_LE(__pdesc+24, 24, 4) | ||
382 | #define GET_TX_DESC_MCS7_SGI_MAX_LEN(__pdesc) \ | ||
383 | SHIFT_AND_MASK_LE(__pdesc+24, 28, 4) | ||
384 | |||
385 | #define SET_TX_DESC_TX_BUFFER_SIZE(__pdesc, __val) \ | ||
386 | SET_BITS_OFFSET_LE(__pdesc+28, 0, 16, __val) | ||
387 | #define SET_TX_DESC_MCSG4_MAX_LEN(__pdesc, __val) \ | ||
388 | SET_BITS_OFFSET_LE(__pdesc+28, 16, 4, __val) | ||
389 | #define SET_TX_DESC_MCSG5_MAX_LEN(__pdesc, __val) \ | ||
390 | SET_BITS_OFFSET_LE(__pdesc+28, 20, 4, __val) | ||
391 | #define SET_TX_DESC_MCSG6_MAX_LEN(__pdesc, __val) \ | ||
392 | SET_BITS_OFFSET_LE(__pdesc+28, 24, 4, __val) | ||
393 | #define SET_TX_DESC_MCS15_SGI_MAX_LEN(__pdesc, __val) \ | ||
394 | SET_BITS_OFFSET_LE(__pdesc+28, 28, 4, __val) | ||
395 | |||
396 | #define GET_TX_DESC_TX_BUFFER_SIZE(__pdesc) \ | ||
397 | SHIFT_AND_MASK_LE(__pdesc+28, 0, 16) | ||
398 | #define GET_TX_DESC_MCSG4_MAX_LEN(__pdesc) \ | ||
399 | SHIFT_AND_MASK_LE(__pdesc+28, 16, 4) | ||
400 | #define GET_TX_DESC_MCSG5_MAX_LEN(__pdesc) \ | ||
401 | SHIFT_AND_MASK_LE(__pdesc+28, 20, 4) | ||
402 | #define GET_TX_DESC_MCSG6_MAX_LEN(__pdesc) \ | ||
403 | SHIFT_AND_MASK_LE(__pdesc+28, 24, 4) | ||
404 | #define GET_TX_DESC_MCS15_SGI_MAX_LEN(__pdesc) \ | ||
405 | SHIFT_AND_MASK_LE(__pdesc+28, 28, 4) | ||
406 | |||
407 | #define SET_TX_DESC_TX_BUFFER_ADDRESS(__pdesc, __val) \ | ||
408 | SET_BITS_OFFSET_LE(__pdesc+32, 0, 32, __val) | ||
409 | #define SET_TX_DESC_TX_BUFFER_ADDRESS64(__pdesc, __val) \ | ||
410 | SET_BITS_OFFSET_LE(__pdesc+36, 0, 32, __val) | ||
411 | |||
412 | #define GET_TX_DESC_TX_BUFFER_ADDRESS(__pdesc) \ | ||
413 | SHIFT_AND_MASK_LE(__pdesc+32, 0, 32) | ||
414 | #define GET_TX_DESC_TX_BUFFER_ADDRESS64(__pdesc) \ | ||
415 | SHIFT_AND_MASK_LE(__pdesc+36, 0, 32) | ||
416 | |||
417 | #define SET_TX_DESC_NEXT_DESC_ADDRESS(__pdesc, __val) \ | ||
418 | SET_BITS_OFFSET_LE(__pdesc+40, 0, 32, __val) | ||
419 | #define SET_TX_DESC_NEXT_DESC_ADDRESS64(__pdesc, __val) \ | ||
420 | SET_BITS_OFFSET_LE(__pdesc+44, 0, 32, __val) | ||
421 | |||
422 | #define GET_TX_DESC_NEXT_DESC_ADDRESS(__pdesc) \ | ||
423 | SHIFT_AND_MASK_LE(__pdesc+40, 0, 32) | ||
424 | #define GET_TX_DESC_NEXT_DESC_ADDRESS64(__pdesc) \ | ||
425 | SHIFT_AND_MASK_LE(__pdesc+44, 0, 32) | ||
426 | |||
427 | #define GET_RX_DESC_PKT_LEN(__pdesc) \ | ||
428 | SHIFT_AND_MASK_LE(__pdesc, 0, 14) | ||
429 | #define GET_RX_DESC_CRC32(__pdesc) \ | ||
430 | SHIFT_AND_MASK_LE(__pdesc, 14, 1) | ||
431 | #define GET_RX_DESC_ICV(__pdesc) \ | ||
432 | SHIFT_AND_MASK_LE(__pdesc, 15, 1) | ||
433 | #define GET_RX_DESC_DRV_INFO_SIZE(__pdesc) \ | ||
434 | SHIFT_AND_MASK_LE(__pdesc, 16, 4) | ||
435 | #define GET_RX_DESC_SECURITY(__pdesc) \ | ||
436 | SHIFT_AND_MASK_LE(__pdesc, 20, 3) | ||
437 | #define GET_RX_DESC_QOS(__pdesc) \ | ||
438 | SHIFT_AND_MASK_LE(__pdesc, 23, 1) | ||
439 | #define GET_RX_DESC_SHIFT(__pdesc) \ | ||
440 | SHIFT_AND_MASK_LE(__pdesc, 24, 2) | ||
441 | #define GET_RX_DESC_PHYST(__pdesc) \ | ||
442 | SHIFT_AND_MASK_LE(__pdesc, 26, 1) | ||
443 | #define GET_RX_DESC_SWDEC(__pdesc) \ | ||
444 | SHIFT_AND_MASK_LE(__pdesc, 27, 1) | ||
445 | #define GET_RX_DESC_LS(__pdesc) \ | ||
446 | SHIFT_AND_MASK_LE(__pdesc, 28, 1) | ||
447 | #define GET_RX_DESC_FS(__pdesc) \ | ||
448 | SHIFT_AND_MASK_LE(__pdesc, 29, 1) | ||
449 | #define GET_RX_DESC_EOR(__pdesc) \ | ||
450 | SHIFT_AND_MASK_LE(__pdesc, 30, 1) | ||
451 | #define GET_RX_DESC_OWN(__pdesc) \ | ||
452 | SHIFT_AND_MASK_LE(__pdesc, 31, 1) | ||
453 | |||
454 | #define SET_RX_DESC_PKT_LEN(__pdesc, __val) \ | ||
455 | SET_BITS_OFFSET_LE(__pdesc, 0, 14, __val) | ||
456 | #define SET_RX_DESC_EOR(__pdesc, __val) \ | ||
457 | SET_BITS_OFFSET_LE(__pdesc, 30, 1, __val) | ||
458 | #define SET_RX_DESC_OWN(__pdesc, __val) \ | ||
459 | SET_BITS_OFFSET_LE(__pdesc, 31, 1, __val) | ||
460 | |||
461 | #define GET_RX_DESC_MACID(__pdesc) \ | ||
462 | SHIFT_AND_MASK_LE(__pdesc+4, 0, 5) | ||
463 | #define GET_RX_DESC_TID(__pdesc) \ | ||
464 | SHIFT_AND_MASK_LE(__pdesc+4, 5, 4) | ||
465 | #define GET_RX_DESC_HWRSVD(__pdesc) \ | ||
466 | SHIFT_AND_MASK_LE(__pdesc+4, 9, 5) | ||
467 | #define GET_RX_DESC_PAGGR(__pdesc) \ | ||
468 | SHIFT_AND_MASK_LE(__pdesc+4, 14, 1) | ||
469 | #define GET_RX_DESC_FAGGR(__pdesc) \ | ||
470 | SHIFT_AND_MASK_LE(__pdesc+4, 15, 1) | ||
471 | #define GET_RX_DESC_A1_FIT(__pdesc) \ | ||
472 | SHIFT_AND_MASK_LE(__pdesc+4, 16, 4) | ||
473 | #define GET_RX_DESC_A2_FIT(__pdesc) \ | ||
474 | SHIFT_AND_MASK_LE(__pdesc+4, 20, 4) | ||
475 | #define GET_RX_DESC_PAM(__pdesc) \ | ||
476 | SHIFT_AND_MASK_LE(__pdesc+4, 24, 1) | ||
477 | #define GET_RX_DESC_PWR(__pdesc) \ | ||
478 | SHIFT_AND_MASK_LE(__pdesc+4, 25, 1) | ||
479 | #define GET_RX_DESC_MD(__pdesc) \ | ||
480 | SHIFT_AND_MASK_LE(__pdesc+4, 26, 1) | ||
481 | #define GET_RX_DESC_MF(__pdesc) \ | ||
482 | SHIFT_AND_MASK_LE(__pdesc+4, 27, 1) | ||
483 | #define GET_RX_DESC_TYPE(__pdesc) \ | ||
484 | SHIFT_AND_MASK_LE(__pdesc+4, 28, 2) | ||
485 | #define GET_RX_DESC_MC(__pdesc) \ | ||
486 | SHIFT_AND_MASK_LE(__pdesc+4, 30, 1) | ||
487 | #define GET_RX_DESC_BC(__pdesc) \ | ||
488 | SHIFT_AND_MASK_LE(__pdesc+4, 31, 1) | ||
489 | #define GET_RX_DESC_SEQ(__pdesc) \ | ||
490 | SHIFT_AND_MASK_LE(__pdesc+8, 0, 12) | ||
491 | #define GET_RX_DESC_FRAG(__pdesc) \ | ||
492 | SHIFT_AND_MASK_LE(__pdesc+8, 12, 4) | ||
493 | #define GET_RX_DESC_NEXT_PKT_LEN(__pdesc) \ | ||
494 | SHIFT_AND_MASK_LE(__pdesc+8, 16, 14) | ||
495 | #define GET_RX_DESC_NEXT_IND(__pdesc) \ | ||
496 | SHIFT_AND_MASK_LE(__pdesc+8, 30, 1) | ||
497 | #define GET_RX_DESC_RSVD(__pdesc) \ | ||
498 | SHIFT_AND_MASK_LE(__pdesc+8, 31, 1) | ||
499 | |||
500 | #define GET_RX_DESC_RXMCS(__pdesc) \ | ||
501 | SHIFT_AND_MASK_LE(__pdesc+12, 0, 6) | ||
502 | #define GET_RX_DESC_RXHT(__pdesc) \ | ||
503 | SHIFT_AND_MASK_LE(__pdesc+12, 6, 1) | ||
504 | #define GET_RX_DESC_SPLCP(__pdesc) \ | ||
505 | SHIFT_AND_MASK_LE(__pdesc+12, 8, 1) | ||
506 | #define GET_RX_DESC_BW(__pdesc) \ | ||
507 | SHIFT_AND_MASK_LE(__pdesc+12, 9, 1) | ||
508 | #define GET_RX_DESC_HTC(__pdesc) \ | ||
509 | SHIFT_AND_MASK_LE(__pdesc+12, 10, 1) | ||
510 | #define GET_RX_DESC_HWPC_ERR(__pdesc) \ | ||
511 | SHIFT_AND_MASK_LE(__pdesc+12, 14, 1) | ||
512 | #define GET_RX_DESC_HWPC_IND(__pdesc) \ | ||
513 | SHIFT_AND_MASK_LE(__pdesc+12, 15, 1) | ||
514 | #define GET_RX_DESC_IV0(__pdesc) \ | ||
515 | SHIFT_AND_MASK_LE(__pdesc+12, 16, 16) | ||
516 | |||
517 | #define GET_RX_DESC_IV1(__pdesc) \ | ||
518 | SHIFT_AND_MASK_LE(__pdesc+16, 0, 32) | ||
519 | #define GET_RX_DESC_TSFL(__pdesc) \ | ||
520 | SHIFT_AND_MASK_LE(__pdesc+20, 0, 32) | ||
521 | |||
522 | #define GET_RX_DESC_BUFF_ADDR(__pdesc) \ | ||
523 | SHIFT_AND_MASK_LE(__pdesc+24, 0, 32) | ||
524 | #define GET_RX_DESC_BUFF_ADDR64(__pdesc) \ | ||
525 | SHIFT_AND_MASK_LE(__pdesc+28, 0, 32) | ||
526 | |||
527 | #define SET_RX_DESC_BUFF_ADDR(__pdesc, __val) \ | ||
528 | SET_BITS_OFFSET_LE(__pdesc+24, 0, 32, __val) | ||
529 | #define SET_RX_DESC_BUFF_ADDR64(__pdesc, __val) \ | ||
530 | SET_BITS_OFFSET_LE(__pdesc+28, 0, 32, __val) | ||
531 | |||
532 | #define CLEAR_PCI_TX_DESC_CONTENT(__pdesc, _size) \ | ||
533 | do { \ | ||
534 | if (_size > TX_DESC_NEXT_DESC_OFFSET) \ | ||
535 | memset(__pdesc, 0, TX_DESC_NEXT_DESC_OFFSET); \ | ||
536 | else \ | ||
537 | memset(__pdesc, 0, _size); \ | ||
538 | } while (0); | ||
539 | |||
540 | #define RX_HAL_IS_CCK_RATE(_pdesc)\ | ||
541 | (_pdesc->rxmcs == DESC92C_RATE1M || \ | ||
542 | _pdesc->rxmcs == DESC92C_RATE2M || \ | ||
543 | _pdesc->rxmcs == DESC92C_RATE5_5M || \ | ||
544 | _pdesc->rxmcs == DESC92C_RATE11M) | ||
545 | |||
546 | struct rx_fwinfo_92c { | ||
547 | u8 gain_trsw[4]; | ||
548 | u8 pwdb_all; | ||
549 | u8 cfosho[4]; | ||
550 | u8 cfotail[4]; | ||
551 | char rxevm[2]; | ||
552 | char rxsnr[4]; | ||
553 | u8 pdsnr[2]; | ||
554 | u8 csi_current[2]; | ||
555 | u8 csi_target[2]; | ||
556 | u8 sigevm; | ||
557 | u8 max_ex_pwr; | ||
558 | u8 ex_intf_flag:1; | ||
559 | u8 sgi_en:1; | ||
560 | u8 rxsc:2; | ||
561 | u8 reserve:4; | ||
562 | } __packed; | ||
563 | |||
564 | struct tx_desc_92c { | ||
565 | u32 pktsize:16; | ||
566 | u32 offset:8; | ||
567 | u32 bmc:1; | ||
568 | u32 htc:1; | ||
569 | u32 lastseg:1; | ||
570 | u32 firstseg:1; | ||
571 | u32 linip:1; | ||
572 | u32 noacm:1; | ||
573 | u32 gf:1; | ||
574 | u32 own:1; | ||
575 | |||
576 | u32 macid:5; | ||
577 | u32 agg_en:1; | ||
578 | u32 bk:1; | ||
579 | u32 rdg_en:1; | ||
580 | u32 queuesel:5; | ||
581 | u32 rd_nav_ext:1; | ||
582 | u32 lsig_txop_en:1; | ||
583 | u32 pifs:1; | ||
584 | u32 rateid:4; | ||
585 | u32 nav_usehdr:1; | ||
586 | u32 en_descid:1; | ||
587 | u32 sectype:2; | ||
588 | u32 pktoffset:8; | ||
589 | |||
590 | u32 rts_rc:6; | ||
591 | u32 data_rc:6; | ||
592 | u32 rsvd0:2; | ||
593 | u32 bar_retryht:2; | ||
594 | u32 rsvd1:1; | ||
595 | u32 morefrag:1; | ||
596 | u32 raw:1; | ||
597 | u32 ccx:1; | ||
598 | u32 ampdudensity:3; | ||
599 | u32 rsvd2:1; | ||
600 | u32 ant_sela:1; | ||
601 | u32 ant_selb:1; | ||
602 | u32 txant_cck:2; | ||
603 | u32 txant_l:2; | ||
604 | u32 txant_ht:2; | ||
605 | |||
606 | u32 nextheadpage:8; | ||
607 | u32 tailpage:8; | ||
608 | u32 seq:12; | ||
609 | u32 pktid:4; | ||
610 | |||
611 | u32 rtsrate:5; | ||
612 | u32 apdcfe:1; | ||
613 | u32 qos:1; | ||
614 | u32 hwseq_enable:1; | ||
615 | u32 userrate:1; | ||
616 | u32 dis_rtsfb:1; | ||
617 | u32 dis_datafb:1; | ||
618 | u32 cts2self:1; | ||
619 | u32 rts_en:1; | ||
620 | u32 hwrts_en:1; | ||
621 | u32 portid:1; | ||
622 | u32 rsvd3:3; | ||
623 | u32 waitdcts:1; | ||
624 | u32 cts2ap_en:1; | ||
625 | u32 txsc:2; | ||
626 | u32 stbc:2; | ||
627 | u32 txshort:1; | ||
628 | u32 txbw:1; | ||
629 | u32 rtsshort:1; | ||
630 | u32 rtsbw:1; | ||
631 | u32 rtssc:2; | ||
632 | u32 rtsstbc:2; | ||
633 | |||
634 | u32 txrate:6; | ||
635 | u32 shortgi:1; | ||
636 | u32 ccxt:1; | ||
637 | u32 txrate_fb_lmt:5; | ||
638 | u32 rtsrate_fb_lmt:4; | ||
639 | u32 retrylmt_en:1; | ||
640 | u32 txretrylmt:6; | ||
641 | u32 usb_txaggnum:8; | ||
642 | |||
643 | u32 txagca:5; | ||
644 | u32 txagcb:5; | ||
645 | u32 usemaxlen:1; | ||
646 | u32 maxaggnum:5; | ||
647 | u32 mcsg1maxlen:4; | ||
648 | u32 mcsg2maxlen:4; | ||
649 | u32 mcsg3maxlen:4; | ||
650 | u32 mcs7sgimaxlen:4; | ||
651 | |||
652 | u32 txbuffersize:16; | ||
653 | u32 mcsg4maxlen:4; | ||
654 | u32 mcsg5maxlen:4; | ||
655 | u32 mcsg6maxlen:4; | ||
656 | u32 mcsg15sgimaxlen:4; | ||
657 | |||
658 | u32 txbuffaddr; | ||
659 | u32 txbufferaddr64; | ||
660 | u32 nextdescaddress; | ||
661 | u32 nextdescaddress64; | ||
662 | |||
663 | u32 reserve_pass_pcie_mm_limit[4]; | ||
664 | } __packed; | ||
665 | |||
666 | struct rx_desc_92c { | ||
667 | u32 length:14; | ||
668 | u32 crc32:1; | ||
669 | u32 icverror:1; | ||
670 | u32 drv_infosize:4; | ||
671 | u32 security:3; | ||
672 | u32 qos:1; | ||
673 | u32 shift:2; | ||
674 | u32 phystatus:1; | ||
675 | u32 swdec:1; | ||
676 | u32 lastseg:1; | ||
677 | u32 firstseg:1; | ||
678 | u32 eor:1; | ||
679 | u32 own:1; | ||
680 | |||
681 | u32 macid:5; | ||
682 | u32 tid:4; | ||
683 | u32 hwrsvd:5; | ||
684 | u32 paggr:1; | ||
685 | u32 faggr:1; | ||
686 | u32 a1_fit:4; | ||
687 | u32 a2_fit:4; | ||
688 | u32 pam:1; | ||
689 | u32 pwr:1; | ||
690 | u32 moredata:1; | ||
691 | u32 morefrag:1; | ||
692 | u32 type:2; | ||
693 | u32 mc:1; | ||
694 | u32 bc:1; | ||
695 | |||
696 | u32 seq:12; | ||
697 | u32 frag:4; | ||
698 | u32 nextpktlen:14; | ||
699 | u32 nextind:1; | ||
700 | u32 rsvd:1; | ||
701 | |||
702 | u32 rxmcs:6; | ||
703 | u32 rxht:1; | ||
704 | u32 amsdu:1; | ||
705 | u32 splcp:1; | ||
706 | u32 bandwidth:1; | ||
707 | u32 htc:1; | ||
708 | u32 tcpchk_rpt:1; | ||
709 | u32 ipcchk_rpt:1; | ||
710 | u32 tcpchk_valid:1; | ||
711 | u32 hwpcerr:1; | ||
712 | u32 hwpcind:1; | ||
713 | u32 iv0:16; | ||
714 | |||
715 | u32 iv1; | ||
716 | |||
717 | u32 tsfl; | ||
718 | |||
719 | u32 bufferaddress; | ||
720 | u32 bufferaddress64; | ||
721 | |||
722 | } __packed; | ||
723 | |||
724 | void rtl92ce_tx_fill_desc(struct ieee80211_hw *hw, | ||
725 | struct ieee80211_hdr *hdr, | ||
726 | u8 *pdesc, struct ieee80211_tx_info *info, | ||
727 | struct sk_buff *skb, u8 hw_queue, | ||
728 | struct rtl_tcb_desc *ptcb_desc); | ||
729 | bool rtl92ce_rx_query_desc(struct ieee80211_hw *hw, | ||
730 | struct rtl_stats *stats, | ||
731 | struct ieee80211_rx_status *rx_status, | ||
732 | u8 *pdesc, struct sk_buff *skb); | ||
733 | void rtl92ce_set_desc(u8 *pdesc, bool istx, u8 desc_name, u8 *val); | ||
734 | u32 rtl92ce_get_desc(u8 *pdesc, bool istx, u8 desc_name); | ||
735 | void rtl92ce_tx_polling(struct ieee80211_hw *hw, u8 hw_queue); | ||
736 | void rtl92ce_tx_fill_cmddesc(struct ieee80211_hw *hw, u8 *pdesc, | ||
737 | bool b_firstseg, bool b_lastseg, | ||
738 | struct sk_buff *skb); | ||
739 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/Makefile b/drivers/net/wireless/rtlwifi/rtl8192cu/Makefile new file mode 100644 index 000000000000..ad2de6b839ef --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/Makefile | |||
@@ -0,0 +1,14 @@ | |||
1 | rtl8192cu-objs := \ | ||
2 | dm.o \ | ||
3 | hw.o \ | ||
4 | led.o \ | ||
5 | mac.o \ | ||
6 | phy.o \ | ||
7 | rf.o \ | ||
8 | sw.o \ | ||
9 | table.o \ | ||
10 | trx.o | ||
11 | |||
12 | obj-$(CONFIG_RTL8192CU) += rtl8192cu.o | ||
13 | |||
14 | ccflags-y += -D__CHECK_ENDIAN__ | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/def.h b/drivers/net/wireless/rtlwifi/rtl8192cu/def.h new file mode 100644 index 000000000000..c54940ea72fe --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/def.h | |||
@@ -0,0 +1,62 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../rtl8192ce/def.h" | ||
31 | |||
32 | /*------------------------------------------------------------------------- | ||
33 | * Chip specific | ||
34 | *-------------------------------------------------------------------------*/ | ||
35 | #define CHIP_8723 BIT(2) /* RTL8723 With BT feature */ | ||
36 | #define CHIP_8723_DRV_REV BIT(3) /* RTL8723 Driver Revised */ | ||
37 | #define NORMAL_CHIP BIT(4) | ||
38 | #define CHIP_VENDOR_UMC BIT(5) | ||
39 | #define CHIP_VENDOR_UMC_B_CUT BIT(6) | ||
40 | |||
41 | #define IS_NORMAL_CHIP(version) \ | ||
42 | (((version) & NORMAL_CHIP) ? true : false) | ||
43 | |||
44 | #define IS_8723_SERIES(version) \ | ||
45 | (((version) & CHIP_8723) ? true : false) | ||
46 | |||
47 | #define IS_92C_1T2R(version) \ | ||
48 | (((version) & CHIP_92C) && ((version) & CHIP_92C_1T2R)) | ||
49 | |||
50 | #define IS_VENDOR_UMC(version) \ | ||
51 | (((version) & CHIP_VENDOR_UMC) ? true : false) | ||
52 | |||
53 | #define IS_VENDOR_UMC_A_CUT(version) \ | ||
54 | (((version) & CHIP_VENDOR_UMC) ? (((version) & (BIT(6) | BIT(7))) ? \ | ||
55 | false : true) : false) | ||
56 | |||
57 | #define IS_VENDOR_8723_A_CUT(version) \ | ||
58 | (((version) & CHIP_VENDOR_UMC) ? (((version) & (BIT(6))) ? \ | ||
59 | false : true) : false) | ||
60 | |||
61 | #define CHIP_BONDING_92C_1T2R 0x1 | ||
62 | #define CHIP_BONDING_IDENTIFIER(_value) (((_value) >> 22) & 0x3) | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/dm.c b/drivers/net/wireless/rtlwifi/rtl8192cu/dm.c new file mode 100644 index 000000000000..f311baee668d --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/dm.c | |||
@@ -0,0 +1,113 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../base.h" | ||
32 | #include "reg.h" | ||
33 | #include "def.h" | ||
34 | #include "phy.h" | ||
35 | #include "dm.h" | ||
36 | |||
37 | void rtl92cu_dm_dynamic_txpower(struct ieee80211_hw *hw) | ||
38 | { | ||
39 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
40 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
41 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
42 | long undecorated_smoothed_pwdb; | ||
43 | |||
44 | if (!rtlpriv->dm.dynamic_txpower_enable) | ||
45 | return; | ||
46 | |||
47 | if (rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) { | ||
48 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
49 | return; | ||
50 | } | ||
51 | |||
52 | if ((mac->link_state < MAC80211_LINKED) && | ||
53 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0)) { | ||
54 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
55 | ("Not connected to any\n")); | ||
56 | |||
57 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
58 | |||
59 | rtlpriv->dm.last_dtp_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
60 | return; | ||
61 | } | ||
62 | |||
63 | if (mac->link_state >= MAC80211_LINKED) { | ||
64 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
65 | undecorated_smoothed_pwdb = | ||
66 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
67 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
68 | ("AP Client PWDB = 0x%lx\n", | ||
69 | undecorated_smoothed_pwdb)); | ||
70 | } else { | ||
71 | undecorated_smoothed_pwdb = | ||
72 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
73 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
74 | ("STA Default Port PWDB = 0x%lx\n", | ||
75 | undecorated_smoothed_pwdb)); | ||
76 | } | ||
77 | } else { | ||
78 | undecorated_smoothed_pwdb = | ||
79 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
80 | |||
81 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
82 | ("AP Ext Port PWDB = 0x%lx\n", | ||
83 | undecorated_smoothed_pwdb)); | ||
84 | } | ||
85 | |||
86 | if (undecorated_smoothed_pwdb >= TX_POWER_NEAR_FIELD_THRESH_LVL2) { | ||
87 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
88 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
89 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x0)\n")); | ||
90 | } else if ((undecorated_smoothed_pwdb < | ||
91 | (TX_POWER_NEAR_FIELD_THRESH_LVL2 - 3)) && | ||
92 | (undecorated_smoothed_pwdb >= | ||
93 | TX_POWER_NEAR_FIELD_THRESH_LVL1)) { | ||
94 | |||
95 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_LEVEL1; | ||
96 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
97 | ("TXHIGHPWRLEVEL_LEVEL1 (TxPwr=0x10)\n")); | ||
98 | } else if (undecorated_smoothed_pwdb < | ||
99 | (TX_POWER_NEAR_FIELD_THRESH_LVL1 - 5)) { | ||
100 | rtlpriv->dm.dynamic_txhighpower_lvl = TXHIGHPWRLEVEL_NORMAL; | ||
101 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
102 | ("TXHIGHPWRLEVEL_NORMAL\n")); | ||
103 | } | ||
104 | |||
105 | if ((rtlpriv->dm.dynamic_txhighpower_lvl != rtlpriv->dm.last_dtp_lvl)) { | ||
106 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
107 | ("PHY_SetTxPowerLevel8192S() Channel = %d\n", | ||
108 | rtlphy->current_channel)); | ||
109 | rtl92c_phy_set_txpower_level(hw, rtlphy->current_channel); | ||
110 | } | ||
111 | |||
112 | rtlpriv->dm.last_dtp_lvl = rtlpriv->dm.dynamic_txhighpower_lvl; | ||
113 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/dm.h b/drivers/net/wireless/rtlwifi/rtl8192cu/dm.h new file mode 100644 index 000000000000..7f966c666b5a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/dm.h | |||
@@ -0,0 +1,32 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../rtl8192ce/dm.h" | ||
31 | |||
32 | void rtl92cu_dm_dynamic_txpower(struct ieee80211_hw *hw); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/hw.c b/drivers/net/wireless/rtlwifi/rtl8192cu/hw.c new file mode 100644 index 000000000000..52e2af58c1ed --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/hw.c | |||
@@ -0,0 +1,2507 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../efuse.h" | ||
32 | #include "../base.h" | ||
33 | #include "../cam.h" | ||
34 | #include "../ps.h" | ||
35 | #include "../usb.h" | ||
36 | #include "reg.h" | ||
37 | #include "def.h" | ||
38 | #include "phy.h" | ||
39 | #include "mac.h" | ||
40 | #include "dm.h" | ||
41 | #include "hw.h" | ||
42 | #include "../rtl8192ce/hw.h" | ||
43 | #include "trx.h" | ||
44 | #include "led.h" | ||
45 | #include "table.h" | ||
46 | |||
47 | static void _rtl92cu_phy_param_tab_init(struct ieee80211_hw *hw) | ||
48 | { | ||
49 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
50 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
51 | struct rtl_efuse *rtlefuse = rtl_efuse(rtlpriv); | ||
52 | |||
53 | rtlphy->hwparam_tables[MAC_REG].length = RTL8192CUMAC_2T_ARRAYLENGTH; | ||
54 | rtlphy->hwparam_tables[MAC_REG].pdata = RTL8192CUMAC_2T_ARRAY; | ||
55 | if (IS_HIGHT_PA(rtlefuse->board_type)) { | ||
56 | rtlphy->hwparam_tables[PHY_REG_PG].length = | ||
57 | RTL8192CUPHY_REG_Array_PG_HPLength; | ||
58 | rtlphy->hwparam_tables[PHY_REG_PG].pdata = | ||
59 | RTL8192CUPHY_REG_Array_PG_HP; | ||
60 | } else { | ||
61 | rtlphy->hwparam_tables[PHY_REG_PG].length = | ||
62 | RTL8192CUPHY_REG_ARRAY_PGLENGTH; | ||
63 | rtlphy->hwparam_tables[PHY_REG_PG].pdata = | ||
64 | RTL8192CUPHY_REG_ARRAY_PG; | ||
65 | } | ||
66 | /* 2T */ | ||
67 | rtlphy->hwparam_tables[PHY_REG_2T].length = | ||
68 | RTL8192CUPHY_REG_2TARRAY_LENGTH; | ||
69 | rtlphy->hwparam_tables[PHY_REG_2T].pdata = | ||
70 | RTL8192CUPHY_REG_2TARRAY; | ||
71 | rtlphy->hwparam_tables[RADIOA_2T].length = | ||
72 | RTL8192CURADIOA_2TARRAYLENGTH; | ||
73 | rtlphy->hwparam_tables[RADIOA_2T].pdata = | ||
74 | RTL8192CURADIOA_2TARRAY; | ||
75 | rtlphy->hwparam_tables[RADIOB_2T].length = | ||
76 | RTL8192CURADIOB_2TARRAYLENGTH; | ||
77 | rtlphy->hwparam_tables[RADIOB_2T].pdata = | ||
78 | RTL8192CU_RADIOB_2TARRAY; | ||
79 | rtlphy->hwparam_tables[AGCTAB_2T].length = | ||
80 | RTL8192CUAGCTAB_2TARRAYLENGTH; | ||
81 | rtlphy->hwparam_tables[AGCTAB_2T].pdata = | ||
82 | RTL8192CUAGCTAB_2TARRAY; | ||
83 | /* 1T */ | ||
84 | if (IS_HIGHT_PA(rtlefuse->board_type)) { | ||
85 | rtlphy->hwparam_tables[PHY_REG_1T].length = | ||
86 | RTL8192CUPHY_REG_1T_HPArrayLength; | ||
87 | rtlphy->hwparam_tables[PHY_REG_1T].pdata = | ||
88 | RTL8192CUPHY_REG_1T_HPArray; | ||
89 | rtlphy->hwparam_tables[RADIOA_1T].length = | ||
90 | RTL8192CURadioA_1T_HPArrayLength; | ||
91 | rtlphy->hwparam_tables[RADIOA_1T].pdata = | ||
92 | RTL8192CURadioA_1T_HPArray; | ||
93 | rtlphy->hwparam_tables[RADIOB_1T].length = | ||
94 | RTL8192CURADIOB_1TARRAYLENGTH; | ||
95 | rtlphy->hwparam_tables[RADIOB_1T].pdata = | ||
96 | RTL8192CU_RADIOB_1TARRAY; | ||
97 | rtlphy->hwparam_tables[AGCTAB_1T].length = | ||
98 | RTL8192CUAGCTAB_1T_HPArrayLength; | ||
99 | rtlphy->hwparam_tables[AGCTAB_1T].pdata = | ||
100 | Rtl8192CUAGCTAB_1T_HPArray; | ||
101 | } else { | ||
102 | rtlphy->hwparam_tables[PHY_REG_1T].length = | ||
103 | RTL8192CUPHY_REG_1TARRAY_LENGTH; | ||
104 | rtlphy->hwparam_tables[PHY_REG_1T].pdata = | ||
105 | RTL8192CUPHY_REG_1TARRAY; | ||
106 | rtlphy->hwparam_tables[RADIOA_1T].length = | ||
107 | RTL8192CURADIOA_1TARRAYLENGTH; | ||
108 | rtlphy->hwparam_tables[RADIOA_1T].pdata = | ||
109 | RTL8192CU_RADIOA_1TARRAY; | ||
110 | rtlphy->hwparam_tables[RADIOB_1T].length = | ||
111 | RTL8192CURADIOB_1TARRAYLENGTH; | ||
112 | rtlphy->hwparam_tables[RADIOB_1T].pdata = | ||
113 | RTL8192CU_RADIOB_1TARRAY; | ||
114 | rtlphy->hwparam_tables[AGCTAB_1T].length = | ||
115 | RTL8192CUAGCTAB_1TARRAYLENGTH; | ||
116 | rtlphy->hwparam_tables[AGCTAB_1T].pdata = | ||
117 | RTL8192CUAGCTAB_1TARRAY; | ||
118 | } | ||
119 | } | ||
120 | |||
121 | static void _rtl92cu_read_txpower_info_from_hwpg(struct ieee80211_hw *hw, | ||
122 | bool autoload_fail, | ||
123 | u8 *hwinfo) | ||
124 | { | ||
125 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
126 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
127 | u8 rf_path, index, tempval; | ||
128 | u16 i; | ||
129 | |||
130 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
131 | for (i = 0; i < 3; i++) { | ||
132 | if (!autoload_fail) { | ||
133 | rtlefuse-> | ||
134 | eeprom_chnlarea_txpwr_cck[rf_path][i] = | ||
135 | hwinfo[EEPROM_TXPOWERCCK + rf_path * 3 + i]; | ||
136 | rtlefuse-> | ||
137 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i] = | ||
138 | hwinfo[EEPROM_TXPOWERHT40_1S + rf_path * 3 + | ||
139 | i]; | ||
140 | } else { | ||
141 | rtlefuse-> | ||
142 | eeprom_chnlarea_txpwr_cck[rf_path][i] = | ||
143 | EEPROM_DEFAULT_TXPOWERLEVEL; | ||
144 | rtlefuse-> | ||
145 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i] = | ||
146 | EEPROM_DEFAULT_TXPOWERLEVEL; | ||
147 | } | ||
148 | } | ||
149 | } | ||
150 | for (i = 0; i < 3; i++) { | ||
151 | if (!autoload_fail) | ||
152 | tempval = hwinfo[EEPROM_TXPOWERHT40_2SDIFF + i]; | ||
153 | else | ||
154 | tempval = EEPROM_DEFAULT_HT40_2SDIFF; | ||
155 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif[RF90_PATH_A][i] = | ||
156 | (tempval & 0xf); | ||
157 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif[RF90_PATH_B][i] = | ||
158 | ((tempval & 0xf0) >> 4); | ||
159 | } | ||
160 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
161 | for (i = 0; i < 3; i++) | ||
162 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
163 | ("RF(%d) EEPROM CCK Area(%d) = 0x%x\n", rf_path, | ||
164 | i, rtlefuse-> | ||
165 | eeprom_chnlarea_txpwr_cck[rf_path][i])); | ||
166 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
167 | for (i = 0; i < 3; i++) | ||
168 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
169 | ("RF(%d) EEPROM HT40 1S Area(%d) = 0x%x\n", | ||
170 | rf_path, i, | ||
171 | rtlefuse-> | ||
172 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][i])); | ||
173 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
174 | for (i = 0; i < 3; i++) | ||
175 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
176 | ("RF(%d) EEPROM HT40 2S Diff Area(%d) = 0x%x\n", | ||
177 | rf_path, i, | ||
178 | rtlefuse-> | ||
179 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path] | ||
180 | [i])); | ||
181 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
182 | for (i = 0; i < 14; i++) { | ||
183 | index = _rtl92c_get_chnl_group((u8) i); | ||
184 | rtlefuse->txpwrlevel_cck[rf_path][i] = | ||
185 | rtlefuse->eeprom_chnlarea_txpwr_cck[rf_path][index]; | ||
186 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i] = | ||
187 | rtlefuse-> | ||
188 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][index]; | ||
189 | if ((rtlefuse-> | ||
190 | eeprom_chnlarea_txpwr_ht40_1s[rf_path][index] - | ||
191 | rtlefuse-> | ||
192 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path][index]) | ||
193 | > 0) { | ||
194 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i] = | ||
195 | rtlefuse-> | ||
196 | eeprom_chnlarea_txpwr_ht40_1s[rf_path] | ||
197 | [index] - rtlefuse-> | ||
198 | eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path] | ||
199 | [index]; | ||
200 | } else { | ||
201 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i] = 0; | ||
202 | } | ||
203 | } | ||
204 | for (i = 0; i < 14; i++) { | ||
205 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
206 | ("RF(%d)-Ch(%d) [CCK / HT40_1S / HT40_2S] = " | ||
207 | "[0x%x / 0x%x / 0x%x]\n", rf_path, i, | ||
208 | rtlefuse->txpwrlevel_cck[rf_path][i], | ||
209 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i], | ||
210 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i])); | ||
211 | } | ||
212 | } | ||
213 | for (i = 0; i < 3; i++) { | ||
214 | if (!autoload_fail) { | ||
215 | rtlefuse->eeprom_pwrlimit_ht40[i] = | ||
216 | hwinfo[EEPROM_TXPWR_GROUP + i]; | ||
217 | rtlefuse->eeprom_pwrlimit_ht20[i] = | ||
218 | hwinfo[EEPROM_TXPWR_GROUP + 3 + i]; | ||
219 | } else { | ||
220 | rtlefuse->eeprom_pwrlimit_ht40[i] = 0; | ||
221 | rtlefuse->eeprom_pwrlimit_ht20[i] = 0; | ||
222 | } | ||
223 | } | ||
224 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
225 | for (i = 0; i < 14; i++) { | ||
226 | index = _rtl92c_get_chnl_group((u8) i); | ||
227 | if (rf_path == RF90_PATH_A) { | ||
228 | rtlefuse->pwrgroup_ht20[rf_path][i] = | ||
229 | (rtlefuse->eeprom_pwrlimit_ht20[index] | ||
230 | & 0xf); | ||
231 | rtlefuse->pwrgroup_ht40[rf_path][i] = | ||
232 | (rtlefuse->eeprom_pwrlimit_ht40[index] | ||
233 | & 0xf); | ||
234 | } else if (rf_path == RF90_PATH_B) { | ||
235 | rtlefuse->pwrgroup_ht20[rf_path][i] = | ||
236 | ((rtlefuse->eeprom_pwrlimit_ht20[index] | ||
237 | & 0xf0) >> 4); | ||
238 | rtlefuse->pwrgroup_ht40[rf_path][i] = | ||
239 | ((rtlefuse->eeprom_pwrlimit_ht40[index] | ||
240 | & 0xf0) >> 4); | ||
241 | } | ||
242 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
243 | ("RF-%d pwrgroup_ht20[%d] = 0x%x\n", | ||
244 | rf_path, i, | ||
245 | rtlefuse->pwrgroup_ht20[rf_path][i])); | ||
246 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
247 | ("RF-%d pwrgroup_ht40[%d] = 0x%x\n", | ||
248 | rf_path, i, | ||
249 | rtlefuse->pwrgroup_ht40[rf_path][i])); | ||
250 | } | ||
251 | } | ||
252 | for (i = 0; i < 14; i++) { | ||
253 | index = _rtl92c_get_chnl_group((u8) i); | ||
254 | if (!autoload_fail) | ||
255 | tempval = hwinfo[EEPROM_TXPOWERHT20DIFF + index]; | ||
256 | else | ||
257 | tempval = EEPROM_DEFAULT_HT20_DIFF; | ||
258 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] = (tempval & 0xF); | ||
259 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] = | ||
260 | ((tempval >> 4) & 0xF); | ||
261 | if (rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] & BIT(3)) | ||
262 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] |= 0xF0; | ||
263 | if (rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] & BIT(3)) | ||
264 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] |= 0xF0; | ||
265 | index = _rtl92c_get_chnl_group((u8) i); | ||
266 | if (!autoload_fail) | ||
267 | tempval = hwinfo[EEPROM_TXPOWER_OFDMDIFF + index]; | ||
268 | else | ||
269 | tempval = EEPROM_DEFAULT_LEGACYHTTXPOWERDIFF; | ||
270 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i] = (tempval & 0xF); | ||
271 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i] = | ||
272 | ((tempval >> 4) & 0xF); | ||
273 | } | ||
274 | rtlefuse->legacy_ht_txpowerdiff = | ||
275 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][7]; | ||
276 | for (i = 0; i < 14; i++) | ||
277 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
278 | ("RF-A Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
279 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i])); | ||
280 | for (i = 0; i < 14; i++) | ||
281 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
282 | ("RF-A Legacy to Ht40 Diff[%d] = 0x%x\n", i, | ||
283 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i])); | ||
284 | for (i = 0; i < 14; i++) | ||
285 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
286 | ("RF-B Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
287 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i])); | ||
288 | for (i = 0; i < 14; i++) | ||
289 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
290 | ("RF-B Legacy to HT40 Diff[%d] = 0x%x\n", i, | ||
291 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i])); | ||
292 | if (!autoload_fail) | ||
293 | rtlefuse->eeprom_regulatory = (hwinfo[RF_OPTION1] & 0x7); | ||
294 | else | ||
295 | rtlefuse->eeprom_regulatory = 0; | ||
296 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
297 | ("eeprom_regulatory = 0x%x\n", rtlefuse->eeprom_regulatory)); | ||
298 | if (!autoload_fail) { | ||
299 | rtlefuse->eeprom_tssi[RF90_PATH_A] = hwinfo[EEPROM_TSSI_A]; | ||
300 | rtlefuse->eeprom_tssi[RF90_PATH_B] = hwinfo[EEPROM_TSSI_B]; | ||
301 | } else { | ||
302 | rtlefuse->eeprom_tssi[RF90_PATH_A] = EEPROM_DEFAULT_TSSI; | ||
303 | rtlefuse->eeprom_tssi[RF90_PATH_B] = EEPROM_DEFAULT_TSSI; | ||
304 | } | ||
305 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
306 | ("TSSI_A = 0x%x, TSSI_B = 0x%x\n", | ||
307 | rtlefuse->eeprom_tssi[RF90_PATH_A], | ||
308 | rtlefuse->eeprom_tssi[RF90_PATH_B])); | ||
309 | if (!autoload_fail) | ||
310 | tempval = hwinfo[EEPROM_THERMAL_METER]; | ||
311 | else | ||
312 | tempval = EEPROM_DEFAULT_THERMALMETER; | ||
313 | rtlefuse->eeprom_thermalmeter = (tempval & 0x1f); | ||
314 | if (rtlefuse->eeprom_thermalmeter < 0x06 || | ||
315 | rtlefuse->eeprom_thermalmeter > 0x1c) | ||
316 | rtlefuse->eeprom_thermalmeter = 0x12; | ||
317 | if (rtlefuse->eeprom_thermalmeter == 0x1f || autoload_fail) | ||
318 | rtlefuse->apk_thermalmeterignore = true; | ||
319 | rtlefuse->thermalmeter[0] = rtlefuse->eeprom_thermalmeter; | ||
320 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
321 | ("thermalmeter = 0x%x\n", rtlefuse->eeprom_thermalmeter)); | ||
322 | } | ||
323 | |||
324 | static void _rtl92cu_read_board_type(struct ieee80211_hw *hw, u8 *contents) | ||
325 | { | ||
326 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
327 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
328 | u8 boardType; | ||
329 | |||
330 | if (IS_NORMAL_CHIP(rtlhal->version)) { | ||
331 | boardType = ((contents[EEPROM_RF_OPT1]) & | ||
332 | BOARD_TYPE_NORMAL_MASK) >> 5; /*bit[7:5]*/ | ||
333 | } else { | ||
334 | boardType = contents[EEPROM_RF_OPT4]; | ||
335 | boardType &= BOARD_TYPE_TEST_MASK; | ||
336 | } | ||
337 | rtlefuse->board_type = boardType; | ||
338 | if (IS_HIGHT_PA(rtlefuse->board_type)) | ||
339 | rtlefuse->external_pa = 1; | ||
340 | printk(KERN_INFO "rtl8192cu: Board Type %x\n", rtlefuse->board_type); | ||
341 | |||
342 | #ifdef CONFIG_ANTENNA_DIVERSITY | ||
343 | /* Antenna Diversity setting. */ | ||
344 | if (registry_par->antdiv_cfg == 2) /* 2: From Efuse */ | ||
345 | rtl_efuse->antenna_cfg = (contents[EEPROM_RF_OPT1]&0x18)>>3; | ||
346 | else | ||
347 | rtl_efuse->antenna_cfg = registry_par->antdiv_cfg; /* 0:OFF, */ | ||
348 | |||
349 | printk(KERN_INFO "rtl8192cu: Antenna Config %x\n", | ||
350 | rtl_efuse->antenna_cfg); | ||
351 | #endif | ||
352 | } | ||
353 | |||
354 | #ifdef CONFIG_BT_COEXIST | ||
355 | static void _update_bt_param(_adapter *padapter) | ||
356 | { | ||
357 | struct btcoexist_priv *pbtpriv = &(padapter->halpriv.bt_coexist); | ||
358 | struct registry_priv *registry_par = &padapter->registrypriv; | ||
359 | if (2 != registry_par->bt_iso) { | ||
360 | /* 0:Low, 1:High, 2:From Efuse */ | ||
361 | pbtpriv->BT_Ant_isolation = registry_par->bt_iso; | ||
362 | } | ||
363 | if (registry_par->bt_sco == 1) { | ||
364 | /* 0:Idle, 1:None-SCO, 2:SCO, 3:From Counter, 4.Busy, | ||
365 | * 5.OtherBusy */ | ||
366 | pbtpriv->BT_Service = BT_OtherAction; | ||
367 | } else if (registry_par->bt_sco == 2) { | ||
368 | pbtpriv->BT_Service = BT_SCO; | ||
369 | } else if (registry_par->bt_sco == 4) { | ||
370 | pbtpriv->BT_Service = BT_Busy; | ||
371 | } else if (registry_par->bt_sco == 5) { | ||
372 | pbtpriv->BT_Service = BT_OtherBusy; | ||
373 | } else { | ||
374 | pbtpriv->BT_Service = BT_Idle; | ||
375 | } | ||
376 | pbtpriv->BT_Ampdu = registry_par->bt_ampdu; | ||
377 | pbtpriv->bCOBT = _TRUE; | ||
378 | pbtpriv->BtEdcaUL = 0; | ||
379 | pbtpriv->BtEdcaDL = 0; | ||
380 | pbtpriv->BtRssiState = 0xff; | ||
381 | pbtpriv->bInitSet = _FALSE; | ||
382 | pbtpriv->bBTBusyTraffic = _FALSE; | ||
383 | pbtpriv->bBTTrafficModeSet = _FALSE; | ||
384 | pbtpriv->bBTNonTrafficModeSet = _FALSE; | ||
385 | pbtpriv->CurrentState = 0; | ||
386 | pbtpriv->PreviousState = 0; | ||
387 | printk(KERN_INFO "rtl8192cu: BT Coexistance = %s\n", | ||
388 | (pbtpriv->BT_Coexist == _TRUE) ? "enable" : "disable"); | ||
389 | if (pbtpriv->BT_Coexist) { | ||
390 | if (pbtpriv->BT_Ant_Num == Ant_x2) | ||
391 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
392 | "Ant_Num = Antx2\n"); | ||
393 | else if (pbtpriv->BT_Ant_Num == Ant_x1) | ||
394 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
395 | "Ant_Num = Antx1\n"); | ||
396 | switch (pbtpriv->BT_CoexistType) { | ||
397 | case BT_2Wire: | ||
398 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
399 | "CoexistType = BT_2Wire\n"); | ||
400 | break; | ||
401 | case BT_ISSC_3Wire: | ||
402 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
403 | "CoexistType = BT_ISSC_3Wire\n"); | ||
404 | break; | ||
405 | case BT_Accel: | ||
406 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
407 | "CoexistType = BT_Accel\n"); | ||
408 | break; | ||
409 | case BT_CSR_BC4: | ||
410 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
411 | "CoexistType = BT_CSR_BC4\n"); | ||
412 | break; | ||
413 | case BT_CSR_BC8: | ||
414 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
415 | "CoexistType = BT_CSR_BC8\n"); | ||
416 | break; | ||
417 | case BT_RTL8756: | ||
418 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
419 | "CoexistType = BT_RTL8756\n"); | ||
420 | break; | ||
421 | default: | ||
422 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_" | ||
423 | "CoexistType = Unknown\n"); | ||
424 | break; | ||
425 | } | ||
426 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Ant_isolation = %d\n", | ||
427 | pbtpriv->BT_Ant_isolation); | ||
428 | switch (pbtpriv->BT_Service) { | ||
429 | case BT_OtherAction: | ||
430 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Service = " | ||
431 | "BT_OtherAction\n"); | ||
432 | break; | ||
433 | case BT_SCO: | ||
434 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Service = " | ||
435 | "BT_SCO\n"); | ||
436 | break; | ||
437 | case BT_Busy: | ||
438 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Service = " | ||
439 | "BT_Busy\n"); | ||
440 | break; | ||
441 | case BT_OtherBusy: | ||
442 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Service = " | ||
443 | "BT_OtherBusy\n"); | ||
444 | break; | ||
445 | default: | ||
446 | printk(KERN_INFO "rtl8192cu: BlueTooth BT_Service = " | ||
447 | "BT_Idle\n"); | ||
448 | break; | ||
449 | } | ||
450 | printk(KERN_INFO "rtl8192cu: BT_RadioSharedType = 0x%x\n", | ||
451 | pbtpriv->BT_RadioSharedType); | ||
452 | } | ||
453 | } | ||
454 | |||
455 | #define GET_BT_COEXIST(priv) (&priv->bt_coexist) | ||
456 | |||
457 | static void _rtl92cu_read_bluetooth_coexistInfo(struct ieee80211_hw *hw, | ||
458 | u8 *contents, | ||
459 | bool bautoloadfailed); | ||
460 | { | ||
461 | HAL_DATA_TYPE *pHalData = GET_HAL_DATA(Adapter); | ||
462 | bool isNormal = IS_NORMAL_CHIP(pHalData->VersionID); | ||
463 | struct btcoexist_priv *pbtpriv = &pHalData->bt_coexist; | ||
464 | u8 rf_opt4; | ||
465 | |||
466 | _rtw_memset(pbtpriv, 0, sizeof(struct btcoexist_priv)); | ||
467 | if (AutoloadFail) { | ||
468 | pbtpriv->BT_Coexist = _FALSE; | ||
469 | pbtpriv->BT_CoexistType = BT_2Wire; | ||
470 | pbtpriv->BT_Ant_Num = Ant_x2; | ||
471 | pbtpriv->BT_Ant_isolation = 0; | ||
472 | pbtpriv->BT_RadioSharedType = BT_Radio_Shared; | ||
473 | return; | ||
474 | } | ||
475 | if (isNormal) { | ||
476 | if (pHalData->BoardType == BOARD_USB_COMBO) | ||
477 | pbtpriv->BT_Coexist = _TRUE; | ||
478 | else | ||
479 | pbtpriv->BT_Coexist = ((PROMContent[EEPROM_RF_OPT3] & | ||
480 | 0x20) >> 5); /* bit[5] */ | ||
481 | rf_opt4 = PROMContent[EEPROM_RF_OPT4]; | ||
482 | pbtpriv->BT_CoexistType = ((rf_opt4&0xe)>>1); /* bit [3:1] */ | ||
483 | pbtpriv->BT_Ant_Num = (rf_opt4&0x1); /* bit [0] */ | ||
484 | pbtpriv->BT_Ant_isolation = ((rf_opt4&0x10)>>4); /* bit [4] */ | ||
485 | pbtpriv->BT_RadioSharedType = ((rf_opt4&0x20)>>5); /* bit [5] */ | ||
486 | } else { | ||
487 | pbtpriv->BT_Coexist = (PROMContent[EEPROM_RF_OPT4] >> 4) ? | ||
488 | _TRUE : _FALSE; | ||
489 | } | ||
490 | _update_bt_param(Adapter); | ||
491 | } | ||
492 | #endif | ||
493 | |||
494 | static void _rtl92cu_read_adapter_info(struct ieee80211_hw *hw) | ||
495 | { | ||
496 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
497 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
498 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
499 | u16 i, usvalue; | ||
500 | u8 hwinfo[HWSET_MAX_SIZE] = {0}; | ||
501 | u16 eeprom_id; | ||
502 | |||
503 | if (rtlefuse->epromtype == EEPROM_BOOT_EFUSE) { | ||
504 | rtl_efuse_shadow_map_update(hw); | ||
505 | memcpy((void *)hwinfo, | ||
506 | (void *)&rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
507 | HWSET_MAX_SIZE); | ||
508 | } else if (rtlefuse->epromtype == EEPROM_93C46) { | ||
509 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
510 | ("RTL819X Not boot from eeprom, check it !!")); | ||
511 | } | ||
512 | RT_PRINT_DATA(rtlpriv, COMP_INIT, DBG_LOUD, ("MAP\n"), | ||
513 | hwinfo, HWSET_MAX_SIZE); | ||
514 | eeprom_id = *((u16 *)&hwinfo[0]); | ||
515 | if (eeprom_id != RTL8190_EEPROM_ID) { | ||
516 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
517 | ("EEPROM ID(%#x) is invalid!!\n", eeprom_id)); | ||
518 | rtlefuse->autoload_failflag = true; | ||
519 | } else { | ||
520 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload OK\n")); | ||
521 | rtlefuse->autoload_failflag = false; | ||
522 | } | ||
523 | if (rtlefuse->autoload_failflag == true) | ||
524 | return; | ||
525 | for (i = 0; i < 6; i += 2) { | ||
526 | usvalue = *(u16 *)&hwinfo[EEPROM_MAC_ADDR + i]; | ||
527 | *((u16 *) (&rtlefuse->dev_addr[i])) = usvalue; | ||
528 | } | ||
529 | printk(KERN_INFO "rtl8192cu: MAC address: %pM\n", rtlefuse->dev_addr); | ||
530 | _rtl92cu_read_txpower_info_from_hwpg(hw, | ||
531 | rtlefuse->autoload_failflag, hwinfo); | ||
532 | rtlefuse->eeprom_vid = *(u16 *)&hwinfo[EEPROM_VID]; | ||
533 | rtlefuse->eeprom_did = *(u16 *)&hwinfo[EEPROM_DID]; | ||
534 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
535 | (" VID = 0x%02x PID = 0x%02x\n", | ||
536 | rtlefuse->eeprom_vid, rtlefuse->eeprom_did)); | ||
537 | rtlefuse->eeprom_channelplan = *(u8 *)&hwinfo[EEPROM_CHANNELPLAN]; | ||
538 | rtlefuse->eeprom_version = *(u16 *)&hwinfo[EEPROM_VERSION]; | ||
539 | rtlefuse->txpwr_fromeprom = true; | ||
540 | rtlefuse->eeprom_oemid = *(u8 *)&hwinfo[EEPROM_CUSTOMER_ID]; | ||
541 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
542 | ("EEPROM Customer ID: 0x%2x\n", rtlefuse->eeprom_oemid)); | ||
543 | if (rtlhal->oem_id == RT_CID_DEFAULT) { | ||
544 | switch (rtlefuse->eeprom_oemid) { | ||
545 | case EEPROM_CID_DEFAULT: | ||
546 | if (rtlefuse->eeprom_did == 0x8176) { | ||
547 | if ((rtlefuse->eeprom_svid == 0x103C && | ||
548 | rtlefuse->eeprom_smid == 0x1629)) | ||
549 | rtlhal->oem_id = RT_CID_819x_HP; | ||
550 | else | ||
551 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
552 | } else { | ||
553 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
554 | } | ||
555 | break; | ||
556 | case EEPROM_CID_TOSHIBA: | ||
557 | rtlhal->oem_id = RT_CID_TOSHIBA; | ||
558 | break; | ||
559 | case EEPROM_CID_QMI: | ||
560 | rtlhal->oem_id = RT_CID_819x_QMI; | ||
561 | break; | ||
562 | case EEPROM_CID_WHQL: | ||
563 | default: | ||
564 | rtlhal->oem_id = RT_CID_DEFAULT; | ||
565 | break; | ||
566 | } | ||
567 | } | ||
568 | _rtl92cu_read_board_type(hw, hwinfo); | ||
569 | #ifdef CONFIG_BT_COEXIST | ||
570 | _rtl92cu_read_bluetooth_coexistInfo(hw, hwinfo, | ||
571 | rtlefuse->autoload_failflag); | ||
572 | #endif | ||
573 | } | ||
574 | |||
575 | static void _rtl92cu_hal_customized_behavior(struct ieee80211_hw *hw) | ||
576 | { | ||
577 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
578 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
579 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
580 | |||
581 | switch (rtlhal->oem_id) { | ||
582 | case RT_CID_819x_HP: | ||
583 | usb_priv->ledctl.led_opendrain = true; | ||
584 | break; | ||
585 | case RT_CID_819x_Lenovo: | ||
586 | case RT_CID_DEFAULT: | ||
587 | case RT_CID_TOSHIBA: | ||
588 | case RT_CID_CCX: | ||
589 | case RT_CID_819x_Acer: | ||
590 | case RT_CID_WHQL: | ||
591 | default: | ||
592 | break; | ||
593 | } | ||
594 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
595 | ("RT Customized ID: 0x%02X\n", rtlhal->oem_id)); | ||
596 | } | ||
597 | |||
598 | void rtl92cu_read_eeprom_info(struct ieee80211_hw *hw) | ||
599 | { | ||
600 | |||
601 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
602 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
603 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
604 | u8 tmp_u1b; | ||
605 | |||
606 | if (!IS_NORMAL_CHIP(rtlhal->version)) | ||
607 | return; | ||
608 | tmp_u1b = rtl_read_byte(rtlpriv, REG_9346CR); | ||
609 | rtlefuse->epromtype = (tmp_u1b & BOOT_FROM_EEPROM) ? | ||
610 | EEPROM_93C46 : EEPROM_BOOT_EFUSE; | ||
611 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Boot from %s\n", | ||
612 | (tmp_u1b & BOOT_FROM_EEPROM) ? "EERROM" : "EFUSE")); | ||
613 | rtlefuse->autoload_failflag = (tmp_u1b & EEPROM_EN) ? false : true; | ||
614 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload %s\n", | ||
615 | (tmp_u1b & EEPROM_EN) ? "OK!!" : "ERR!!")); | ||
616 | _rtl92cu_read_adapter_info(hw); | ||
617 | _rtl92cu_hal_customized_behavior(hw); | ||
618 | return; | ||
619 | } | ||
620 | |||
621 | static int _rtl92cu_init_power_on(struct ieee80211_hw *hw) | ||
622 | { | ||
623 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
624 | int status = 0; | ||
625 | u16 value16; | ||
626 | u8 value8; | ||
627 | /* polling autoload done. */ | ||
628 | u32 pollingCount = 0; | ||
629 | |||
630 | do { | ||
631 | if (rtl_read_byte(rtlpriv, REG_APS_FSMCO) & PFM_ALDN) { | ||
632 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
633 | ("Autoload Done!\n")); | ||
634 | break; | ||
635 | } | ||
636 | if (pollingCount++ > 100) { | ||
637 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
638 | ("Failed to polling REG_APS_FSMCO[PFM_ALDN]" | ||
639 | " done!\n")); | ||
640 | return -ENODEV; | ||
641 | } | ||
642 | } while (true); | ||
643 | /* 0. RSV_CTRL 0x1C[7:0] = 0 unlock ISO/CLK/Power control register */ | ||
644 | rtl_write_byte(rtlpriv, REG_RSV_CTRL, 0x0); | ||
645 | /* Power on when re-enter from IPS/Radio off/card disable */ | ||
646 | /* enable SPS into PWM mode */ | ||
647 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x2b); | ||
648 | udelay(100); | ||
649 | value8 = rtl_read_byte(rtlpriv, REG_LDOV12D_CTRL); | ||
650 | if (0 == (value8 & LDV12_EN)) { | ||
651 | value8 |= LDV12_EN; | ||
652 | rtl_write_byte(rtlpriv, REG_LDOV12D_CTRL, value8); | ||
653 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
654 | (" power-on :REG_LDOV12D_CTRL Reg0x21:0x%02x.\n", | ||
655 | value8)); | ||
656 | udelay(100); | ||
657 | value8 = rtl_read_byte(rtlpriv, REG_SYS_ISO_CTRL); | ||
658 | value8 &= ~ISO_MD2PP; | ||
659 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL, value8); | ||
660 | } | ||
661 | /* auto enable WLAN */ | ||
662 | pollingCount = 0; | ||
663 | value16 = rtl_read_word(rtlpriv, REG_APS_FSMCO); | ||
664 | value16 |= APFM_ONMAC; | ||
665 | rtl_write_word(rtlpriv, REG_APS_FSMCO, value16); | ||
666 | do { | ||
667 | if (!(rtl_read_word(rtlpriv, REG_APS_FSMCO) & APFM_ONMAC)) { | ||
668 | printk(KERN_INFO "rtl8192cu: MAC auto ON okay!\n"); | ||
669 | break; | ||
670 | } | ||
671 | if (pollingCount++ > 100) { | ||
672 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
673 | ("Failed to polling REG_APS_FSMCO[APFM_ONMAC]" | ||
674 | " done!\n")); | ||
675 | return -ENODEV; | ||
676 | } | ||
677 | } while (true); | ||
678 | /* Enable Radio ,GPIO ,and LED function */ | ||
679 | rtl_write_word(rtlpriv, REG_APS_FSMCO, 0x0812); | ||
680 | /* release RF digital isolation */ | ||
681 | value16 = rtl_read_word(rtlpriv, REG_SYS_ISO_CTRL); | ||
682 | value16 &= ~ISO_DIOR; | ||
683 | rtl_write_word(rtlpriv, REG_SYS_ISO_CTRL, value16); | ||
684 | /* Reconsider when to do this operation after asking HWSD. */ | ||
685 | pollingCount = 0; | ||
686 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, (rtl_read_byte(rtlpriv, | ||
687 | REG_APSD_CTRL) & ~BIT(6))); | ||
688 | do { | ||
689 | pollingCount++; | ||
690 | } while ((pollingCount < 200) && | ||
691 | (rtl_read_byte(rtlpriv, REG_APSD_CTRL) & BIT(7))); | ||
692 | /* Enable MAC DMA/WMAC/SCHEDULE/SEC block */ | ||
693 | value16 = rtl_read_word(rtlpriv, REG_CR); | ||
694 | value16 |= (HCI_TXDMA_EN | HCI_RXDMA_EN | TXDMA_EN | RXDMA_EN | | ||
695 | PROTOCOL_EN | SCHEDULE_EN | MACTXEN | MACRXEN | ENSEC); | ||
696 | rtl_write_word(rtlpriv, REG_CR, value16); | ||
697 | return status; | ||
698 | } | ||
699 | |||
700 | static void _rtl92cu_init_queue_reserved_page(struct ieee80211_hw *hw, | ||
701 | bool wmm_enable, | ||
702 | u8 out_ep_num, | ||
703 | u8 queue_sel) | ||
704 | { | ||
705 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
706 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
707 | bool isChipN = IS_NORMAL_CHIP(rtlhal->version); | ||
708 | u32 outEPNum = (u32)out_ep_num; | ||
709 | u32 numHQ = 0; | ||
710 | u32 numLQ = 0; | ||
711 | u32 numNQ = 0; | ||
712 | u32 numPubQ; | ||
713 | u32 value32; | ||
714 | u8 value8; | ||
715 | u32 txQPageNum, txQPageUnit, txQRemainPage; | ||
716 | |||
717 | if (!wmm_enable) { | ||
718 | numPubQ = (isChipN) ? CHIP_B_PAGE_NUM_PUBQ : | ||
719 | CHIP_A_PAGE_NUM_PUBQ; | ||
720 | txQPageNum = TX_TOTAL_PAGE_NUMBER - numPubQ; | ||
721 | |||
722 | txQPageUnit = txQPageNum/outEPNum; | ||
723 | txQRemainPage = txQPageNum % outEPNum; | ||
724 | if (queue_sel & TX_SELE_HQ) | ||
725 | numHQ = txQPageUnit; | ||
726 | if (queue_sel & TX_SELE_LQ) | ||
727 | numLQ = txQPageUnit; | ||
728 | /* HIGH priority queue always present in the configuration of | ||
729 | * 2 out-ep. Remainder pages have assigned to High queue */ | ||
730 | if ((outEPNum > 1) && (txQRemainPage)) | ||
731 | numHQ += txQRemainPage; | ||
732 | /* NOTE: This step done before writting REG_RQPN. */ | ||
733 | if (isChipN) { | ||
734 | if (queue_sel & TX_SELE_NQ) | ||
735 | numNQ = txQPageUnit; | ||
736 | value8 = (u8)_NPQ(numNQ); | ||
737 | rtl_write_byte(rtlpriv, REG_RQPN_NPQ, value8); | ||
738 | } | ||
739 | } else { | ||
740 | /* for WMM ,number of out-ep must more than or equal to 2! */ | ||
741 | numPubQ = isChipN ? WMM_CHIP_B_PAGE_NUM_PUBQ : | ||
742 | WMM_CHIP_A_PAGE_NUM_PUBQ; | ||
743 | if (queue_sel & TX_SELE_HQ) { | ||
744 | numHQ = isChipN ? WMM_CHIP_B_PAGE_NUM_HPQ : | ||
745 | WMM_CHIP_A_PAGE_NUM_HPQ; | ||
746 | } | ||
747 | if (queue_sel & TX_SELE_LQ) { | ||
748 | numLQ = isChipN ? WMM_CHIP_B_PAGE_NUM_LPQ : | ||
749 | WMM_CHIP_A_PAGE_NUM_LPQ; | ||
750 | } | ||
751 | /* NOTE: This step done before writting REG_RQPN. */ | ||
752 | if (isChipN) { | ||
753 | if (queue_sel & TX_SELE_NQ) | ||
754 | numNQ = WMM_CHIP_B_PAGE_NUM_NPQ; | ||
755 | value8 = (u8)_NPQ(numNQ); | ||
756 | rtl_write_byte(rtlpriv, REG_RQPN_NPQ, value8); | ||
757 | } | ||
758 | } | ||
759 | /* TX DMA */ | ||
760 | value32 = _HPQ(numHQ) | _LPQ(numLQ) | _PUBQ(numPubQ) | LD_RQPN; | ||
761 | rtl_write_dword(rtlpriv, REG_RQPN, value32); | ||
762 | } | ||
763 | |||
764 | static void _rtl92c_init_trx_buffer(struct ieee80211_hw *hw, bool wmm_enable) | ||
765 | { | ||
766 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
767 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
768 | u8 txpktbuf_bndy; | ||
769 | u8 value8; | ||
770 | |||
771 | if (!wmm_enable) | ||
772 | txpktbuf_bndy = TX_PAGE_BOUNDARY; | ||
773 | else /* for WMM */ | ||
774 | txpktbuf_bndy = (IS_NORMAL_CHIP(rtlhal->version)) | ||
775 | ? WMM_CHIP_B_TX_PAGE_BOUNDARY | ||
776 | : WMM_CHIP_A_TX_PAGE_BOUNDARY; | ||
777 | rtl_write_byte(rtlpriv, REG_TXPKTBUF_BCNQ_BDNY, txpktbuf_bndy); | ||
778 | rtl_write_byte(rtlpriv, REG_TXPKTBUF_MGQ_BDNY, txpktbuf_bndy); | ||
779 | rtl_write_byte(rtlpriv, REG_TXPKTBUF_WMAC_LBK_BF_HD, txpktbuf_bndy); | ||
780 | rtl_write_byte(rtlpriv, REG_TRXFF_BNDY, txpktbuf_bndy); | ||
781 | rtl_write_byte(rtlpriv, REG_TDECTRL+1, txpktbuf_bndy); | ||
782 | rtl_write_word(rtlpriv, (REG_TRXFF_BNDY + 2), 0x27FF); | ||
783 | value8 = _PSRX(RX_PAGE_SIZE_REG_VALUE) | _PSTX(PBP_128); | ||
784 | rtl_write_byte(rtlpriv, REG_PBP, value8); | ||
785 | } | ||
786 | |||
787 | static void _rtl92c_init_chipN_reg_priority(struct ieee80211_hw *hw, u16 beQ, | ||
788 | u16 bkQ, u16 viQ, u16 voQ, | ||
789 | u16 mgtQ, u16 hiQ) | ||
790 | { | ||
791 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
792 | u16 value16 = (rtl_read_word(rtlpriv, REG_TRXDMA_CTRL) & 0x7); | ||
793 | |||
794 | value16 |= _TXDMA_BEQ_MAP(beQ) | _TXDMA_BKQ_MAP(bkQ) | | ||
795 | _TXDMA_VIQ_MAP(viQ) | _TXDMA_VOQ_MAP(voQ) | | ||
796 | _TXDMA_MGQ_MAP(mgtQ) | _TXDMA_HIQ_MAP(hiQ); | ||
797 | rtl_write_word(rtlpriv, REG_TRXDMA_CTRL, value16); | ||
798 | } | ||
799 | |||
800 | static void _rtl92cu_init_chipN_one_out_ep_priority(struct ieee80211_hw *hw, | ||
801 | bool wmm_enable, | ||
802 | u8 queue_sel) | ||
803 | { | ||
804 | u16 uninitialized_var(value); | ||
805 | |||
806 | switch (queue_sel) { | ||
807 | case TX_SELE_HQ: | ||
808 | value = QUEUE_HIGH; | ||
809 | break; | ||
810 | case TX_SELE_LQ: | ||
811 | value = QUEUE_LOW; | ||
812 | break; | ||
813 | case TX_SELE_NQ: | ||
814 | value = QUEUE_NORMAL; | ||
815 | break; | ||
816 | default: | ||
817 | WARN_ON(1); /* Shall not reach here! */ | ||
818 | break; | ||
819 | } | ||
820 | _rtl92c_init_chipN_reg_priority(hw, value, value, value, value, | ||
821 | value, value); | ||
822 | printk(KERN_INFO "rtl8192cu: Tx queue select: 0x%02x\n", queue_sel); | ||
823 | } | ||
824 | |||
825 | static void _rtl92cu_init_chipN_two_out_ep_priority(struct ieee80211_hw *hw, | ||
826 | bool wmm_enable, | ||
827 | u8 queue_sel) | ||
828 | { | ||
829 | u16 beQ, bkQ, viQ, voQ, mgtQ, hiQ; | ||
830 | u16 uninitialized_var(valueHi); | ||
831 | u16 uninitialized_var(valueLow); | ||
832 | |||
833 | switch (queue_sel) { | ||
834 | case (TX_SELE_HQ | TX_SELE_LQ): | ||
835 | valueHi = QUEUE_HIGH; | ||
836 | valueLow = QUEUE_LOW; | ||
837 | break; | ||
838 | case (TX_SELE_NQ | TX_SELE_LQ): | ||
839 | valueHi = QUEUE_NORMAL; | ||
840 | valueLow = QUEUE_LOW; | ||
841 | break; | ||
842 | case (TX_SELE_HQ | TX_SELE_NQ): | ||
843 | valueHi = QUEUE_HIGH; | ||
844 | valueLow = QUEUE_NORMAL; | ||
845 | break; | ||
846 | default: | ||
847 | WARN_ON(1); | ||
848 | break; | ||
849 | } | ||
850 | if (!wmm_enable) { | ||
851 | beQ = valueLow; | ||
852 | bkQ = valueLow; | ||
853 | viQ = valueHi; | ||
854 | voQ = valueHi; | ||
855 | mgtQ = valueHi; | ||
856 | hiQ = valueHi; | ||
857 | } else {/* for WMM ,CONFIG_OUT_EP_WIFI_MODE */ | ||
858 | beQ = valueHi; | ||
859 | bkQ = valueLow; | ||
860 | viQ = valueLow; | ||
861 | voQ = valueHi; | ||
862 | mgtQ = valueHi; | ||
863 | hiQ = valueHi; | ||
864 | } | ||
865 | _rtl92c_init_chipN_reg_priority(hw, beQ, bkQ, viQ, voQ, mgtQ, hiQ); | ||
866 | printk(KERN_INFO "rtl8192cu: Tx queue select: 0x%02x\n", queue_sel); | ||
867 | } | ||
868 | |||
869 | static void _rtl92cu_init_chipN_three_out_ep_priority(struct ieee80211_hw *hw, | ||
870 | bool wmm_enable, | ||
871 | u8 queue_sel) | ||
872 | { | ||
873 | u16 beQ, bkQ, viQ, voQ, mgtQ, hiQ; | ||
874 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
875 | |||
876 | if (!wmm_enable) { /* typical setting */ | ||
877 | beQ = QUEUE_LOW; | ||
878 | bkQ = QUEUE_LOW; | ||
879 | viQ = QUEUE_NORMAL; | ||
880 | voQ = QUEUE_HIGH; | ||
881 | mgtQ = QUEUE_HIGH; | ||
882 | hiQ = QUEUE_HIGH; | ||
883 | } else { /* for WMM */ | ||
884 | beQ = QUEUE_LOW; | ||
885 | bkQ = QUEUE_NORMAL; | ||
886 | viQ = QUEUE_NORMAL; | ||
887 | voQ = QUEUE_HIGH; | ||
888 | mgtQ = QUEUE_HIGH; | ||
889 | hiQ = QUEUE_HIGH; | ||
890 | } | ||
891 | _rtl92c_init_chipN_reg_priority(hw, beQ, bkQ, viQ, voQ, mgtQ, hiQ); | ||
892 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
893 | ("Tx queue select :0x%02x..\n", queue_sel)); | ||
894 | } | ||
895 | |||
896 | static void _rtl92cu_init_chipN_queue_priority(struct ieee80211_hw *hw, | ||
897 | bool wmm_enable, | ||
898 | u8 out_ep_num, | ||
899 | u8 queue_sel) | ||
900 | { | ||
901 | switch (out_ep_num) { | ||
902 | case 1: | ||
903 | _rtl92cu_init_chipN_one_out_ep_priority(hw, wmm_enable, | ||
904 | queue_sel); | ||
905 | break; | ||
906 | case 2: | ||
907 | _rtl92cu_init_chipN_two_out_ep_priority(hw, wmm_enable, | ||
908 | queue_sel); | ||
909 | break; | ||
910 | case 3: | ||
911 | _rtl92cu_init_chipN_three_out_ep_priority(hw, wmm_enable, | ||
912 | queue_sel); | ||
913 | break; | ||
914 | default: | ||
915 | WARN_ON(1); /* Shall not reach here! */ | ||
916 | break; | ||
917 | } | ||
918 | } | ||
919 | |||
920 | static void _rtl92cu_init_chipT_queue_priority(struct ieee80211_hw *hw, | ||
921 | bool wmm_enable, | ||
922 | u8 out_ep_num, | ||
923 | u8 queue_sel) | ||
924 | { | ||
925 | u8 hq_sele = 0; | ||
926 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
927 | |||
928 | switch (out_ep_num) { | ||
929 | case 2: /* (TX_SELE_HQ|TX_SELE_LQ) */ | ||
930 | if (!wmm_enable) /* typical setting */ | ||
931 | hq_sele = HQSEL_VOQ | HQSEL_VIQ | HQSEL_MGTQ | | ||
932 | HQSEL_HIQ; | ||
933 | else /* for WMM */ | ||
934 | hq_sele = HQSEL_VOQ | HQSEL_BEQ | HQSEL_MGTQ | | ||
935 | HQSEL_HIQ; | ||
936 | break; | ||
937 | case 1: | ||
938 | if (TX_SELE_LQ == queue_sel) { | ||
939 | /* map all endpoint to Low queue */ | ||
940 | hq_sele = 0; | ||
941 | } else if (TX_SELE_HQ == queue_sel) { | ||
942 | /* map all endpoint to High queue */ | ||
943 | hq_sele = HQSEL_VOQ | HQSEL_VIQ | HQSEL_BEQ | | ||
944 | HQSEL_BKQ | HQSEL_MGTQ | HQSEL_HIQ; | ||
945 | } | ||
946 | break; | ||
947 | default: | ||
948 | WARN_ON(1); /* Shall not reach here! */ | ||
949 | break; | ||
950 | } | ||
951 | rtl_write_byte(rtlpriv, (REG_TRXDMA_CTRL+1), hq_sele); | ||
952 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
953 | ("Tx queue select :0x%02x..\n", hq_sele)); | ||
954 | } | ||
955 | |||
956 | static void _rtl92cu_init_queue_priority(struct ieee80211_hw *hw, | ||
957 | bool wmm_enable, | ||
958 | u8 out_ep_num, | ||
959 | u8 queue_sel) | ||
960 | { | ||
961 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
962 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
963 | _rtl92cu_init_chipN_queue_priority(hw, wmm_enable, out_ep_num, | ||
964 | queue_sel); | ||
965 | else | ||
966 | _rtl92cu_init_chipT_queue_priority(hw, wmm_enable, out_ep_num, | ||
967 | queue_sel); | ||
968 | } | ||
969 | |||
970 | static void _rtl92cu_init_usb_aggregation(struct ieee80211_hw *hw) | ||
971 | { | ||
972 | } | ||
973 | |||
974 | static void _rtl92cu_init_wmac_setting(struct ieee80211_hw *hw) | ||
975 | { | ||
976 | u16 value16; | ||
977 | |||
978 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
979 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
980 | |||
981 | mac->rx_conf = (RCR_APM | RCR_AM | RCR_ADF | RCR_AB | RCR_APPFCS | | ||
982 | RCR_APP_ICV | RCR_AMF | RCR_HTC_LOC_CTRL | | ||
983 | RCR_APP_MIC | RCR_APP_PHYSTS | RCR_ACRC32); | ||
984 | rtl_write_dword(rtlpriv, REG_RCR, mac->rx_conf); | ||
985 | /* Accept all multicast address */ | ||
986 | rtl_write_dword(rtlpriv, REG_MAR, 0xFFFFFFFF); | ||
987 | rtl_write_dword(rtlpriv, REG_MAR + 4, 0xFFFFFFFF); | ||
988 | /* Accept all management frames */ | ||
989 | value16 = 0xFFFF; | ||
990 | rtl92c_set_mgt_filter(hw, value16); | ||
991 | /* Reject all control frame - default value is 0 */ | ||
992 | rtl92c_set_ctrl_filter(hw, 0x0); | ||
993 | /* Accept all data frames */ | ||
994 | value16 = 0xFFFF; | ||
995 | rtl92c_set_data_filter(hw, value16); | ||
996 | } | ||
997 | |||
998 | static int _rtl92cu_init_mac(struct ieee80211_hw *hw) | ||
999 | { | ||
1000 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1001 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1002 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
1003 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
1004 | int err = 0; | ||
1005 | u32 boundary = 0; | ||
1006 | u8 wmm_enable = false; /* TODO */ | ||
1007 | u8 out_ep_nums = rtlusb->out_ep_nums; | ||
1008 | u8 queue_sel = rtlusb->out_queue_sel; | ||
1009 | err = _rtl92cu_init_power_on(hw); | ||
1010 | |||
1011 | if (err) { | ||
1012 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1013 | ("Failed to init power on!\n")); | ||
1014 | return err; | ||
1015 | } | ||
1016 | if (!wmm_enable) { | ||
1017 | boundary = TX_PAGE_BOUNDARY; | ||
1018 | } else { /* for WMM */ | ||
1019 | boundary = (IS_NORMAL_CHIP(rtlhal->version)) | ||
1020 | ? WMM_CHIP_B_TX_PAGE_BOUNDARY | ||
1021 | : WMM_CHIP_A_TX_PAGE_BOUNDARY; | ||
1022 | } | ||
1023 | if (false == rtl92c_init_llt_table(hw, boundary)) { | ||
1024 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1025 | ("Failed to init LLT Table!\n")); | ||
1026 | return -EINVAL; | ||
1027 | } | ||
1028 | _rtl92cu_init_queue_reserved_page(hw, wmm_enable, out_ep_nums, | ||
1029 | queue_sel); | ||
1030 | _rtl92c_init_trx_buffer(hw, wmm_enable); | ||
1031 | _rtl92cu_init_queue_priority(hw, wmm_enable, out_ep_nums, | ||
1032 | queue_sel); | ||
1033 | /* Get Rx PHY status in order to report RSSI and others. */ | ||
1034 | rtl92c_init_driver_info_size(hw, RTL92C_DRIVER_INFO_SIZE); | ||
1035 | rtl92c_init_interrupt(hw); | ||
1036 | rtl92c_init_network_type(hw); | ||
1037 | _rtl92cu_init_wmac_setting(hw); | ||
1038 | rtl92c_init_adaptive_ctrl(hw); | ||
1039 | rtl92c_init_edca(hw); | ||
1040 | rtl92c_init_rate_fallback(hw); | ||
1041 | rtl92c_init_retry_function(hw); | ||
1042 | _rtl92cu_init_usb_aggregation(hw); | ||
1043 | rtlpriv->cfg->ops->set_bw_mode(hw, NL80211_CHAN_HT20); | ||
1044 | rtl92c_set_min_space(hw, IS_92C_SERIAL(rtlhal->version)); | ||
1045 | rtl92c_init_beacon_parameters(hw, rtlhal->version); | ||
1046 | rtl92c_init_ampdu_aggregation(hw); | ||
1047 | rtl92c_init_beacon_max_error(hw, true); | ||
1048 | return err; | ||
1049 | } | ||
1050 | |||
1051 | void rtl92cu_enable_hw_security_config(struct ieee80211_hw *hw) | ||
1052 | { | ||
1053 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1054 | u8 sec_reg_value = 0x0; | ||
1055 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1056 | |||
1057 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1058 | ("PairwiseEncAlgorithm = %d GroupEncAlgorithm = %d\n", | ||
1059 | rtlpriv->sec.pairwise_enc_algorithm, | ||
1060 | rtlpriv->sec.group_enc_algorithm)); | ||
1061 | if (rtlpriv->cfg->mod_params->sw_crypto || rtlpriv->sec.use_sw_sec) { | ||
1062 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
1063 | ("not open sw encryption\n")); | ||
1064 | return; | ||
1065 | } | ||
1066 | sec_reg_value = SCR_TxEncEnable | SCR_RxDecEnable; | ||
1067 | if (rtlpriv->sec.use_defaultkey) { | ||
1068 | sec_reg_value |= SCR_TxUseDK; | ||
1069 | sec_reg_value |= SCR_RxUseDK; | ||
1070 | } | ||
1071 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
1072 | sec_reg_value |= (SCR_RXBCUSEDK | SCR_TXBCUSEDK); | ||
1073 | rtl_write_byte(rtlpriv, REG_CR + 1, 0x02); | ||
1074 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
1075 | ("The SECR-value %x\n", sec_reg_value)); | ||
1076 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_WPA_CONFIG, &sec_reg_value); | ||
1077 | } | ||
1078 | |||
1079 | static void _rtl92cu_hw_configure(struct ieee80211_hw *hw) | ||
1080 | { | ||
1081 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1082 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
1083 | |||
1084 | /* To Fix MAC loopback mode fail. */ | ||
1085 | rtl_write_byte(rtlpriv, REG_LDOHCI12_CTRL, 0x0f); | ||
1086 | rtl_write_byte(rtlpriv, 0x15, 0xe9); | ||
1087 | /* HW SEQ CTRL */ | ||
1088 | /* set 0x0 to 0xFF by tynli. Default enable HW SEQ NUM. */ | ||
1089 | rtl_write_byte(rtlpriv, REG_HWSEQ_CTRL, 0xFF); | ||
1090 | /* fixed USB interface interference issue */ | ||
1091 | rtl_write_byte(rtlpriv, 0xfe40, 0xe0); | ||
1092 | rtl_write_byte(rtlpriv, 0xfe41, 0x8d); | ||
1093 | rtl_write_byte(rtlpriv, 0xfe42, 0x80); | ||
1094 | rtlusb->reg_bcn_ctrl_val = 0x18; | ||
1095 | rtl_write_byte(rtlpriv, REG_BCN_CTRL, (u8)rtlusb->reg_bcn_ctrl_val); | ||
1096 | } | ||
1097 | |||
1098 | static void _InitPABias(struct ieee80211_hw *hw) | ||
1099 | { | ||
1100 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1101 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1102 | u8 pa_setting; | ||
1103 | |||
1104 | /* FIXED PA current issue */ | ||
1105 | pa_setting = efuse_read_1byte(hw, 0x1FA); | ||
1106 | if (!(pa_setting & BIT(0))) { | ||
1107 | rtl_set_rfreg(hw, RF90_PATH_A, 0x15, 0x0FFFFF, 0x0F406); | ||
1108 | rtl_set_rfreg(hw, RF90_PATH_A, 0x15, 0x0FFFFF, 0x4F406); | ||
1109 | rtl_set_rfreg(hw, RF90_PATH_A, 0x15, 0x0FFFFF, 0x8F406); | ||
1110 | rtl_set_rfreg(hw, RF90_PATH_A, 0x15, 0x0FFFFF, 0xCF406); | ||
1111 | } | ||
1112 | if (!(pa_setting & BIT(1)) && IS_NORMAL_CHIP(rtlhal->version) && | ||
1113 | IS_92C_SERIAL(rtlhal->version)) { | ||
1114 | rtl_set_rfreg(hw, RF90_PATH_B, 0x15, 0x0FFFFF, 0x0F406); | ||
1115 | rtl_set_rfreg(hw, RF90_PATH_B, 0x15, 0x0FFFFF, 0x4F406); | ||
1116 | rtl_set_rfreg(hw, RF90_PATH_B, 0x15, 0x0FFFFF, 0x8F406); | ||
1117 | rtl_set_rfreg(hw, RF90_PATH_B, 0x15, 0x0FFFFF, 0xCF406); | ||
1118 | } | ||
1119 | if (!(pa_setting & BIT(4))) { | ||
1120 | pa_setting = rtl_read_byte(rtlpriv, 0x16); | ||
1121 | pa_setting &= 0x0F; | ||
1122 | rtl_write_byte(rtlpriv, 0x16, pa_setting | 0x90); | ||
1123 | } | ||
1124 | } | ||
1125 | |||
1126 | static void _InitAntenna_Selection(struct ieee80211_hw *hw) | ||
1127 | { | ||
1128 | #ifdef CONFIG_ANTENNA_DIVERSITY | ||
1129 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1130 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1131 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1132 | |||
1133 | if (pHalData->AntDivCfg == 0) | ||
1134 | return; | ||
1135 | |||
1136 | if (rtlphy->rf_type == RF_1T1R) { | ||
1137 | rtl_write_dword(rtlpriv, REG_LEDCFG0, | ||
1138 | rtl_read_dword(rtlpriv, | ||
1139 | REG_LEDCFG0)|BIT(23)); | ||
1140 | rtl_set_bbreg(hw, rFPGA0_XAB_RFPARAMETER, BIT(13), 0x01); | ||
1141 | if (rtl_get_bbreg(hw, RFPGA0_XA_RFINTERFACEOE, 0x300) == | ||
1142 | Antenna_A) | ||
1143 | pHalData->CurAntenna = Antenna_A; | ||
1144 | else | ||
1145 | pHalData->CurAntenna = Antenna_B; | ||
1146 | } | ||
1147 | #endif | ||
1148 | } | ||
1149 | |||
1150 | static void _dump_registers(struct ieee80211_hw *hw) | ||
1151 | { | ||
1152 | } | ||
1153 | |||
1154 | static void _update_mac_setting(struct ieee80211_hw *hw) | ||
1155 | { | ||
1156 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1157 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1158 | |||
1159 | mac->rx_conf = rtl_read_dword(rtlpriv, REG_RCR); | ||
1160 | mac->rx_mgt_filter = rtl_read_word(rtlpriv, REG_RXFLTMAP0); | ||
1161 | mac->rx_ctrl_filter = rtl_read_word(rtlpriv, REG_RXFLTMAP1); | ||
1162 | mac->rx_data_filter = rtl_read_word(rtlpriv, REG_RXFLTMAP2); | ||
1163 | } | ||
1164 | |||
1165 | int rtl92cu_hw_init(struct ieee80211_hw *hw) | ||
1166 | { | ||
1167 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1168 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1169 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1170 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1171 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1172 | int err = 0; | ||
1173 | static bool iqk_initialized; | ||
1174 | |||
1175 | rtlhal->hw_type = HARDWARE_TYPE_RTL8192CU; | ||
1176 | err = _rtl92cu_init_mac(hw); | ||
1177 | if (err) { | ||
1178 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("init mac failed!\n")); | ||
1179 | return err; | ||
1180 | } | ||
1181 | err = rtl92c_download_fw(hw); | ||
1182 | if (err) { | ||
1183 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1184 | ("Failed to download FW. Init HW without FW now..\n")); | ||
1185 | err = 1; | ||
1186 | rtlhal->fw_ready = false; | ||
1187 | return err; | ||
1188 | } else { | ||
1189 | rtlhal->fw_ready = true; | ||
1190 | } | ||
1191 | rtlhal->last_hmeboxnum = 0; /* h2c */ | ||
1192 | _rtl92cu_phy_param_tab_init(hw); | ||
1193 | rtl92cu_phy_mac_config(hw); | ||
1194 | rtl92cu_phy_bb_config(hw); | ||
1195 | rtlphy->rf_mode = RF_OP_BY_SW_3WIRE; | ||
1196 | rtl92c_phy_rf_config(hw); | ||
1197 | if (IS_VENDOR_UMC_A_CUT(rtlhal->version) && | ||
1198 | !IS_92C_SERIAL(rtlhal->version)) { | ||
1199 | rtl_set_rfreg(hw, RF90_PATH_A, RF_RX_G1, MASKDWORD, 0x30255); | ||
1200 | rtl_set_rfreg(hw, RF90_PATH_A, RF_RX_G2, MASKDWORD, 0x50a00); | ||
1201 | } | ||
1202 | rtlphy->rfreg_chnlval[0] = rtl_get_rfreg(hw, (enum radio_path)0, | ||
1203 | RF_CHNLBW, RFREG_OFFSET_MASK); | ||
1204 | rtlphy->rfreg_chnlval[1] = rtl_get_rfreg(hw, (enum radio_path)1, | ||
1205 | RF_CHNLBW, RFREG_OFFSET_MASK); | ||
1206 | rtl92cu_bb_block_on(hw); | ||
1207 | rtl_cam_reset_all_entry(hw); | ||
1208 | rtl92cu_enable_hw_security_config(hw); | ||
1209 | ppsc->rfpwr_state = ERFON; | ||
1210 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_ETHER_ADDR, mac->mac_addr); | ||
1211 | if (ppsc->rfpwr_state == ERFON) { | ||
1212 | rtl92c_phy_set_rfpath_switch(hw, 1); | ||
1213 | if (iqk_initialized) { | ||
1214 | rtl92c_phy_iq_calibrate(hw, false); | ||
1215 | } else { | ||
1216 | rtl92c_phy_iq_calibrate(hw, false); | ||
1217 | iqk_initialized = true; | ||
1218 | } | ||
1219 | rtl92c_dm_check_txpower_tracking(hw); | ||
1220 | rtl92c_phy_lc_calibrate(hw); | ||
1221 | } | ||
1222 | _rtl92cu_hw_configure(hw); | ||
1223 | _InitPABias(hw); | ||
1224 | _InitAntenna_Selection(hw); | ||
1225 | _update_mac_setting(hw); | ||
1226 | rtl92c_dm_init(hw); | ||
1227 | _dump_registers(hw); | ||
1228 | return err; | ||
1229 | } | ||
1230 | |||
1231 | static void _DisableRFAFEAndResetBB(struct ieee80211_hw *hw) | ||
1232 | { | ||
1233 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1234 | /************************************** | ||
1235 | a. TXPAUSE 0x522[7:0] = 0xFF Pause MAC TX queue | ||
1236 | b. RF path 0 offset 0x00 = 0x00 disable RF | ||
1237 | c. APSD_CTRL 0x600[7:0] = 0x40 | ||
1238 | d. SYS_FUNC_EN 0x02[7:0] = 0x16 reset BB state machine | ||
1239 | e. SYS_FUNC_EN 0x02[7:0] = 0x14 reset BB state machine | ||
1240 | ***************************************/ | ||
1241 | u8 eRFPath = 0, value8 = 0; | ||
1242 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
1243 | rtl_set_rfreg(hw, (enum radio_path)eRFPath, 0x0, MASKBYTE0, 0x0); | ||
1244 | |||
1245 | value8 |= APSDOFF; | ||
1246 | rtl_write_byte(rtlpriv, REG_APSD_CTRL, value8); /*0x40*/ | ||
1247 | value8 = 0; | ||
1248 | value8 |= (FEN_USBD | FEN_USBA | FEN_BB_GLB_RSTn); | ||
1249 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, value8);/*0x16*/ | ||
1250 | value8 &= (~FEN_BB_GLB_RSTn); | ||
1251 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, value8); /*0x14*/ | ||
1252 | } | ||
1253 | |||
1254 | static void _ResetDigitalProcedure1(struct ieee80211_hw *hw, bool bWithoutHWSM) | ||
1255 | { | ||
1256 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1257 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1258 | |||
1259 | if (rtlhal->fw_version <= 0x20) { | ||
1260 | /***************************** | ||
1261 | f. MCUFWDL 0x80[7:0]=0 reset MCU ready status | ||
1262 | g. SYS_FUNC_EN 0x02[10]= 0 reset MCU reg, (8051 reset) | ||
1263 | h. SYS_FUNC_EN 0x02[15-12]= 5 reset MAC reg, DCORE | ||
1264 | i. SYS_FUNC_EN 0x02[10]= 1 enable MCU reg, (8051 enable) | ||
1265 | ******************************/ | ||
1266 | u16 valu16 = 0; | ||
1267 | |||
1268 | rtl_write_byte(rtlpriv, REG_MCUFWDL, 0); | ||
1269 | valu16 = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
1270 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (valu16 & | ||
1271 | (~FEN_CPUEN))); /* reset MCU ,8051 */ | ||
1272 | valu16 = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN)&0x0FFF; | ||
1273 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (valu16 | | ||
1274 | (FEN_HWPDN|FEN_ELDR))); /* reset MAC */ | ||
1275 | valu16 = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
1276 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (valu16 | | ||
1277 | FEN_CPUEN)); /* enable MCU ,8051 */ | ||
1278 | } else { | ||
1279 | u8 retry_cnts = 0; | ||
1280 | |||
1281 | /* IF fw in RAM code, do reset */ | ||
1282 | if (rtl_read_byte(rtlpriv, REG_MCUFWDL) & BIT(1)) { | ||
1283 | /* reset MCU ready status */ | ||
1284 | rtl_write_byte(rtlpriv, REG_MCUFWDL, 0); | ||
1285 | if (rtlhal->fw_ready) { | ||
1286 | /* 8051 reset by self */ | ||
1287 | rtl_write_byte(rtlpriv, REG_HMETFR+3, 0x20); | ||
1288 | while ((retry_cnts++ < 100) && | ||
1289 | (FEN_CPUEN & rtl_read_word(rtlpriv, | ||
1290 | REG_SYS_FUNC_EN))) { | ||
1291 | udelay(50); | ||
1292 | } | ||
1293 | if (retry_cnts >= 100) { | ||
1294 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1295 | ("#####=> 8051 reset failed!.." | ||
1296 | ".......................\n");); | ||
1297 | /* if 8051 reset fail, reset MAC. */ | ||
1298 | rtl_write_byte(rtlpriv, | ||
1299 | REG_SYS_FUNC_EN + 1, | ||
1300 | 0x50); | ||
1301 | udelay(100); | ||
1302 | } | ||
1303 | } | ||
1304 | } | ||
1305 | /* Reset MAC and Enable 8051 */ | ||
1306 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, 0x54); | ||
1307 | rtl_write_byte(rtlpriv, REG_MCUFWDL, 0); | ||
1308 | } | ||
1309 | if (bWithoutHWSM) { | ||
1310 | /***************************** | ||
1311 | Without HW auto state machine | ||
1312 | g.SYS_CLKR 0x08[15:0] = 0x30A3 disable MAC clock | ||
1313 | h.AFE_PLL_CTRL 0x28[7:0] = 0x80 disable AFE PLL | ||
1314 | i.AFE_XTAL_CTRL 0x24[15:0] = 0x880F gated AFE DIG_CLOCK | ||
1315 | j.SYS_ISu_CTRL 0x00[7:0] = 0xF9 isolated digital to PON | ||
1316 | ******************************/ | ||
1317 | rtl_write_word(rtlpriv, REG_SYS_CLKR, 0x70A3); | ||
1318 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL, 0x80); | ||
1319 | rtl_write_word(rtlpriv, REG_AFE_XTAL_CTRL, 0x880F); | ||
1320 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL, 0xF9); | ||
1321 | } | ||
1322 | } | ||
1323 | |||
1324 | static void _ResetDigitalProcedure2(struct ieee80211_hw *hw) | ||
1325 | { | ||
1326 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1327 | /***************************** | ||
1328 | k. SYS_FUNC_EN 0x03[7:0] = 0x44 disable ELDR runction | ||
1329 | l. SYS_CLKR 0x08[15:0] = 0x3083 disable ELDR clock | ||
1330 | m. SYS_ISO_CTRL 0x01[7:0] = 0x83 isolated ELDR to PON | ||
1331 | ******************************/ | ||
1332 | rtl_write_word(rtlpriv, REG_SYS_CLKR, 0x70A3); | ||
1333 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL+1, 0x82); | ||
1334 | } | ||
1335 | |||
1336 | static void _DisableGPIO(struct ieee80211_hw *hw) | ||
1337 | { | ||
1338 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1339 | /*************************************** | ||
1340 | j. GPIO_PIN_CTRL 0x44[31:0]=0x000 | ||
1341 | k. Value = GPIO_PIN_CTRL[7:0] | ||
1342 | l. GPIO_PIN_CTRL 0x44[31:0] = 0x00FF0000 | (value <<8); write ext PIN level | ||
1343 | m. GPIO_MUXCFG 0x42 [15:0] = 0x0780 | ||
1344 | n. LEDCFG 0x4C[15:0] = 0x8080 | ||
1345 | ***************************************/ | ||
1346 | u8 value8; | ||
1347 | u16 value16; | ||
1348 | u32 value32; | ||
1349 | |||
1350 | /* 1. Disable GPIO[7:0] */ | ||
1351 | rtl_write_word(rtlpriv, REG_GPIO_PIN_CTRL+2, 0x0000); | ||
1352 | value32 = rtl_read_dword(rtlpriv, REG_GPIO_PIN_CTRL) & 0xFFFF00FF; | ||
1353 | value8 = (u8) (value32&0x000000FF); | ||
1354 | value32 |= ((value8<<8) | 0x00FF0000); | ||
1355 | rtl_write_dword(rtlpriv, REG_GPIO_PIN_CTRL, value32); | ||
1356 | /* 2. Disable GPIO[10:8] */ | ||
1357 | rtl_write_byte(rtlpriv, REG_GPIO_MUXCFG+3, 0x00); | ||
1358 | value16 = rtl_read_word(rtlpriv, REG_GPIO_MUXCFG+2) & 0xFF0F; | ||
1359 | value8 = (u8) (value16&0x000F); | ||
1360 | value16 |= ((value8<<4) | 0x0780); | ||
1361 | rtl_write_word(rtlpriv, REG_GPIO_PIN_CTRL+2, value16); | ||
1362 | /* 3. Disable LED0 & 1 */ | ||
1363 | rtl_write_word(rtlpriv, REG_LEDCFG0, 0x8080); | ||
1364 | } | ||
1365 | |||
1366 | static void _DisableAnalog(struct ieee80211_hw *hw, bool bWithoutHWSM) | ||
1367 | { | ||
1368 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1369 | u16 value16 = 0; | ||
1370 | u8 value8 = 0; | ||
1371 | |||
1372 | if (bWithoutHWSM) { | ||
1373 | /***************************** | ||
1374 | n. LDOA15_CTRL 0x20[7:0] = 0x04 disable A15 power | ||
1375 | o. LDOV12D_CTRL 0x21[7:0] = 0x54 disable digital core power | ||
1376 | r. When driver call disable, the ASIC will turn off remaining | ||
1377 | clock automatically | ||
1378 | ******************************/ | ||
1379 | rtl_write_byte(rtlpriv, REG_LDOA15_CTRL, 0x04); | ||
1380 | value8 = rtl_read_byte(rtlpriv, REG_LDOV12D_CTRL); | ||
1381 | value8 &= (~LDV12_EN); | ||
1382 | rtl_write_byte(rtlpriv, REG_LDOV12D_CTRL, value8); | ||
1383 | } | ||
1384 | |||
1385 | /***************************** | ||
1386 | h. SPS0_CTRL 0x11[7:0] = 0x23 enter PFM mode | ||
1387 | i. APS_FSMCO 0x04[15:0] = 0x4802 set USB suspend | ||
1388 | ******************************/ | ||
1389 | rtl_write_byte(rtlpriv, REG_SPS0_CTRL, 0x23); | ||
1390 | value16 |= (APDM_HOST | AFSM_HSUS | PFM_ALDN); | ||
1391 | rtl_write_word(rtlpriv, REG_APS_FSMCO, (u16)value16); | ||
1392 | rtl_write_byte(rtlpriv, REG_RSV_CTRL, 0x0E); | ||
1393 | } | ||
1394 | |||
1395 | static void _CardDisableHWSM(struct ieee80211_hw *hw) | ||
1396 | { | ||
1397 | /* ==== RF Off Sequence ==== */ | ||
1398 | _DisableRFAFEAndResetBB(hw); | ||
1399 | /* ==== Reset digital sequence ====== */ | ||
1400 | _ResetDigitalProcedure1(hw, false); | ||
1401 | /* ==== Pull GPIO PIN to balance level and LED control ====== */ | ||
1402 | _DisableGPIO(hw); | ||
1403 | /* ==== Disable analog sequence === */ | ||
1404 | _DisableAnalog(hw, false); | ||
1405 | } | ||
1406 | |||
1407 | static void _CardDisableWithoutHWSM(struct ieee80211_hw *hw) | ||
1408 | { | ||
1409 | /*==== RF Off Sequence ==== */ | ||
1410 | _DisableRFAFEAndResetBB(hw); | ||
1411 | /* ==== Reset digital sequence ====== */ | ||
1412 | _ResetDigitalProcedure1(hw, true); | ||
1413 | /* ==== Pull GPIO PIN to balance level and LED control ====== */ | ||
1414 | _DisableGPIO(hw); | ||
1415 | /* ==== Reset digital sequence ====== */ | ||
1416 | _ResetDigitalProcedure2(hw); | ||
1417 | /* ==== Disable analog sequence === */ | ||
1418 | _DisableAnalog(hw, true); | ||
1419 | } | ||
1420 | |||
1421 | static void _rtl92cu_set_bcn_ctrl_reg(struct ieee80211_hw *hw, | ||
1422 | u8 set_bits, u8 clear_bits) | ||
1423 | { | ||
1424 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1425 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
1426 | |||
1427 | rtlusb->reg_bcn_ctrl_val |= set_bits; | ||
1428 | rtlusb->reg_bcn_ctrl_val &= ~clear_bits; | ||
1429 | rtl_write_byte(rtlpriv, REG_BCN_CTRL, (u8) rtlusb->reg_bcn_ctrl_val); | ||
1430 | } | ||
1431 | |||
1432 | static void _rtl92cu_stop_tx_beacon(struct ieee80211_hw *hw) | ||
1433 | { | ||
1434 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1435 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1436 | u8 tmp1byte = 0; | ||
1437 | if (IS_NORMAL_CHIP(rtlhal->version)) { | ||
1438 | tmp1byte = rtl_read_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2); | ||
1439 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, | ||
1440 | tmp1byte & (~BIT(6))); | ||
1441 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0x64); | ||
1442 | tmp1byte = rtl_read_byte(rtlpriv, REG_TBTT_PROHIBIT + 2); | ||
1443 | tmp1byte &= ~(BIT(0)); | ||
1444 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 2, tmp1byte); | ||
1445 | } else { | ||
1446 | rtl_write_byte(rtlpriv, REG_TXPAUSE, | ||
1447 | rtl_read_byte(rtlpriv, REG_TXPAUSE) | BIT(6)); | ||
1448 | } | ||
1449 | } | ||
1450 | |||
1451 | static void _rtl92cu_resume_tx_beacon(struct ieee80211_hw *hw) | ||
1452 | { | ||
1453 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1454 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1455 | u8 tmp1byte = 0; | ||
1456 | |||
1457 | if (IS_NORMAL_CHIP(rtlhal->version)) { | ||
1458 | tmp1byte = rtl_read_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2); | ||
1459 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, | ||
1460 | tmp1byte | BIT(6)); | ||
1461 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 1, 0xff); | ||
1462 | tmp1byte = rtl_read_byte(rtlpriv, REG_TBTT_PROHIBIT + 2); | ||
1463 | tmp1byte |= BIT(0); | ||
1464 | rtl_write_byte(rtlpriv, REG_TBTT_PROHIBIT + 2, tmp1byte); | ||
1465 | } else { | ||
1466 | rtl_write_byte(rtlpriv, REG_TXPAUSE, | ||
1467 | rtl_read_byte(rtlpriv, REG_TXPAUSE) & (~BIT(6))); | ||
1468 | } | ||
1469 | } | ||
1470 | |||
1471 | static void _rtl92cu_enable_bcn_sub_func(struct ieee80211_hw *hw) | ||
1472 | { | ||
1473 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1474 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1475 | |||
1476 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
1477 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(1)); | ||
1478 | else | ||
1479 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(4)); | ||
1480 | } | ||
1481 | |||
1482 | static void _rtl92cu_disable_bcn_sub_func(struct ieee80211_hw *hw) | ||
1483 | { | ||
1484 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1485 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1486 | |||
1487 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
1488 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(1), 0); | ||
1489 | else | ||
1490 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
1491 | } | ||
1492 | |||
1493 | static int _rtl92cu_set_media_status(struct ieee80211_hw *hw, | ||
1494 | enum nl80211_iftype type) | ||
1495 | { | ||
1496 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1497 | u8 bt_msr = rtl_read_byte(rtlpriv, MSR); | ||
1498 | enum led_ctl_mode ledaction = LED_CTL_NO_LINK; | ||
1499 | |||
1500 | bt_msr &= 0xfc; | ||
1501 | rtl_write_byte(rtlpriv, REG_BCN_MAX_ERR, 0xFF); | ||
1502 | if (type == NL80211_IFTYPE_UNSPECIFIED || type == | ||
1503 | NL80211_IFTYPE_STATION) { | ||
1504 | _rtl92cu_stop_tx_beacon(hw); | ||
1505 | _rtl92cu_enable_bcn_sub_func(hw); | ||
1506 | } else if (type == NL80211_IFTYPE_ADHOC || type == NL80211_IFTYPE_AP) { | ||
1507 | _rtl92cu_resume_tx_beacon(hw); | ||
1508 | _rtl92cu_disable_bcn_sub_func(hw); | ||
1509 | } else { | ||
1510 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, ("Set HW_VAR_MEDIA_" | ||
1511 | "STATUS:No such media status(%x).\n", type)); | ||
1512 | } | ||
1513 | switch (type) { | ||
1514 | case NL80211_IFTYPE_UNSPECIFIED: | ||
1515 | bt_msr |= MSR_NOLINK; | ||
1516 | ledaction = LED_CTL_LINK; | ||
1517 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1518 | ("Set Network type to NO LINK!\n")); | ||
1519 | break; | ||
1520 | case NL80211_IFTYPE_ADHOC: | ||
1521 | bt_msr |= MSR_ADHOC; | ||
1522 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1523 | ("Set Network type to Ad Hoc!\n")); | ||
1524 | break; | ||
1525 | case NL80211_IFTYPE_STATION: | ||
1526 | bt_msr |= MSR_INFRA; | ||
1527 | ledaction = LED_CTL_LINK; | ||
1528 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1529 | ("Set Network type to STA!\n")); | ||
1530 | break; | ||
1531 | case NL80211_IFTYPE_AP: | ||
1532 | bt_msr |= MSR_AP; | ||
1533 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1534 | ("Set Network type to AP!\n")); | ||
1535 | break; | ||
1536 | default: | ||
1537 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1538 | ("Network type %d not support!\n", type)); | ||
1539 | goto error_out; | ||
1540 | } | ||
1541 | rtl_write_byte(rtlpriv, (MSR), bt_msr); | ||
1542 | rtlpriv->cfg->ops->led_control(hw, ledaction); | ||
1543 | if ((bt_msr & 0xfc) == MSR_AP) | ||
1544 | rtl_write_byte(rtlpriv, REG_BCNTCFG + 1, 0x00); | ||
1545 | else | ||
1546 | rtl_write_byte(rtlpriv, REG_BCNTCFG + 1, 0x66); | ||
1547 | return 0; | ||
1548 | error_out: | ||
1549 | return 1; | ||
1550 | } | ||
1551 | |||
1552 | void rtl92cu_card_disable(struct ieee80211_hw *hw) | ||
1553 | { | ||
1554 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1555 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1556 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
1557 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1558 | enum nl80211_iftype opmode; | ||
1559 | |||
1560 | mac->link_state = MAC80211_NOLINK; | ||
1561 | opmode = NL80211_IFTYPE_UNSPECIFIED; | ||
1562 | _rtl92cu_set_media_status(hw, opmode); | ||
1563 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_POWER_OFF); | ||
1564 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
1565 | if (rtlusb->disableHWSM) | ||
1566 | _CardDisableHWSM(hw); | ||
1567 | else | ||
1568 | _CardDisableWithoutHWSM(hw); | ||
1569 | } | ||
1570 | |||
1571 | void rtl92cu_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid) | ||
1572 | { | ||
1573 | /* dummy routine needed for callback from rtl_op_configure_filter() */ | ||
1574 | } | ||
1575 | |||
1576 | /*========================================================================== */ | ||
1577 | |||
1578 | static void _rtl92cu_set_check_bssid(struct ieee80211_hw *hw, | ||
1579 | enum nl80211_iftype type) | ||
1580 | { | ||
1581 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1582 | u32 reg_rcr = rtl_read_dword(rtlpriv, REG_RCR); | ||
1583 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1584 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1585 | u8 filterout_non_associated_bssid = false; | ||
1586 | |||
1587 | switch (type) { | ||
1588 | case NL80211_IFTYPE_ADHOC: | ||
1589 | case NL80211_IFTYPE_STATION: | ||
1590 | filterout_non_associated_bssid = true; | ||
1591 | break; | ||
1592 | case NL80211_IFTYPE_UNSPECIFIED: | ||
1593 | case NL80211_IFTYPE_AP: | ||
1594 | default: | ||
1595 | break; | ||
1596 | } | ||
1597 | if (filterout_non_associated_bssid == true) { | ||
1598 | if (IS_NORMAL_CHIP(rtlhal->version)) { | ||
1599 | switch (rtlphy->current_io_type) { | ||
1600 | case IO_CMD_RESUME_DM_BY_SCAN: | ||
1601 | reg_rcr |= (RCR_CBSSID_DATA | RCR_CBSSID_BCN); | ||
1602 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
1603 | HW_VAR_RCR, (u8 *)(®_rcr)); | ||
1604 | /* enable update TSF */ | ||
1605 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(4)); | ||
1606 | break; | ||
1607 | case IO_CMD_PAUSE_DM_BY_SCAN: | ||
1608 | reg_rcr &= ~(RCR_CBSSID_DATA | RCR_CBSSID_BCN); | ||
1609 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
1610 | HW_VAR_RCR, (u8 *)(®_rcr)); | ||
1611 | /* disable update TSF */ | ||
1612 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
1613 | break; | ||
1614 | } | ||
1615 | } else { | ||
1616 | reg_rcr |= (RCR_CBSSID); | ||
1617 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, | ||
1618 | (u8 *)(®_rcr)); | ||
1619 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, (BIT(4)|BIT(5))); | ||
1620 | } | ||
1621 | } else if (filterout_non_associated_bssid == false) { | ||
1622 | if (IS_NORMAL_CHIP(rtlhal->version)) { | ||
1623 | reg_rcr &= (~(RCR_CBSSID_DATA | RCR_CBSSID_BCN)); | ||
1624 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, | ||
1625 | (u8 *)(®_rcr)); | ||
1626 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
1627 | } else { | ||
1628 | reg_rcr &= (~RCR_CBSSID); | ||
1629 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, | ||
1630 | (u8 *)(®_rcr)); | ||
1631 | _rtl92cu_set_bcn_ctrl_reg(hw, (BIT(4)|BIT(5)), 0); | ||
1632 | } | ||
1633 | } | ||
1634 | } | ||
1635 | |||
1636 | int rtl92cu_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type) | ||
1637 | { | ||
1638 | if (_rtl92cu_set_media_status(hw, type)) | ||
1639 | return -EOPNOTSUPP; | ||
1640 | _rtl92cu_set_check_bssid(hw, type); | ||
1641 | return 0; | ||
1642 | } | ||
1643 | |||
1644 | static void _InitBeaconParameters(struct ieee80211_hw *hw) | ||
1645 | { | ||
1646 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1647 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
1648 | |||
1649 | rtl_write_word(rtlpriv, REG_BCN_CTRL, 0x1010); | ||
1650 | |||
1651 | /* TODO: Remove these magic number */ | ||
1652 | rtl_write_word(rtlpriv, REG_TBTT_PROHIBIT, 0x6404); | ||
1653 | rtl_write_byte(rtlpriv, REG_DRVERLYINT, DRIVER_EARLY_INT_TIME); | ||
1654 | rtl_write_byte(rtlpriv, REG_BCNDMATIM, BCN_DMA_ATIME_INT_TIME); | ||
1655 | /* Change beacon AIFS to the largest number | ||
1656 | * beacause test chip does not contension before sending beacon. */ | ||
1657 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
1658 | rtl_write_word(rtlpriv, REG_BCNTCFG, 0x660F); | ||
1659 | else | ||
1660 | rtl_write_word(rtlpriv, REG_BCNTCFG, 0x66FF); | ||
1661 | } | ||
1662 | |||
1663 | static void _beacon_function_enable(struct ieee80211_hw *hw, bool Enable, | ||
1664 | bool Linked) | ||
1665 | { | ||
1666 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1667 | |||
1668 | _rtl92cu_set_bcn_ctrl_reg(hw, (BIT(4) | BIT(3) | BIT(1)), 0x00); | ||
1669 | rtl_write_byte(rtlpriv, REG_RD_CTRL+1, 0x6F); | ||
1670 | } | ||
1671 | |||
1672 | void rtl92cu_set_beacon_related_registers(struct ieee80211_hw *hw) | ||
1673 | { | ||
1674 | |||
1675 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1676 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1677 | u16 bcn_interval, atim_window; | ||
1678 | u32 value32; | ||
1679 | |||
1680 | bcn_interval = mac->beacon_interval; | ||
1681 | atim_window = 2; /*FIX MERGE */ | ||
1682 | rtl_write_word(rtlpriv, REG_ATIMWND, atim_window); | ||
1683 | rtl_write_word(rtlpriv, REG_BCN_INTERVAL, bcn_interval); | ||
1684 | _InitBeaconParameters(hw); | ||
1685 | rtl_write_byte(rtlpriv, REG_SLOT, 0x09); | ||
1686 | /* | ||
1687 | * Force beacon frame transmission even after receiving beacon frame | ||
1688 | * from other ad hoc STA | ||
1689 | * | ||
1690 | * | ||
1691 | * Reset TSF Timer to zero, added by Roger. 2008.06.24 | ||
1692 | */ | ||
1693 | value32 = rtl_read_dword(rtlpriv, REG_TCR); | ||
1694 | value32 &= ~TSFRST; | ||
1695 | rtl_write_dword(rtlpriv, REG_TCR, value32); | ||
1696 | value32 |= TSFRST; | ||
1697 | rtl_write_dword(rtlpriv, REG_TCR, value32); | ||
1698 | RT_TRACE(rtlpriv, COMP_INIT|COMP_BEACON, DBG_LOUD, | ||
1699 | ("SetBeaconRelatedRegisters8192CUsb(): Set TCR(%x)\n", | ||
1700 | value32)); | ||
1701 | /* TODO: Modify later (Find the right parameters) | ||
1702 | * NOTE: Fix test chip's bug (about contention windows's randomness) */ | ||
1703 | if ((mac->opmode == NL80211_IFTYPE_ADHOC) || | ||
1704 | (mac->opmode == NL80211_IFTYPE_AP)) { | ||
1705 | rtl_write_byte(rtlpriv, REG_RXTSF_OFFSET_CCK, 0x50); | ||
1706 | rtl_write_byte(rtlpriv, REG_RXTSF_OFFSET_OFDM, 0x50); | ||
1707 | } | ||
1708 | _beacon_function_enable(hw, true, true); | ||
1709 | } | ||
1710 | |||
1711 | void rtl92cu_set_beacon_interval(struct ieee80211_hw *hw) | ||
1712 | { | ||
1713 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1714 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1715 | u16 bcn_interval = mac->beacon_interval; | ||
1716 | |||
1717 | RT_TRACE(rtlpriv, COMP_BEACON, DBG_DMESG, | ||
1718 | ("beacon_interval:%d\n", bcn_interval)); | ||
1719 | rtl_write_word(rtlpriv, REG_BCN_INTERVAL, bcn_interval); | ||
1720 | } | ||
1721 | |||
1722 | void rtl92cu_update_interrupt_mask(struct ieee80211_hw *hw, | ||
1723 | u32 add_msr, u32 rm_msr) | ||
1724 | { | ||
1725 | } | ||
1726 | |||
1727 | void rtl92cu_get_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
1728 | { | ||
1729 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1730 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1731 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1732 | |||
1733 | switch (variable) { | ||
1734 | case HW_VAR_RCR: | ||
1735 | *((u32 *)(val)) = mac->rx_conf; | ||
1736 | break; | ||
1737 | case HW_VAR_RF_STATE: | ||
1738 | *((enum rf_pwrstate *)(val)) = ppsc->rfpwr_state; | ||
1739 | break; | ||
1740 | case HW_VAR_FWLPS_RF_ON:{ | ||
1741 | enum rf_pwrstate rfState; | ||
1742 | u32 val_rcr; | ||
1743 | |||
1744 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_RF_STATE, | ||
1745 | (u8 *)(&rfState)); | ||
1746 | if (rfState == ERFOFF) { | ||
1747 | *((bool *) (val)) = true; | ||
1748 | } else { | ||
1749 | val_rcr = rtl_read_dword(rtlpriv, REG_RCR); | ||
1750 | val_rcr &= 0x00070000; | ||
1751 | if (val_rcr) | ||
1752 | *((bool *) (val)) = false; | ||
1753 | else | ||
1754 | *((bool *) (val)) = true; | ||
1755 | } | ||
1756 | break; | ||
1757 | } | ||
1758 | case HW_VAR_FW_PSMODE_STATUS: | ||
1759 | *((bool *) (val)) = ppsc->fw_current_inpsmode; | ||
1760 | break; | ||
1761 | case HW_VAR_CORRECT_TSF:{ | ||
1762 | u64 tsf; | ||
1763 | u32 *ptsf_low = (u32 *)&tsf; | ||
1764 | u32 *ptsf_high = ((u32 *)&tsf) + 1; | ||
1765 | |||
1766 | *ptsf_high = rtl_read_dword(rtlpriv, (REG_TSFTR + 4)); | ||
1767 | *ptsf_low = rtl_read_dword(rtlpriv, REG_TSFTR); | ||
1768 | *((u64 *)(val)) = tsf; | ||
1769 | break; | ||
1770 | } | ||
1771 | case HW_VAR_MGT_FILTER: | ||
1772 | *((u16 *) (val)) = rtl_read_word(rtlpriv, REG_RXFLTMAP0); | ||
1773 | break; | ||
1774 | case HW_VAR_CTRL_FILTER: | ||
1775 | *((u16 *) (val)) = rtl_read_word(rtlpriv, REG_RXFLTMAP1); | ||
1776 | break; | ||
1777 | case HW_VAR_DATA_FILTER: | ||
1778 | *((u16 *) (val)) = rtl_read_word(rtlpriv, REG_RXFLTMAP2); | ||
1779 | break; | ||
1780 | default: | ||
1781 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1782 | ("switch case not process\n")); | ||
1783 | break; | ||
1784 | } | ||
1785 | } | ||
1786 | |||
1787 | void rtl92cu_set_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
1788 | { | ||
1789 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1790 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1791 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1792 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1793 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1794 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
1795 | enum wireless_mode wirelessmode = mac->mode; | ||
1796 | u8 idx = 0; | ||
1797 | |||
1798 | switch (variable) { | ||
1799 | case HW_VAR_ETHER_ADDR:{ | ||
1800 | for (idx = 0; idx < ETH_ALEN; idx++) { | ||
1801 | rtl_write_byte(rtlpriv, (REG_MACID + idx), | ||
1802 | val[idx]); | ||
1803 | } | ||
1804 | break; | ||
1805 | } | ||
1806 | case HW_VAR_BASIC_RATE:{ | ||
1807 | u16 rate_cfg = ((u16 *) val)[0]; | ||
1808 | u8 rate_index = 0; | ||
1809 | |||
1810 | rate_cfg &= 0x15f; | ||
1811 | /* TODO */ | ||
1812 | /* if (mac->current_network.vender == HT_IOT_PEER_CISCO | ||
1813 | * && ((rate_cfg & 0x150) == 0)) { | ||
1814 | * rate_cfg |= 0x010; | ||
1815 | * } */ | ||
1816 | rate_cfg |= 0x01; | ||
1817 | rtl_write_byte(rtlpriv, REG_RRSR, rate_cfg & 0xff); | ||
1818 | rtl_write_byte(rtlpriv, REG_RRSR + 1, | ||
1819 | (rate_cfg >> 8) & 0xff); | ||
1820 | while (rate_cfg > 0x1) { | ||
1821 | rate_cfg >>= 1; | ||
1822 | rate_index++; | ||
1823 | } | ||
1824 | rtl_write_byte(rtlpriv, REG_INIRTS_RATE_SEL, | ||
1825 | rate_index); | ||
1826 | break; | ||
1827 | } | ||
1828 | case HW_VAR_BSSID:{ | ||
1829 | for (idx = 0; idx < ETH_ALEN; idx++) { | ||
1830 | rtl_write_byte(rtlpriv, (REG_BSSID + idx), | ||
1831 | val[idx]); | ||
1832 | } | ||
1833 | break; | ||
1834 | } | ||
1835 | case HW_VAR_SIFS:{ | ||
1836 | rtl_write_byte(rtlpriv, REG_SIFS_CCK + 1, val[0]); | ||
1837 | rtl_write_byte(rtlpriv, REG_SIFS_OFDM + 1, val[1]); | ||
1838 | rtl_write_byte(rtlpriv, REG_SPEC_SIFS + 1, val[0]); | ||
1839 | rtl_write_byte(rtlpriv, REG_MAC_SPEC_SIFS + 1, val[0]); | ||
1840 | rtl_write_byte(rtlpriv, REG_R2T_SIFS+1, val[0]); | ||
1841 | rtl_write_byte(rtlpriv, REG_T2T_SIFS+1, val[0]); | ||
1842 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1843 | ("HW_VAR_SIFS\n")); | ||
1844 | break; | ||
1845 | } | ||
1846 | case HW_VAR_SLOT_TIME:{ | ||
1847 | u8 e_aci; | ||
1848 | u8 QOS_MODE = 1; | ||
1849 | |||
1850 | rtl_write_byte(rtlpriv, REG_SLOT, val[0]); | ||
1851 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1852 | ("HW_VAR_SLOT_TIME %x\n", val[0])); | ||
1853 | if (QOS_MODE) { | ||
1854 | for (e_aci = 0; e_aci < AC_MAX; e_aci++) | ||
1855 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
1856 | HW_VAR_AC_PARAM, | ||
1857 | (u8 *)(&e_aci)); | ||
1858 | } else { | ||
1859 | u8 sifstime = 0; | ||
1860 | u8 u1bAIFS; | ||
1861 | |||
1862 | if (IS_WIRELESS_MODE_A(wirelessmode) || | ||
1863 | IS_WIRELESS_MODE_N_24G(wirelessmode) || | ||
1864 | IS_WIRELESS_MODE_N_5G(wirelessmode)) | ||
1865 | sifstime = 16; | ||
1866 | else | ||
1867 | sifstime = 10; | ||
1868 | u1bAIFS = sifstime + (2 * val[0]); | ||
1869 | rtl_write_byte(rtlpriv, REG_EDCA_VO_PARAM, | ||
1870 | u1bAIFS); | ||
1871 | rtl_write_byte(rtlpriv, REG_EDCA_VI_PARAM, | ||
1872 | u1bAIFS); | ||
1873 | rtl_write_byte(rtlpriv, REG_EDCA_BE_PARAM, | ||
1874 | u1bAIFS); | ||
1875 | rtl_write_byte(rtlpriv, REG_EDCA_BK_PARAM, | ||
1876 | u1bAIFS); | ||
1877 | } | ||
1878 | break; | ||
1879 | } | ||
1880 | case HW_VAR_ACK_PREAMBLE:{ | ||
1881 | u8 reg_tmp; | ||
1882 | u8 short_preamble = (bool) (*(u8 *) val); | ||
1883 | reg_tmp = 0; | ||
1884 | if (short_preamble) | ||
1885 | reg_tmp |= 0x80; | ||
1886 | rtl_write_byte(rtlpriv, REG_RRSR + 2, reg_tmp); | ||
1887 | break; | ||
1888 | } | ||
1889 | case HW_VAR_AMPDU_MIN_SPACE:{ | ||
1890 | u8 min_spacing_to_set; | ||
1891 | u8 sec_min_space; | ||
1892 | |||
1893 | min_spacing_to_set = *((u8 *) val); | ||
1894 | if (min_spacing_to_set <= 7) { | ||
1895 | switch (rtlpriv->sec.pairwise_enc_algorithm) { | ||
1896 | case NO_ENCRYPTION: | ||
1897 | case AESCCMP_ENCRYPTION: | ||
1898 | sec_min_space = 0; | ||
1899 | break; | ||
1900 | case WEP40_ENCRYPTION: | ||
1901 | case WEP104_ENCRYPTION: | ||
1902 | case TKIP_ENCRYPTION: | ||
1903 | sec_min_space = 6; | ||
1904 | break; | ||
1905 | default: | ||
1906 | sec_min_space = 7; | ||
1907 | break; | ||
1908 | } | ||
1909 | if (min_spacing_to_set < sec_min_space) | ||
1910 | min_spacing_to_set = sec_min_space; | ||
1911 | mac->min_space_cfg = ((mac->min_space_cfg & | ||
1912 | 0xf8) | | ||
1913 | min_spacing_to_set); | ||
1914 | *val = min_spacing_to_set; | ||
1915 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1916 | ("Set HW_VAR_AMPDU_MIN_SPACE: %#x\n", | ||
1917 | mac->min_space_cfg)); | ||
1918 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, | ||
1919 | mac->min_space_cfg); | ||
1920 | } | ||
1921 | break; | ||
1922 | } | ||
1923 | case HW_VAR_SHORTGI_DENSITY:{ | ||
1924 | u8 density_to_set; | ||
1925 | |||
1926 | density_to_set = *((u8 *) val); | ||
1927 | density_to_set &= 0x1f; | ||
1928 | mac->min_space_cfg &= 0x07; | ||
1929 | mac->min_space_cfg |= (density_to_set << 3); | ||
1930 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1931 | ("Set HW_VAR_SHORTGI_DENSITY: %#x\n", | ||
1932 | mac->min_space_cfg)); | ||
1933 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, | ||
1934 | mac->min_space_cfg); | ||
1935 | break; | ||
1936 | } | ||
1937 | case HW_VAR_AMPDU_FACTOR:{ | ||
1938 | u8 regtoset_normal[4] = {0x41, 0xa8, 0x72, 0xb9}; | ||
1939 | u8 factor_toset; | ||
1940 | u8 *p_regtoset = NULL; | ||
1941 | u8 index = 0; | ||
1942 | |||
1943 | p_regtoset = regtoset_normal; | ||
1944 | factor_toset = *((u8 *) val); | ||
1945 | if (factor_toset <= 3) { | ||
1946 | factor_toset = (1 << (factor_toset + 2)); | ||
1947 | if (factor_toset > 0xf) | ||
1948 | factor_toset = 0xf; | ||
1949 | for (index = 0; index < 4; index++) { | ||
1950 | if ((p_regtoset[index] & 0xf0) > | ||
1951 | (factor_toset << 4)) | ||
1952 | p_regtoset[index] = | ||
1953 | (p_regtoset[index] & 0x0f) | ||
1954 | | (factor_toset << 4); | ||
1955 | if ((p_regtoset[index] & 0x0f) > | ||
1956 | factor_toset) | ||
1957 | p_regtoset[index] = | ||
1958 | (p_regtoset[index] & 0xf0) | ||
1959 | | (factor_toset); | ||
1960 | rtl_write_byte(rtlpriv, | ||
1961 | (REG_AGGLEN_LMT + index), | ||
1962 | p_regtoset[index]); | ||
1963 | } | ||
1964 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1965 | ("Set HW_VAR_AMPDU_FACTOR: %#x\n", | ||
1966 | factor_toset)); | ||
1967 | } | ||
1968 | break; | ||
1969 | } | ||
1970 | case HW_VAR_AC_PARAM:{ | ||
1971 | u8 e_aci = *((u8 *) val); | ||
1972 | u32 u4b_ac_param; | ||
1973 | u16 cw_min = le16_to_cpu(mac->ac[e_aci].cw_min); | ||
1974 | u16 cw_max = le16_to_cpu(mac->ac[e_aci].cw_max); | ||
1975 | u16 tx_op = le16_to_cpu(mac->ac[e_aci].tx_op); | ||
1976 | |||
1977 | u4b_ac_param = (u32) mac->ac[e_aci].aifs; | ||
1978 | u4b_ac_param |= (u32) ((cw_min & 0xF) << | ||
1979 | AC_PARAM_ECW_MIN_OFFSET); | ||
1980 | u4b_ac_param |= (u32) ((cw_max & 0xF) << | ||
1981 | AC_PARAM_ECW_MAX_OFFSET); | ||
1982 | u4b_ac_param |= (u32) tx_op << AC_PARAM_TXOP_OFFSET; | ||
1983 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
1984 | ("queue:%x, ac_param:%x\n", e_aci, | ||
1985 | u4b_ac_param)); | ||
1986 | switch (e_aci) { | ||
1987 | case AC1_BK: | ||
1988 | rtl_write_dword(rtlpriv, REG_EDCA_BK_PARAM, | ||
1989 | u4b_ac_param); | ||
1990 | break; | ||
1991 | case AC0_BE: | ||
1992 | rtl_write_dword(rtlpriv, REG_EDCA_BE_PARAM, | ||
1993 | u4b_ac_param); | ||
1994 | break; | ||
1995 | case AC2_VI: | ||
1996 | rtl_write_dword(rtlpriv, REG_EDCA_VI_PARAM, | ||
1997 | u4b_ac_param); | ||
1998 | break; | ||
1999 | case AC3_VO: | ||
2000 | rtl_write_dword(rtlpriv, REG_EDCA_VO_PARAM, | ||
2001 | u4b_ac_param); | ||
2002 | break; | ||
2003 | default: | ||
2004 | RT_ASSERT(false, ("SetHwReg8185(): invalid" | ||
2005 | " aci: %d !\n", e_aci)); | ||
2006 | break; | ||
2007 | } | ||
2008 | if (rtlusb->acm_method != eAcmWay2_SW) | ||
2009 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
2010 | HW_VAR_ACM_CTRL, (u8 *)(&e_aci)); | ||
2011 | break; | ||
2012 | } | ||
2013 | case HW_VAR_ACM_CTRL:{ | ||
2014 | u8 e_aci = *((u8 *) val); | ||
2015 | union aci_aifsn *p_aci_aifsn = (union aci_aifsn *) | ||
2016 | (&(mac->ac[0].aifs)); | ||
2017 | u8 acm = p_aci_aifsn->f.acm; | ||
2018 | u8 acm_ctrl = rtl_read_byte(rtlpriv, REG_ACMHWCTRL); | ||
2019 | |||
2020 | acm_ctrl = | ||
2021 | acm_ctrl | ((rtlusb->acm_method == 2) ? 0x0 : 0x1); | ||
2022 | if (acm) { | ||
2023 | switch (e_aci) { | ||
2024 | case AC0_BE: | ||
2025 | acm_ctrl |= AcmHw_BeqEn; | ||
2026 | break; | ||
2027 | case AC2_VI: | ||
2028 | acm_ctrl |= AcmHw_ViqEn; | ||
2029 | break; | ||
2030 | case AC3_VO: | ||
2031 | acm_ctrl |= AcmHw_VoqEn; | ||
2032 | break; | ||
2033 | default: | ||
2034 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
2035 | ("HW_VAR_ACM_CTRL acm set " | ||
2036 | "failed: eACI is %d\n", acm)); | ||
2037 | break; | ||
2038 | } | ||
2039 | } else { | ||
2040 | switch (e_aci) { | ||
2041 | case AC0_BE: | ||
2042 | acm_ctrl &= (~AcmHw_BeqEn); | ||
2043 | break; | ||
2044 | case AC2_VI: | ||
2045 | acm_ctrl &= (~AcmHw_ViqEn); | ||
2046 | break; | ||
2047 | case AC3_VO: | ||
2048 | acm_ctrl &= (~AcmHw_BeqEn); | ||
2049 | break; | ||
2050 | default: | ||
2051 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
2052 | ("switch case not process\n")); | ||
2053 | break; | ||
2054 | } | ||
2055 | } | ||
2056 | RT_TRACE(rtlpriv, COMP_QOS, DBG_TRACE, | ||
2057 | ("SetHwReg8190pci(): [HW_VAR_ACM_CTRL] " | ||
2058 | "Write 0x%X\n", acm_ctrl)); | ||
2059 | rtl_write_byte(rtlpriv, REG_ACMHWCTRL, acm_ctrl); | ||
2060 | break; | ||
2061 | } | ||
2062 | case HW_VAR_RCR:{ | ||
2063 | rtl_write_dword(rtlpriv, REG_RCR, ((u32 *) (val))[0]); | ||
2064 | mac->rx_conf = ((u32 *) (val))[0]; | ||
2065 | RT_TRACE(rtlpriv, COMP_RECV, DBG_DMESG, | ||
2066 | ("### Set RCR(0x%08x) ###\n", mac->rx_conf)); | ||
2067 | break; | ||
2068 | } | ||
2069 | case HW_VAR_RETRY_LIMIT:{ | ||
2070 | u8 retry_limit = ((u8 *) (val))[0]; | ||
2071 | |||
2072 | rtl_write_word(rtlpriv, REG_RL, | ||
2073 | retry_limit << RETRY_LIMIT_SHORT_SHIFT | | ||
2074 | retry_limit << RETRY_LIMIT_LONG_SHIFT); | ||
2075 | RT_TRACE(rtlpriv, COMP_MLME, DBG_DMESG, ("Set HW_VAR_R" | ||
2076 | "ETRY_LIMIT(0x%08x)\n", retry_limit)); | ||
2077 | break; | ||
2078 | } | ||
2079 | case HW_VAR_DUAL_TSF_RST: | ||
2080 | rtl_write_byte(rtlpriv, REG_DUAL_TSF_RST, (BIT(0) | BIT(1))); | ||
2081 | break; | ||
2082 | case HW_VAR_EFUSE_BYTES: | ||
2083 | rtlefuse->efuse_usedbytes = *((u16 *) val); | ||
2084 | break; | ||
2085 | case HW_VAR_EFUSE_USAGE: | ||
2086 | rtlefuse->efuse_usedpercentage = *((u8 *) val); | ||
2087 | break; | ||
2088 | case HW_VAR_IO_CMD: | ||
2089 | rtl92c_phy_set_io_cmd(hw, (*(enum io_type *)val)); | ||
2090 | break; | ||
2091 | case HW_VAR_WPA_CONFIG: | ||
2092 | rtl_write_byte(rtlpriv, REG_SECCFG, *((u8 *) val)); | ||
2093 | break; | ||
2094 | case HW_VAR_SET_RPWM:{ | ||
2095 | u8 rpwm_val = rtl_read_byte(rtlpriv, REG_USB_HRPWM); | ||
2096 | |||
2097 | if (rpwm_val & BIT(7)) | ||
2098 | rtl_write_byte(rtlpriv, REG_USB_HRPWM, | ||
2099 | (*(u8 *)val)); | ||
2100 | else | ||
2101 | rtl_write_byte(rtlpriv, REG_USB_HRPWM, | ||
2102 | ((*(u8 *)val) | BIT(7))); | ||
2103 | break; | ||
2104 | } | ||
2105 | case HW_VAR_H2C_FW_PWRMODE:{ | ||
2106 | u8 psmode = (*(u8 *) val); | ||
2107 | |||
2108 | if ((psmode != FW_PS_ACTIVE_MODE) && | ||
2109 | (!IS_92C_SERIAL(rtlhal->version))) | ||
2110 | rtl92c_dm_rf_saving(hw, true); | ||
2111 | rtl92c_set_fw_pwrmode_cmd(hw, (*(u8 *) val)); | ||
2112 | break; | ||
2113 | } | ||
2114 | case HW_VAR_FW_PSMODE_STATUS: | ||
2115 | ppsc->fw_current_inpsmode = *((bool *) val); | ||
2116 | break; | ||
2117 | case HW_VAR_H2C_FW_JOINBSSRPT:{ | ||
2118 | u8 mstatus = (*(u8 *) val); | ||
2119 | u8 tmp_reg422; | ||
2120 | bool recover = false; | ||
2121 | |||
2122 | if (mstatus == RT_MEDIA_CONNECT) { | ||
2123 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
2124 | HW_VAR_AID, NULL); | ||
2125 | rtl_write_byte(rtlpriv, REG_CR + 1, 0x03); | ||
2126 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(3)); | ||
2127 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(4), 0); | ||
2128 | tmp_reg422 = rtl_read_byte(rtlpriv, | ||
2129 | REG_FWHW_TXQ_CTRL + 2); | ||
2130 | if (tmp_reg422 & BIT(6)) | ||
2131 | recover = true; | ||
2132 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL + 2, | ||
2133 | tmp_reg422 & (~BIT(6))); | ||
2134 | rtl92c_set_fw_rsvdpagepkt(hw, 0); | ||
2135 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(3), 0); | ||
2136 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(4)); | ||
2137 | if (recover) | ||
2138 | rtl_write_byte(rtlpriv, | ||
2139 | REG_FWHW_TXQ_CTRL + 2, | ||
2140 | tmp_reg422 | BIT(6)); | ||
2141 | rtl_write_byte(rtlpriv, REG_CR + 1, 0x02); | ||
2142 | } | ||
2143 | rtl92c_set_fw_joinbss_report_cmd(hw, (*(u8 *) val)); | ||
2144 | break; | ||
2145 | } | ||
2146 | case HW_VAR_AID:{ | ||
2147 | u16 u2btmp; | ||
2148 | |||
2149 | u2btmp = rtl_read_word(rtlpriv, REG_BCN_PSR_RPT); | ||
2150 | u2btmp &= 0xC000; | ||
2151 | rtl_write_word(rtlpriv, REG_BCN_PSR_RPT, | ||
2152 | (u2btmp | mac->assoc_id)); | ||
2153 | break; | ||
2154 | } | ||
2155 | case HW_VAR_CORRECT_TSF:{ | ||
2156 | u8 btype_ibss = ((u8 *) (val))[0]; | ||
2157 | |||
2158 | if (btype_ibss == true) | ||
2159 | _rtl92cu_stop_tx_beacon(hw); | ||
2160 | _rtl92cu_set_bcn_ctrl_reg(hw, 0, BIT(3)); | ||
2161 | rtl_write_dword(rtlpriv, REG_TSFTR, (u32)(mac->tsf & | ||
2162 | 0xffffffff)); | ||
2163 | rtl_write_dword(rtlpriv, REG_TSFTR + 4, | ||
2164 | (u32)((mac->tsf >> 32) & 0xffffffff)); | ||
2165 | _rtl92cu_set_bcn_ctrl_reg(hw, BIT(3), 0); | ||
2166 | if (btype_ibss == true) | ||
2167 | _rtl92cu_resume_tx_beacon(hw); | ||
2168 | break; | ||
2169 | } | ||
2170 | case HW_VAR_MGT_FILTER: | ||
2171 | rtl_write_word(rtlpriv, REG_RXFLTMAP0, *(u16 *)val); | ||
2172 | break; | ||
2173 | case HW_VAR_CTRL_FILTER: | ||
2174 | rtl_write_word(rtlpriv, REG_RXFLTMAP1, *(u16 *)val); | ||
2175 | break; | ||
2176 | case HW_VAR_DATA_FILTER: | ||
2177 | rtl_write_word(rtlpriv, REG_RXFLTMAP2, *(u16 *)val); | ||
2178 | break; | ||
2179 | default: | ||
2180 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("switch case " | ||
2181 | "not process\n")); | ||
2182 | break; | ||
2183 | } | ||
2184 | } | ||
2185 | |||
2186 | void rtl92cu_update_hal_rate_table(struct ieee80211_hw *hw, | ||
2187 | struct ieee80211_sta *sta, | ||
2188 | u8 rssi_level) | ||
2189 | { | ||
2190 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2191 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
2192 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2193 | u32 ratr_value = (u32) mac->basic_rates; | ||
2194 | u8 *mcsrate = mac->mcs; | ||
2195 | u8 ratr_index = 0; | ||
2196 | u8 nmode = mac->ht_enable; | ||
2197 | u8 mimo_ps = 1; | ||
2198 | u16 shortgi_rate = 0; | ||
2199 | u32 tmp_ratr_value = 0; | ||
2200 | u8 curtxbw_40mhz = mac->bw_40; | ||
2201 | u8 curshortgi_40mhz = mac->sgi_40; | ||
2202 | u8 curshortgi_20mhz = mac->sgi_20; | ||
2203 | enum wireless_mode wirelessmode = mac->mode; | ||
2204 | |||
2205 | ratr_value |= ((*(u16 *) (mcsrate))) << 12; | ||
2206 | switch (wirelessmode) { | ||
2207 | case WIRELESS_MODE_B: | ||
2208 | if (ratr_value & 0x0000000c) | ||
2209 | ratr_value &= 0x0000000d; | ||
2210 | else | ||
2211 | ratr_value &= 0x0000000f; | ||
2212 | break; | ||
2213 | case WIRELESS_MODE_G: | ||
2214 | ratr_value &= 0x00000FF5; | ||
2215 | break; | ||
2216 | case WIRELESS_MODE_N_24G: | ||
2217 | case WIRELESS_MODE_N_5G: | ||
2218 | nmode = 1; | ||
2219 | if (mimo_ps == 0) { | ||
2220 | ratr_value &= 0x0007F005; | ||
2221 | } else { | ||
2222 | u32 ratr_mask; | ||
2223 | |||
2224 | if (get_rf_type(rtlphy) == RF_1T2R || | ||
2225 | get_rf_type(rtlphy) == RF_1T1R) | ||
2226 | ratr_mask = 0x000ff005; | ||
2227 | else | ||
2228 | ratr_mask = 0x0f0ff005; | ||
2229 | if (curtxbw_40mhz) | ||
2230 | ratr_mask |= 0x00000010; | ||
2231 | ratr_value &= ratr_mask; | ||
2232 | } | ||
2233 | break; | ||
2234 | default: | ||
2235 | if (rtlphy->rf_type == RF_1T2R) | ||
2236 | ratr_value &= 0x000ff0ff; | ||
2237 | else | ||
2238 | ratr_value &= 0x0f0ff0ff; | ||
2239 | break; | ||
2240 | } | ||
2241 | ratr_value &= 0x0FFFFFFF; | ||
2242 | if (nmode && ((curtxbw_40mhz && curshortgi_40mhz) || | ||
2243 | (!curtxbw_40mhz && curshortgi_20mhz))) { | ||
2244 | ratr_value |= 0x10000000; | ||
2245 | tmp_ratr_value = (ratr_value >> 12); | ||
2246 | for (shortgi_rate = 15; shortgi_rate > 0; shortgi_rate--) { | ||
2247 | if ((1 << shortgi_rate) & tmp_ratr_value) | ||
2248 | break; | ||
2249 | } | ||
2250 | shortgi_rate = (shortgi_rate << 12) | (shortgi_rate << 8) | | ||
2251 | (shortgi_rate << 4) | (shortgi_rate); | ||
2252 | } | ||
2253 | rtl_write_dword(rtlpriv, REG_ARFR0 + ratr_index * 4, ratr_value); | ||
2254 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, ("%x\n", rtl_read_dword(rtlpriv, | ||
2255 | REG_ARFR0))); | ||
2256 | } | ||
2257 | |||
2258 | void rtl92cu_update_hal_rate_mask(struct ieee80211_hw *hw, u8 rssi_level) | ||
2259 | { | ||
2260 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2261 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
2262 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2263 | u32 ratr_bitmap = (u32) mac->basic_rates; | ||
2264 | u8 *p_mcsrate = mac->mcs; | ||
2265 | u8 ratr_index = 0; | ||
2266 | u8 curtxbw_40mhz = mac->bw_40; | ||
2267 | u8 curshortgi_40mhz = mac->sgi_40; | ||
2268 | u8 curshortgi_20mhz = mac->sgi_20; | ||
2269 | enum wireless_mode wirelessmode = mac->mode; | ||
2270 | bool shortgi = false; | ||
2271 | u8 rate_mask[5]; | ||
2272 | u8 macid = 0; | ||
2273 | u8 mimops = 1; | ||
2274 | |||
2275 | ratr_bitmap |= (p_mcsrate[1] << 20) | (p_mcsrate[0] << 12); | ||
2276 | switch (wirelessmode) { | ||
2277 | case WIRELESS_MODE_B: | ||
2278 | ratr_index = RATR_INX_WIRELESS_B; | ||
2279 | if (ratr_bitmap & 0x0000000c) | ||
2280 | ratr_bitmap &= 0x0000000d; | ||
2281 | else | ||
2282 | ratr_bitmap &= 0x0000000f; | ||
2283 | break; | ||
2284 | case WIRELESS_MODE_G: | ||
2285 | ratr_index = RATR_INX_WIRELESS_GB; | ||
2286 | if (rssi_level == 1) | ||
2287 | ratr_bitmap &= 0x00000f00; | ||
2288 | else if (rssi_level == 2) | ||
2289 | ratr_bitmap &= 0x00000ff0; | ||
2290 | else | ||
2291 | ratr_bitmap &= 0x00000ff5; | ||
2292 | break; | ||
2293 | case WIRELESS_MODE_A: | ||
2294 | ratr_index = RATR_INX_WIRELESS_A; | ||
2295 | ratr_bitmap &= 0x00000ff0; | ||
2296 | break; | ||
2297 | case WIRELESS_MODE_N_24G: | ||
2298 | case WIRELESS_MODE_N_5G: | ||
2299 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
2300 | if (mimops == 0) { | ||
2301 | if (rssi_level == 1) | ||
2302 | ratr_bitmap &= 0x00070000; | ||
2303 | else if (rssi_level == 2) | ||
2304 | ratr_bitmap &= 0x0007f000; | ||
2305 | else | ||
2306 | ratr_bitmap &= 0x0007f005; | ||
2307 | } else { | ||
2308 | if (rtlphy->rf_type == RF_1T2R || | ||
2309 | rtlphy->rf_type == RF_1T1R) { | ||
2310 | if (curtxbw_40mhz) { | ||
2311 | if (rssi_level == 1) | ||
2312 | ratr_bitmap &= 0x000f0000; | ||
2313 | else if (rssi_level == 2) | ||
2314 | ratr_bitmap &= 0x000ff000; | ||
2315 | else | ||
2316 | ratr_bitmap &= 0x000ff015; | ||
2317 | } else { | ||
2318 | if (rssi_level == 1) | ||
2319 | ratr_bitmap &= 0x000f0000; | ||
2320 | else if (rssi_level == 2) | ||
2321 | ratr_bitmap &= 0x000ff000; | ||
2322 | else | ||
2323 | ratr_bitmap &= 0x000ff005; | ||
2324 | } | ||
2325 | } else { | ||
2326 | if (curtxbw_40mhz) { | ||
2327 | if (rssi_level == 1) | ||
2328 | ratr_bitmap &= 0x0f0f0000; | ||
2329 | else if (rssi_level == 2) | ||
2330 | ratr_bitmap &= 0x0f0ff000; | ||
2331 | else | ||
2332 | ratr_bitmap &= 0x0f0ff015; | ||
2333 | } else { | ||
2334 | if (rssi_level == 1) | ||
2335 | ratr_bitmap &= 0x0f0f0000; | ||
2336 | else if (rssi_level == 2) | ||
2337 | ratr_bitmap &= 0x0f0ff000; | ||
2338 | else | ||
2339 | ratr_bitmap &= 0x0f0ff005; | ||
2340 | } | ||
2341 | } | ||
2342 | } | ||
2343 | if ((curtxbw_40mhz && curshortgi_40mhz) || | ||
2344 | (!curtxbw_40mhz && curshortgi_20mhz)) { | ||
2345 | if (macid == 0) | ||
2346 | shortgi = true; | ||
2347 | else if (macid == 1) | ||
2348 | shortgi = false; | ||
2349 | } | ||
2350 | break; | ||
2351 | default: | ||
2352 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
2353 | if (rtlphy->rf_type == RF_1T2R) | ||
2354 | ratr_bitmap &= 0x000ff0ff; | ||
2355 | else | ||
2356 | ratr_bitmap &= 0x0f0ff0ff; | ||
2357 | break; | ||
2358 | } | ||
2359 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, ("ratr_bitmap :%x\n", | ||
2360 | ratr_bitmap)); | ||
2361 | *(u32 *)&rate_mask = ((ratr_bitmap & 0x0fffffff) | | ||
2362 | ratr_index << 28); | ||
2363 | rate_mask[4] = macid | (shortgi ? 0x20 : 0x00) | 0x80; | ||
2364 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, ("Rate_index:%x, " | ||
2365 | "ratr_val:%x, %x:%x:%x:%x:%x\n", | ||
2366 | ratr_index, ratr_bitmap, | ||
2367 | rate_mask[0], rate_mask[1], | ||
2368 | rate_mask[2], rate_mask[3], | ||
2369 | rate_mask[4])); | ||
2370 | rtl92c_fill_h2c_cmd(hw, H2C_RA_MASK, 5, rate_mask); | ||
2371 | } | ||
2372 | |||
2373 | void rtl92cu_update_channel_access_setting(struct ieee80211_hw *hw) | ||
2374 | { | ||
2375 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2376 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2377 | u16 sifs_timer; | ||
2378 | |||
2379 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SLOT_TIME, | ||
2380 | (u8 *)&mac->slot_time); | ||
2381 | if (!mac->ht_enable) | ||
2382 | sifs_timer = 0x0a0a; | ||
2383 | else | ||
2384 | sifs_timer = 0x0e0e; | ||
2385 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SIFS, (u8 *)&sifs_timer); | ||
2386 | } | ||
2387 | |||
2388 | bool rtl92cu_gpio_radio_on_off_checking(struct ieee80211_hw *hw, u8 * valid) | ||
2389 | { | ||
2390 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2391 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
2392 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
2393 | enum rf_pwrstate e_rfpowerstate_toset, cur_rfstate; | ||
2394 | u8 u1tmp = 0; | ||
2395 | bool actuallyset = false; | ||
2396 | unsigned long flag = 0; | ||
2397 | /* to do - usb autosuspend */ | ||
2398 | u8 usb_autosuspend = 0; | ||
2399 | |||
2400 | if (ppsc->swrf_processing) | ||
2401 | return false; | ||
2402 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2403 | if (ppsc->rfchange_inprogress) { | ||
2404 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2405 | return false; | ||
2406 | } else { | ||
2407 | ppsc->rfchange_inprogress = true; | ||
2408 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2409 | } | ||
2410 | cur_rfstate = ppsc->rfpwr_state; | ||
2411 | if (usb_autosuspend) { | ||
2412 | /* to do................... */ | ||
2413 | } else { | ||
2414 | if (ppsc->pwrdown_mode) { | ||
2415 | u1tmp = rtl_read_byte(rtlpriv, REG_HSISR); | ||
2416 | e_rfpowerstate_toset = (u1tmp & BIT(7)) ? | ||
2417 | ERFOFF : ERFON; | ||
2418 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
2419 | ("pwrdown, 0x5c(BIT7)=%02x\n", u1tmp)); | ||
2420 | } else { | ||
2421 | rtl_write_byte(rtlpriv, REG_MAC_PINMUX_CFG, | ||
2422 | rtl_read_byte(rtlpriv, | ||
2423 | REG_MAC_PINMUX_CFG) & ~(BIT(3))); | ||
2424 | u1tmp = rtl_read_byte(rtlpriv, REG_GPIO_IO_SEL); | ||
2425 | e_rfpowerstate_toset = (u1tmp & BIT(3)) ? | ||
2426 | ERFON : ERFOFF; | ||
2427 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
2428 | ("GPIO_IN=%02x\n", u1tmp)); | ||
2429 | } | ||
2430 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("N-SS RF =%x\n", | ||
2431 | e_rfpowerstate_toset)); | ||
2432 | } | ||
2433 | if ((ppsc->hwradiooff) && (e_rfpowerstate_toset == ERFON)) { | ||
2434 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("GPIOChangeRF - HW " | ||
2435 | "Radio ON, RF ON\n")); | ||
2436 | ppsc->hwradiooff = false; | ||
2437 | actuallyset = true; | ||
2438 | } else if ((!ppsc->hwradiooff) && (e_rfpowerstate_toset == | ||
2439 | ERFOFF)) { | ||
2440 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("GPIOChangeRF - HW" | ||
2441 | " Radio OFF\n")); | ||
2442 | ppsc->hwradiooff = true; | ||
2443 | actuallyset = true; | ||
2444 | } else { | ||
2445 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD , | ||
2446 | ("pHalData->bHwRadioOff and eRfPowerStateToSet do not" | ||
2447 | " match: pHalData->bHwRadioOff %x, eRfPowerStateToSet " | ||
2448 | "%x\n", ppsc->hwradiooff, e_rfpowerstate_toset)); | ||
2449 | } | ||
2450 | if (actuallyset) { | ||
2451 | ppsc->hwradiooff = 1; | ||
2452 | if (e_rfpowerstate_toset == ERFON) { | ||
2453 | if ((ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM) && | ||
2454 | RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_ASPM)) | ||
2455 | RT_CLEAR_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_ASPM); | ||
2456 | else if ((ppsc->reg_rfps_level & RT_RF_OFF_LEVL_PCI_D3) | ||
2457 | && RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_PCI_D3)) | ||
2458 | RT_CLEAR_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_PCI_D3); | ||
2459 | } | ||
2460 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2461 | ppsc->rfchange_inprogress = false; | ||
2462 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2463 | /* For power down module, we need to enable register block | ||
2464 | * contrl reg at 0x1c. Then enable power down control bit | ||
2465 | * of register 0x04 BIT4 and BIT15 as 1. | ||
2466 | */ | ||
2467 | if (ppsc->pwrdown_mode && e_rfpowerstate_toset == ERFOFF) { | ||
2468 | /* Enable register area 0x0-0xc. */ | ||
2469 | rtl_write_byte(rtlpriv, REG_RSV_CTRL, 0x0); | ||
2470 | if (IS_HARDWARE_TYPE_8723U(rtlhal)) { | ||
2471 | /* | ||
2472 | * We should configure HW PDn source for WiFi | ||
2473 | * ONLY, and then our HW will be set in | ||
2474 | * power-down mode if PDn source from all | ||
2475 | * functions are configured. | ||
2476 | */ | ||
2477 | u1tmp = rtl_read_byte(rtlpriv, | ||
2478 | REG_MULTI_FUNC_CTRL); | ||
2479 | rtl_write_byte(rtlpriv, REG_MULTI_FUNC_CTRL, | ||
2480 | (u1tmp|WL_HWPDN_EN)); | ||
2481 | } else { | ||
2482 | rtl_write_word(rtlpriv, REG_APS_FSMCO, 0x8812); | ||
2483 | } | ||
2484 | } | ||
2485 | if (e_rfpowerstate_toset == ERFOFF) { | ||
2486 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM) | ||
2487 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_ASPM); | ||
2488 | else if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_PCI_D3) | ||
2489 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_PCI_D3); | ||
2490 | } | ||
2491 | } else if (e_rfpowerstate_toset == ERFOFF || cur_rfstate == ERFOFF) { | ||
2492 | /* Enter D3 or ASPM after GPIO had been done. */ | ||
2493 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_ASPM) | ||
2494 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_ASPM); | ||
2495 | else if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_PCI_D3) | ||
2496 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_PCI_D3); | ||
2497 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2498 | ppsc->rfchange_inprogress = false; | ||
2499 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2500 | } else { | ||
2501 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2502 | ppsc->rfchange_inprogress = false; | ||
2503 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2504 | } | ||
2505 | *valid = 1; | ||
2506 | return !ppsc->hwradiooff; | ||
2507 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/hw.h b/drivers/net/wireless/rtlwifi/rtl8192cu/hw.h new file mode 100644 index 000000000000..32f85cba106a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/hw.h | |||
@@ -0,0 +1,117 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CU_HW_H__ | ||
31 | #define __RTL92CU_HW_H__ | ||
32 | |||
33 | #define H2C_RA_MASK 6 | ||
34 | |||
35 | #define LLT_POLLING_LLT_THRESHOLD 20 | ||
36 | #define LLT_POLLING_READY_TIMEOUT_COUNT 100 | ||
37 | #define LLT_LAST_ENTRY_OF_TX_PKT_BUFFER 255 | ||
38 | |||
39 | #define RX_PAGE_SIZE_REG_VALUE PBP_128 | ||
40 | /* Note: We will divide number of page equally for each queue | ||
41 | * other than public queue! */ | ||
42 | #define TX_TOTAL_PAGE_NUMBER 0xF8 | ||
43 | #define TX_PAGE_BOUNDARY (TX_TOTAL_PAGE_NUMBER + 1) | ||
44 | |||
45 | |||
46 | #define CHIP_B_PAGE_NUM_PUBQ 0xE7 | ||
47 | |||
48 | /* For Test Chip Setting | ||
49 | * (HPQ + LPQ + PUBQ) shall be TX_TOTAL_PAGE_NUMBER */ | ||
50 | #define CHIP_A_PAGE_NUM_PUBQ 0x7E | ||
51 | |||
52 | |||
53 | /* For Chip A Setting */ | ||
54 | #define WMM_CHIP_A_TX_TOTAL_PAGE_NUMBER 0xF5 | ||
55 | #define WMM_CHIP_A_TX_PAGE_BOUNDARY \ | ||
56 | (WMM_CHIP_A_TX_TOTAL_PAGE_NUMBER + 1) /* F6 */ | ||
57 | |||
58 | #define WMM_CHIP_A_PAGE_NUM_PUBQ 0xA3 | ||
59 | #define WMM_CHIP_A_PAGE_NUM_HPQ 0x29 | ||
60 | #define WMM_CHIP_A_PAGE_NUM_LPQ 0x29 | ||
61 | |||
62 | |||
63 | |||
64 | /* Note: For Chip B Setting ,modify later */ | ||
65 | #define WMM_CHIP_B_TX_TOTAL_PAGE_NUMBER 0xF5 | ||
66 | #define WMM_CHIP_B_TX_PAGE_BOUNDARY \ | ||
67 | (WMM_CHIP_B_TX_TOTAL_PAGE_NUMBER + 1) /* F6 */ | ||
68 | |||
69 | #define WMM_CHIP_B_PAGE_NUM_PUBQ 0xB0 | ||
70 | #define WMM_CHIP_B_PAGE_NUM_HPQ 0x29 | ||
71 | #define WMM_CHIP_B_PAGE_NUM_LPQ 0x1C | ||
72 | #define WMM_CHIP_B_PAGE_NUM_NPQ 0x1C | ||
73 | |||
74 | #define BOARD_TYPE_NORMAL_MASK 0xE0 | ||
75 | #define BOARD_TYPE_TEST_MASK 0x0F | ||
76 | |||
77 | /* should be renamed and moved to another file */ | ||
78 | enum _BOARD_TYPE_8192CUSB { | ||
79 | BOARD_USB_DONGLE = 0, /* USB dongle */ | ||
80 | BOARD_USB_High_PA = 1, /* USB dongle - high power PA */ | ||
81 | BOARD_MINICARD = 2, /* Minicard */ | ||
82 | BOARD_USB_SOLO = 3, /* USB solo-Slim module */ | ||
83 | BOARD_USB_COMBO = 4, /* USB Combo-Slim module */ | ||
84 | }; | ||
85 | |||
86 | #define IS_HIGHT_PA(boardtype) \ | ||
87 | ((boardtype == BOARD_USB_High_PA) ? true : false) | ||
88 | |||
89 | #define RTL92C_DRIVER_INFO_SIZE 4 | ||
90 | void rtl92cu_read_eeprom_info(struct ieee80211_hw *hw); | ||
91 | void rtl92cu_enable_hw_security_config(struct ieee80211_hw *hw); | ||
92 | int rtl92cu_hw_init(struct ieee80211_hw *hw); | ||
93 | void rtl92cu_card_disable(struct ieee80211_hw *hw); | ||
94 | int rtl92cu_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type); | ||
95 | void rtl92cu_set_beacon_related_registers(struct ieee80211_hw *hw); | ||
96 | void rtl92cu_set_beacon_interval(struct ieee80211_hw *hw); | ||
97 | void rtl92cu_update_interrupt_mask(struct ieee80211_hw *hw, | ||
98 | u32 add_msr, u32 rm_msr); | ||
99 | void rtl92cu_get_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
100 | void rtl92cu_set_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
101 | void rtl92cu_update_hal_rate_table(struct ieee80211_hw *hw, | ||
102 | struct ieee80211_sta *sta, | ||
103 | u8 rssi_level); | ||
104 | void rtl92cu_update_hal_rate_mask(struct ieee80211_hw *hw, u8 rssi_level); | ||
105 | |||
106 | void rtl92cu_update_channel_access_setting(struct ieee80211_hw *hw); | ||
107 | bool rtl92cu_gpio_radio_on_off_checking(struct ieee80211_hw *hw, u8 * valid); | ||
108 | void rtl92cu_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid); | ||
109 | int rtl92c_download_fw(struct ieee80211_hw *hw); | ||
110 | void rtl92c_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 mode); | ||
111 | void rtl92c_set_fw_rsvdpagepkt(struct ieee80211_hw *hw, bool dl_finished); | ||
112 | void rtl92c_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, u8 mstatus); | ||
113 | void rtl92c_fill_h2c_cmd(struct ieee80211_hw *hw, | ||
114 | u8 element_id, u32 cmd_len, u8 *p_cmdbuffer); | ||
115 | bool rtl92cu_phy_mac_config(struct ieee80211_hw *hw); | ||
116 | |||
117 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/led.c b/drivers/net/wireless/rtlwifi/rtl8192cu/led.c new file mode 100644 index 000000000000..332c74348a69 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/led.c | |||
@@ -0,0 +1,142 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | *****************************************************************************/ | ||
27 | |||
28 | #include "../wifi.h" | ||
29 | #include "../usb.h" | ||
30 | #include "reg.h" | ||
31 | #include "led.h" | ||
32 | |||
33 | static void _rtl92cu_init_led(struct ieee80211_hw *hw, | ||
34 | struct rtl_led *pled, enum rtl_led_pin ledpin) | ||
35 | { | ||
36 | pled->hw = hw; | ||
37 | pled->ledpin = ledpin; | ||
38 | pled->ledon = false; | ||
39 | } | ||
40 | |||
41 | static void _rtl92cu_deInit_led(struct rtl_led *pled) | ||
42 | { | ||
43 | } | ||
44 | |||
45 | void rtl92cu_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
46 | { | ||
47 | u8 ledcfg; | ||
48 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
49 | |||
50 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
51 | ("LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pled->ledpin)); | ||
52 | ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2); | ||
53 | switch (pled->ledpin) { | ||
54 | case LED_PIN_GPIO0: | ||
55 | break; | ||
56 | case LED_PIN_LED0: | ||
57 | rtl_write_byte(rtlpriv, | ||
58 | REG_LEDCFG2, (ledcfg & 0xf0) | BIT(5) | BIT(6)); | ||
59 | break; | ||
60 | case LED_PIN_LED1: | ||
61 | rtl_write_byte(rtlpriv, REG_LEDCFG2, (ledcfg & 0x0f) | BIT(5)); | ||
62 | break; | ||
63 | default: | ||
64 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
65 | ("switch case not process\n")); | ||
66 | break; | ||
67 | } | ||
68 | pled->ledon = true; | ||
69 | } | ||
70 | |||
71 | void rtl92cu_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
72 | { | ||
73 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
74 | struct rtl_usb_priv *usbpriv = rtl_usbpriv(hw); | ||
75 | u8 ledcfg; | ||
76 | |||
77 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
78 | ("LedAddr:%X ledpin=%d\n", REG_LEDCFG2, pled->ledpin)); | ||
79 | ledcfg = rtl_read_byte(rtlpriv, REG_LEDCFG2); | ||
80 | switch (pled->ledpin) { | ||
81 | case LED_PIN_GPIO0: | ||
82 | break; | ||
83 | case LED_PIN_LED0: | ||
84 | ledcfg &= 0xf0; | ||
85 | if (usbpriv->ledctl.led_opendrain == true) | ||
86 | rtl_write_byte(rtlpriv, REG_LEDCFG2, | ||
87 | (ledcfg | BIT(1) | BIT(5) | BIT(6))); | ||
88 | else | ||
89 | rtl_write_byte(rtlpriv, REG_LEDCFG2, | ||
90 | (ledcfg | BIT(3) | BIT(5) | BIT(6))); | ||
91 | break; | ||
92 | case LED_PIN_LED1: | ||
93 | ledcfg &= 0x0f; | ||
94 | rtl_write_byte(rtlpriv, REG_LEDCFG2, (ledcfg | BIT(3))); | ||
95 | break; | ||
96 | default: | ||
97 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
98 | ("switch case not process\n")); | ||
99 | break; | ||
100 | } | ||
101 | pled->ledon = false; | ||
102 | } | ||
103 | |||
104 | void rtl92cu_init_sw_leds(struct ieee80211_hw *hw) | ||
105 | { | ||
106 | struct rtl_usb_priv *usbpriv = rtl_usbpriv(hw); | ||
107 | _rtl92cu_init_led(hw, &(usbpriv->ledctl.sw_led0), LED_PIN_LED0); | ||
108 | _rtl92cu_init_led(hw, &(usbpriv->ledctl.sw_led1), LED_PIN_LED1); | ||
109 | } | ||
110 | |||
111 | void rtl92cu_deinit_sw_leds(struct ieee80211_hw *hw) | ||
112 | { | ||
113 | struct rtl_usb_priv *usbpriv = rtl_usbpriv(hw); | ||
114 | _rtl92cu_deInit_led(&(usbpriv->ledctl.sw_led0)); | ||
115 | _rtl92cu_deInit_led(&(usbpriv->ledctl.sw_led1)); | ||
116 | } | ||
117 | |||
118 | static void _rtl92cu_sw_led_control(struct ieee80211_hw *hw, | ||
119 | enum led_ctl_mode ledaction) | ||
120 | { | ||
121 | } | ||
122 | |||
123 | void rtl92cu_led_control(struct ieee80211_hw *hw, | ||
124 | enum led_ctl_mode ledaction) | ||
125 | { | ||
126 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
127 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
128 | |||
129 | if ((ppsc->rfoff_reason > RF_CHANGE_BY_PS) && | ||
130 | (ledaction == LED_CTL_TX || | ||
131 | ledaction == LED_CTL_RX || | ||
132 | ledaction == LED_CTL_SITE_SURVEY || | ||
133 | ledaction == LED_CTL_LINK || | ||
134 | ledaction == LED_CTL_NO_LINK || | ||
135 | ledaction == LED_CTL_START_TO_LINK || | ||
136 | ledaction == LED_CTL_POWER_ON)) { | ||
137 | return; | ||
138 | } | ||
139 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, ("ledaction %d,\n", | ||
140 | ledaction)); | ||
141 | _rtl92cu_sw_led_control(hw, ledaction); | ||
142 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/led.h b/drivers/net/wireless/rtlwifi/rtl8192cu/led.h new file mode 100644 index 000000000000..decaee4d1eb1 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/led.h | |||
@@ -0,0 +1,37 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | *****************************************************************************/ | ||
27 | |||
28 | #ifndef __RTL92CU_LED_H__ | ||
29 | #define __RTL92CU_LED_H__ | ||
30 | |||
31 | void rtl92cu_init_sw_leds(struct ieee80211_hw *hw); | ||
32 | void rtl92cu_deinit_sw_leds(struct ieee80211_hw *hw); | ||
33 | void rtl92cu_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
34 | void rtl92cu_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
35 | void rtl92cu_led_control(struct ieee80211_hw *hw, enum led_ctl_mode ledaction); | ||
36 | |||
37 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/mac.c b/drivers/net/wireless/rtlwifi/rtl8192cu/mac.c new file mode 100644 index 000000000000..f8514cba17b6 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/mac.c | |||
@@ -0,0 +1,1144 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | ****************************************************************************/ | ||
29 | #include <linux/module.h> | ||
30 | |||
31 | #include "../wifi.h" | ||
32 | #include "../pci.h" | ||
33 | #include "../usb.h" | ||
34 | #include "../ps.h" | ||
35 | #include "../cam.h" | ||
36 | #include "reg.h" | ||
37 | #include "def.h" | ||
38 | #include "phy.h" | ||
39 | #include "rf.h" | ||
40 | #include "dm.h" | ||
41 | #include "mac.h" | ||
42 | #include "trx.h" | ||
43 | |||
44 | /* macro to shorten lines */ | ||
45 | |||
46 | #define LINK_Q ui_link_quality | ||
47 | #define RX_EVM rx_evm_percentage | ||
48 | #define RX_SIGQ rx_mimo_signalquality | ||
49 | |||
50 | |||
51 | void rtl92c_read_chip_version(struct ieee80211_hw *hw) | ||
52 | { | ||
53 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
54 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
55 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
56 | enum version_8192c chip_version = VERSION_UNKNOWN; | ||
57 | u32 value32; | ||
58 | |||
59 | value32 = rtl_read_dword(rtlpriv, REG_SYS_CFG); | ||
60 | if (value32 & TRP_VAUX_EN) { | ||
61 | chip_version = (value32 & TYPE_ID) ? VERSION_TEST_CHIP_92C : | ||
62 | VERSION_TEST_CHIP_88C; | ||
63 | } else { | ||
64 | /* Normal mass production chip. */ | ||
65 | chip_version = NORMAL_CHIP; | ||
66 | chip_version |= ((value32 & TYPE_ID) ? CHIP_92C : 0); | ||
67 | chip_version |= ((value32 & VENDOR_ID) ? CHIP_VENDOR_UMC : 0); | ||
68 | /* RTL8723 with BT function. */ | ||
69 | chip_version |= ((value32 & BT_FUNC) ? CHIP_8723 : 0); | ||
70 | if (IS_VENDOR_UMC(chip_version)) | ||
71 | chip_version |= ((value32 & CHIP_VER_RTL_MASK) ? | ||
72 | CHIP_VENDOR_UMC_B_CUT : 0); | ||
73 | if (IS_92C_SERIAL(chip_version)) { | ||
74 | value32 = rtl_read_dword(rtlpriv, REG_HPON_FSM); | ||
75 | chip_version |= ((CHIP_BONDING_IDENTIFIER(value32) == | ||
76 | CHIP_BONDING_92C_1T2R) ? CHIP_92C_1T2R : 0); | ||
77 | } else if (IS_8723_SERIES(chip_version)) { | ||
78 | value32 = rtl_read_dword(rtlpriv, REG_GPIO_OUTSTS); | ||
79 | chip_version |= ((value32 & RF_RL_ID) ? | ||
80 | CHIP_8723_DRV_REV : 0); | ||
81 | } | ||
82 | } | ||
83 | rtlhal->version = (enum version_8192c)chip_version; | ||
84 | switch (rtlhal->version) { | ||
85 | case VERSION_NORMAL_TSMC_CHIP_92C_1T2R: | ||
86 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
87 | ("Chip Version ID: VERSION_B_CHIP_92C.\n")); | ||
88 | break; | ||
89 | case VERSION_NORMAL_TSMC_CHIP_92C: | ||
90 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
91 | ("Chip Version ID: VERSION_NORMAL_TSMC_CHIP_92C.\n")); | ||
92 | break; | ||
93 | case VERSION_NORMAL_TSMC_CHIP_88C: | ||
94 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
95 | ("Chip Version ID: VERSION_NORMAL_TSMC_CHIP_88C.\n")); | ||
96 | break; | ||
97 | case VERSION_NORMAL_UMC_CHIP_92C_1T2R_A_CUT: | ||
98 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
99 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP_i" | ||
100 | "92C_1T2R_A_CUT.\n")); | ||
101 | break; | ||
102 | case VERSION_NORMAL_UMC_CHIP_92C_A_CUT: | ||
103 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
104 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP_" | ||
105 | "92C_A_CUT.\n")); | ||
106 | break; | ||
107 | case VERSION_NORMAL_UMC_CHIP_88C_A_CUT: | ||
108 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
109 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP" | ||
110 | "_88C_A_CUT.\n")); | ||
111 | break; | ||
112 | case VERSION_NORMAL_UMC_CHIP_92C_1T2R_B_CUT: | ||
113 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
114 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP" | ||
115 | "_92C_1T2R_B_CUT.\n")); | ||
116 | break; | ||
117 | case VERSION_NORMAL_UMC_CHIP_92C_B_CUT: | ||
118 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
119 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP" | ||
120 | "_92C_B_CUT.\n")); | ||
121 | break; | ||
122 | case VERSION_NORMAL_UMC_CHIP_88C_B_CUT: | ||
123 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
124 | ("Chip Version ID: VERSION_NORMAL_UMC_CHIP" | ||
125 | "_88C_B_CUT.\n")); | ||
126 | break; | ||
127 | case VERSION_NORMA_UMC_CHIP_8723_1T1R_A_CUT: | ||
128 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
129 | ("Chip Version ID: VERSION_NORMA_UMC_CHIP" | ||
130 | "_8723_1T1R_A_CUT.\n")); | ||
131 | break; | ||
132 | case VERSION_NORMA_UMC_CHIP_8723_1T1R_B_CUT: | ||
133 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
134 | ("Chip Version ID: VERSION_NORMA_UMC_CHIP" | ||
135 | "_8723_1T1R_B_CUT.\n")); | ||
136 | break; | ||
137 | case VERSION_TEST_CHIP_92C: | ||
138 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
139 | ("Chip Version ID: VERSION_TEST_CHIP_92C.\n")); | ||
140 | break; | ||
141 | case VERSION_TEST_CHIP_88C: | ||
142 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
143 | ("Chip Version ID: VERSION_TEST_CHIP_88C.\n")); | ||
144 | break; | ||
145 | default: | ||
146 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
147 | ("Chip Version ID: ???????????????.\n")); | ||
148 | break; | ||
149 | } | ||
150 | if (IS_92C_SERIAL(rtlhal->version)) | ||
151 | rtlphy->rf_type = | ||
152 | (IS_92C_1T2R(rtlhal->version)) ? RF_1T2R : RF_2T2R; | ||
153 | else | ||
154 | rtlphy->rf_type = RF_1T1R; | ||
155 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
156 | ("Chip RF Type: %s\n", (rtlphy->rf_type == RF_2T2R) ? | ||
157 | "RF_2T2R" : "RF_1T1R")); | ||
158 | if (get_rf_type(rtlphy) == RF_1T1R) | ||
159 | rtlpriv->dm.rfpath_rxenable[0] = true; | ||
160 | else | ||
161 | rtlpriv->dm.rfpath_rxenable[0] = | ||
162 | rtlpriv->dm.rfpath_rxenable[1] = true; | ||
163 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("VersionID = 0x%4x\n", | ||
164 | rtlhal->version)); | ||
165 | } | ||
166 | |||
167 | /** | ||
168 | * writeLLT - LLT table write access | ||
169 | * @io: io callback | ||
170 | * @address: LLT logical address. | ||
171 | * @data: LLT data content | ||
172 | * | ||
173 | * Realtek hardware access function. | ||
174 | * | ||
175 | */ | ||
176 | bool rtl92c_llt_write(struct ieee80211_hw *hw, u32 address, u32 data) | ||
177 | { | ||
178 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
179 | bool status = true; | ||
180 | long count = 0; | ||
181 | u32 value = _LLT_INIT_ADDR(address) | | ||
182 | _LLT_INIT_DATA(data) | _LLT_OP(_LLT_WRITE_ACCESS); | ||
183 | |||
184 | rtl_write_dword(rtlpriv, REG_LLT_INIT, value); | ||
185 | do { | ||
186 | value = rtl_read_dword(rtlpriv, REG_LLT_INIT); | ||
187 | if (_LLT_NO_ACTIVE == _LLT_OP_VALUE(value)) | ||
188 | break; | ||
189 | if (count > POLLING_LLT_THRESHOLD) { | ||
190 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
191 | ("Failed to polling write LLT done at" | ||
192 | " address %d! _LLT_OP_VALUE(%x)\n", | ||
193 | address, _LLT_OP_VALUE(value))); | ||
194 | status = false; | ||
195 | break; | ||
196 | } | ||
197 | } while (++count); | ||
198 | return status; | ||
199 | } | ||
200 | /** | ||
201 | * rtl92c_init_LLT_table - Init LLT table | ||
202 | * @io: io callback | ||
203 | * @boundary: | ||
204 | * | ||
205 | * Realtek hardware access function. | ||
206 | * | ||
207 | */ | ||
208 | bool rtl92c_init_llt_table(struct ieee80211_hw *hw, u32 boundary) | ||
209 | { | ||
210 | bool rst = true; | ||
211 | u32 i; | ||
212 | |||
213 | for (i = 0; i < (boundary - 1); i++) { | ||
214 | rst = rtl92c_llt_write(hw, i , i + 1); | ||
215 | if (true != rst) { | ||
216 | printk(KERN_ERR "===> %s #1 fail\n", __func__); | ||
217 | return rst; | ||
218 | } | ||
219 | } | ||
220 | /* end of list */ | ||
221 | rst = rtl92c_llt_write(hw, (boundary - 1), 0xFF); | ||
222 | if (true != rst) { | ||
223 | printk(KERN_ERR "===> %s #2 fail\n", __func__); | ||
224 | return rst; | ||
225 | } | ||
226 | /* Make the other pages as ring buffer | ||
227 | * This ring buffer is used as beacon buffer if we config this MAC | ||
228 | * as two MAC transfer. | ||
229 | * Otherwise used as local loopback buffer. | ||
230 | */ | ||
231 | for (i = boundary; i < LLT_LAST_ENTRY_OF_TX_PKT_BUFFER; i++) { | ||
232 | rst = rtl92c_llt_write(hw, i, (i + 1)); | ||
233 | if (true != rst) { | ||
234 | printk(KERN_ERR "===> %s #3 fail\n", __func__); | ||
235 | return rst; | ||
236 | } | ||
237 | } | ||
238 | /* Let last entry point to the start entry of ring buffer */ | ||
239 | rst = rtl92c_llt_write(hw, LLT_LAST_ENTRY_OF_TX_PKT_BUFFER, boundary); | ||
240 | if (true != rst) { | ||
241 | printk(KERN_ERR "===> %s #4 fail\n", __func__); | ||
242 | return rst; | ||
243 | } | ||
244 | return rst; | ||
245 | } | ||
246 | void rtl92c_set_key(struct ieee80211_hw *hw, u32 key_index, | ||
247 | u8 *p_macaddr, bool is_group, u8 enc_algo, | ||
248 | bool is_wepkey, bool clear_all) | ||
249 | { | ||
250 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
251 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
252 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
253 | u8 *macaddr = p_macaddr; | ||
254 | u32 entry_id = 0; | ||
255 | bool is_pairwise = false; | ||
256 | static u8 cam_const_addr[4][6] = { | ||
257 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, | ||
258 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, | ||
259 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x02}, | ||
260 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x03} | ||
261 | }; | ||
262 | static u8 cam_const_broad[] = { | ||
263 | 0xff, 0xff, 0xff, 0xff, 0xff, 0xff | ||
264 | }; | ||
265 | |||
266 | if (clear_all) { | ||
267 | u8 idx = 0; | ||
268 | u8 cam_offset = 0; | ||
269 | u8 clear_number = 5; | ||
270 | |||
271 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("clear_all\n")); | ||
272 | for (idx = 0; idx < clear_number; idx++) { | ||
273 | rtl_cam_mark_invalid(hw, cam_offset + idx); | ||
274 | rtl_cam_empty_entry(hw, cam_offset + idx); | ||
275 | if (idx < 5) { | ||
276 | memset(rtlpriv->sec.key_buf[idx], 0, | ||
277 | MAX_KEY_LEN); | ||
278 | rtlpriv->sec.key_len[idx] = 0; | ||
279 | } | ||
280 | } | ||
281 | } else { | ||
282 | switch (enc_algo) { | ||
283 | case WEP40_ENCRYPTION: | ||
284 | enc_algo = CAM_WEP40; | ||
285 | break; | ||
286 | case WEP104_ENCRYPTION: | ||
287 | enc_algo = CAM_WEP104; | ||
288 | break; | ||
289 | case TKIP_ENCRYPTION: | ||
290 | enc_algo = CAM_TKIP; | ||
291 | break; | ||
292 | case AESCCMP_ENCRYPTION: | ||
293 | enc_algo = CAM_AES; | ||
294 | break; | ||
295 | default: | ||
296 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
297 | ("iillegal switch case\n")); | ||
298 | enc_algo = CAM_TKIP; | ||
299 | break; | ||
300 | } | ||
301 | if (is_wepkey || rtlpriv->sec.use_defaultkey) { | ||
302 | macaddr = cam_const_addr[key_index]; | ||
303 | entry_id = key_index; | ||
304 | } else { | ||
305 | if (is_group) { | ||
306 | macaddr = cam_const_broad; | ||
307 | entry_id = key_index; | ||
308 | } else { | ||
309 | key_index = PAIRWISE_KEYIDX; | ||
310 | entry_id = CAM_PAIRWISE_KEY_POSITION; | ||
311 | is_pairwise = true; | ||
312 | } | ||
313 | } | ||
314 | if (rtlpriv->sec.key_len[key_index] == 0) { | ||
315 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
316 | ("delete one entry\n")); | ||
317 | rtl_cam_delete_one_entry(hw, p_macaddr, entry_id); | ||
318 | } else { | ||
319 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
320 | ("The insert KEY length is %d\n", | ||
321 | rtlpriv->sec.key_len[PAIRWISE_KEYIDX])); | ||
322 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
323 | ("The insert KEY is %x %x\n", | ||
324 | rtlpriv->sec.key_buf[0][0], | ||
325 | rtlpriv->sec.key_buf[0][1])); | ||
326 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
327 | ("add one entry\n")); | ||
328 | if (is_pairwise) { | ||
329 | RT_PRINT_DATA(rtlpriv, COMP_SEC, DBG_LOUD, | ||
330 | "Pairwiase Key content :", | ||
331 | rtlpriv->sec.pairwise_key, | ||
332 | rtlpriv->sec. | ||
333 | key_len[PAIRWISE_KEYIDX]); | ||
334 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
335 | ("set Pairwiase key\n")); | ||
336 | |||
337 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
338 | entry_id, enc_algo, | ||
339 | CAM_CONFIG_NO_USEDK, | ||
340 | rtlpriv->sec. | ||
341 | key_buf[key_index]); | ||
342 | } else { | ||
343 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
344 | ("set group key\n")); | ||
345 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
346 | rtl_cam_add_one_entry(hw, | ||
347 | rtlefuse->dev_addr, | ||
348 | PAIRWISE_KEYIDX, | ||
349 | CAM_PAIRWISE_KEY_POSITION, | ||
350 | enc_algo, | ||
351 | CAM_CONFIG_NO_USEDK, | ||
352 | rtlpriv->sec.key_buf | ||
353 | [entry_id]); | ||
354 | } | ||
355 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
356 | entry_id, enc_algo, | ||
357 | CAM_CONFIG_NO_USEDK, | ||
358 | rtlpriv->sec.key_buf[entry_id]); | ||
359 | } | ||
360 | } | ||
361 | } | ||
362 | } | ||
363 | |||
364 | u32 rtl92c_get_txdma_status(struct ieee80211_hw *hw) | ||
365 | { | ||
366 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
367 | |||
368 | return rtl_read_dword(rtlpriv, REG_TXDMA_STATUS); | ||
369 | } | ||
370 | |||
371 | void rtl92c_enable_interrupt(struct ieee80211_hw *hw) | ||
372 | { | ||
373 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
374 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
375 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
376 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
377 | |||
378 | if (IS_HARDWARE_TYPE_8192CE(rtlhal)) { | ||
379 | rtl_write_dword(rtlpriv, REG_HIMR, rtlpci->irq_mask[0] & | ||
380 | 0xFFFFFFFF); | ||
381 | rtl_write_dword(rtlpriv, REG_HIMRE, rtlpci->irq_mask[1] & | ||
382 | 0xFFFFFFFF); | ||
383 | rtlpci->irq_enabled = true; | ||
384 | } else { | ||
385 | rtl_write_dword(rtlpriv, REG_HIMR, rtlusb->irq_mask[0] & | ||
386 | 0xFFFFFFFF); | ||
387 | rtl_write_dword(rtlpriv, REG_HIMRE, rtlusb->irq_mask[1] & | ||
388 | 0xFFFFFFFF); | ||
389 | rtlusb->irq_enabled = true; | ||
390 | } | ||
391 | } | ||
392 | |||
393 | void rtl92c_init_interrupt(struct ieee80211_hw *hw) | ||
394 | { | ||
395 | rtl92c_enable_interrupt(hw); | ||
396 | } | ||
397 | |||
398 | void rtl92c_disable_interrupt(struct ieee80211_hw *hw) | ||
399 | { | ||
400 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
401 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
402 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
403 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
404 | |||
405 | rtl_write_dword(rtlpriv, REG_HIMR, IMR8190_DISABLED); | ||
406 | rtl_write_dword(rtlpriv, REG_HIMRE, IMR8190_DISABLED); | ||
407 | if (IS_HARDWARE_TYPE_8192CE(rtlhal)) | ||
408 | rtlpci->irq_enabled = false; | ||
409 | else if (IS_HARDWARE_TYPE_8192CU(rtlhal)) | ||
410 | rtlusb->irq_enabled = false; | ||
411 | } | ||
412 | |||
413 | void rtl92c_set_qos(struct ieee80211_hw *hw, int aci) | ||
414 | { | ||
415 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
416 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
417 | u32 u4b_ac_param; | ||
418 | |||
419 | rtl92c_dm_init_edca_turbo(hw); | ||
420 | u4b_ac_param = (u32) mac->ac[aci].aifs; | ||
421 | u4b_ac_param |= | ||
422 | ((u32) le16_to_cpu(mac->ac[aci].cw_min) & 0xF) << | ||
423 | AC_PARAM_ECW_MIN_OFFSET; | ||
424 | u4b_ac_param |= | ||
425 | ((u32) le16_to_cpu(mac->ac[aci].cw_max) & 0xF) << | ||
426 | AC_PARAM_ECW_MAX_OFFSET; | ||
427 | u4b_ac_param |= (u32) le16_to_cpu(mac->ac[aci].tx_op) << | ||
428 | AC_PARAM_TXOP_OFFSET; | ||
429 | RT_TRACE(rtlpriv, COMP_QOS, DBG_LOUD, | ||
430 | ("queue:%x, ac_param:%x\n", aci, u4b_ac_param)); | ||
431 | switch (aci) { | ||
432 | case AC1_BK: | ||
433 | rtl_write_dword(rtlpriv, REG_EDCA_BK_PARAM, u4b_ac_param); | ||
434 | break; | ||
435 | case AC0_BE: | ||
436 | rtl_write_dword(rtlpriv, REG_EDCA_BE_PARAM, u4b_ac_param); | ||
437 | break; | ||
438 | case AC2_VI: | ||
439 | rtl_write_dword(rtlpriv, REG_EDCA_VI_PARAM, u4b_ac_param); | ||
440 | break; | ||
441 | case AC3_VO: | ||
442 | rtl_write_dword(rtlpriv, REG_EDCA_VO_PARAM, u4b_ac_param); | ||
443 | break; | ||
444 | default: | ||
445 | RT_ASSERT(false, ("invalid aci: %d !\n", aci)); | ||
446 | break; | ||
447 | } | ||
448 | } | ||
449 | |||
450 | /*------------------------------------------------------------------------- | ||
451 | * HW MAC Address | ||
452 | *-------------------------------------------------------------------------*/ | ||
453 | void rtl92c_set_mac_addr(struct ieee80211_hw *hw, const u8 *addr) | ||
454 | { | ||
455 | u32 i; | ||
456 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
457 | |||
458 | for (i = 0 ; i < ETH_ALEN ; i++) | ||
459 | rtl_write_byte(rtlpriv, (REG_MACID + i), *(addr+i)); | ||
460 | |||
461 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, ("MAC Address: %02X-%02X-%02X-" | ||
462 | "%02X-%02X-%02X\n", | ||
463 | rtl_read_byte(rtlpriv, REG_MACID), | ||
464 | rtl_read_byte(rtlpriv, REG_MACID+1), | ||
465 | rtl_read_byte(rtlpriv, REG_MACID+2), | ||
466 | rtl_read_byte(rtlpriv, REG_MACID+3), | ||
467 | rtl_read_byte(rtlpriv, REG_MACID+4), | ||
468 | rtl_read_byte(rtlpriv, REG_MACID+5))); | ||
469 | } | ||
470 | |||
471 | void rtl92c_init_driver_info_size(struct ieee80211_hw *hw, u8 size) | ||
472 | { | ||
473 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
474 | rtl_write_byte(rtlpriv, REG_RX_DRVINFO_SZ, size); | ||
475 | } | ||
476 | |||
477 | int rtl92c_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type) | ||
478 | { | ||
479 | u8 value; | ||
480 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
481 | |||
482 | switch (type) { | ||
483 | case NL80211_IFTYPE_UNSPECIFIED: | ||
484 | value = NT_NO_LINK; | ||
485 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
486 | ("Set Network type to NO LINK!\n")); | ||
487 | break; | ||
488 | case NL80211_IFTYPE_ADHOC: | ||
489 | value = NT_LINK_AD_HOC; | ||
490 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
491 | ("Set Network type to Ad Hoc!\n")); | ||
492 | break; | ||
493 | case NL80211_IFTYPE_STATION: | ||
494 | value = NT_LINK_AP; | ||
495 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
496 | ("Set Network type to STA!\n")); | ||
497 | break; | ||
498 | case NL80211_IFTYPE_AP: | ||
499 | value = NT_AS_AP; | ||
500 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
501 | ("Set Network type to AP!\n")); | ||
502 | break; | ||
503 | default: | ||
504 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
505 | ("Network type %d not support!\n", type)); | ||
506 | return -EOPNOTSUPP; | ||
507 | } | ||
508 | rtl_write_byte(rtlpriv, (REG_CR + 2), value); | ||
509 | return 0; | ||
510 | } | ||
511 | |||
512 | void rtl92c_init_network_type(struct ieee80211_hw *hw) | ||
513 | { | ||
514 | rtl92c_set_network_type(hw, NL80211_IFTYPE_UNSPECIFIED); | ||
515 | } | ||
516 | |||
517 | void rtl92c_init_adaptive_ctrl(struct ieee80211_hw *hw) | ||
518 | { | ||
519 | u16 value16; | ||
520 | u32 value32; | ||
521 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
522 | |||
523 | /* Response Rate Set */ | ||
524 | value32 = rtl_read_dword(rtlpriv, REG_RRSR); | ||
525 | value32 &= ~RATE_BITMAP_ALL; | ||
526 | value32 |= RATE_RRSR_CCK_ONLY_1M; | ||
527 | rtl_write_dword(rtlpriv, REG_RRSR, value32); | ||
528 | /* SIFS (used in NAV) */ | ||
529 | value16 = _SPEC_SIFS_CCK(0x10) | _SPEC_SIFS_OFDM(0x10); | ||
530 | rtl_write_word(rtlpriv, REG_SPEC_SIFS, value16); | ||
531 | /* Retry Limit */ | ||
532 | value16 = _LRL(0x30) | _SRL(0x30); | ||
533 | rtl_write_dword(rtlpriv, REG_RL, value16); | ||
534 | } | ||
535 | |||
536 | void rtl92c_init_rate_fallback(struct ieee80211_hw *hw) | ||
537 | { | ||
538 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
539 | |||
540 | /* Set Data Auto Rate Fallback Retry Count register. */ | ||
541 | rtl_write_dword(rtlpriv, REG_DARFRC, 0x00000000); | ||
542 | rtl_write_dword(rtlpriv, REG_DARFRC+4, 0x10080404); | ||
543 | rtl_write_dword(rtlpriv, REG_RARFRC, 0x04030201); | ||
544 | rtl_write_dword(rtlpriv, REG_RARFRC+4, 0x08070605); | ||
545 | } | ||
546 | |||
547 | static void rtl92c_set_cck_sifs(struct ieee80211_hw *hw, u8 trx_sifs, | ||
548 | u8 ctx_sifs) | ||
549 | { | ||
550 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
551 | |||
552 | rtl_write_byte(rtlpriv, REG_SIFS_CCK, trx_sifs); | ||
553 | rtl_write_byte(rtlpriv, (REG_SIFS_CCK + 1), ctx_sifs); | ||
554 | } | ||
555 | |||
556 | static void rtl92c_set_ofdm_sifs(struct ieee80211_hw *hw, u8 trx_sifs, | ||
557 | u8 ctx_sifs) | ||
558 | { | ||
559 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
560 | |||
561 | rtl_write_byte(rtlpriv, REG_SIFS_OFDM, trx_sifs); | ||
562 | rtl_write_byte(rtlpriv, (REG_SIFS_OFDM + 1), ctx_sifs); | ||
563 | } | ||
564 | |||
565 | void rtl92c_init_edca_param(struct ieee80211_hw *hw, | ||
566 | u16 queue, u16 txop, u8 cw_min, u8 cw_max, u8 aifs) | ||
567 | { | ||
568 | /* sequence: VO, VI, BE, BK ==> the same as 92C hardware design. | ||
569 | * referenc : enum nl80211_txq_q or ieee80211_set_wmm_default function. | ||
570 | */ | ||
571 | u32 value; | ||
572 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
573 | |||
574 | value = (u32)aifs; | ||
575 | value |= ((u32)cw_min & 0xF) << 8; | ||
576 | value |= ((u32)cw_max & 0xF) << 12; | ||
577 | value |= (u32)txop << 16; | ||
578 | /* 92C hardware register sequence is the same as queue number. */ | ||
579 | rtl_write_dword(rtlpriv, (REG_EDCA_VO_PARAM + (queue * 4)), value); | ||
580 | } | ||
581 | |||
582 | void rtl92c_init_edca(struct ieee80211_hw *hw) | ||
583 | { | ||
584 | u16 value16; | ||
585 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
586 | |||
587 | /* disable EDCCA count down, to reduce collison and retry */ | ||
588 | value16 = rtl_read_word(rtlpriv, REG_RD_CTRL); | ||
589 | value16 |= DIS_EDCA_CNT_DWN; | ||
590 | rtl_write_word(rtlpriv, REG_RD_CTRL, value16); | ||
591 | /* Update SIFS timing. ?????????? | ||
592 | * pHalData->SifsTime = 0x0e0e0a0a; */ | ||
593 | rtl92c_set_cck_sifs(hw, 0xa, 0xa); | ||
594 | rtl92c_set_ofdm_sifs(hw, 0xe, 0xe); | ||
595 | /* Set CCK/OFDM SIFS to be 10us. */ | ||
596 | rtl_write_word(rtlpriv, REG_SIFS_CCK, 0x0a0a); | ||
597 | rtl_write_word(rtlpriv, REG_SIFS_OFDM, 0x1010); | ||
598 | rtl_write_word(rtlpriv, REG_PROT_MODE_CTRL, 0x0204); | ||
599 | rtl_write_dword(rtlpriv, REG_BAR_MODE_CTRL, 0x014004); | ||
600 | /* TXOP */ | ||
601 | rtl_write_dword(rtlpriv, REG_EDCA_BE_PARAM, 0x005EA42B); | ||
602 | rtl_write_dword(rtlpriv, REG_EDCA_BK_PARAM, 0x0000A44F); | ||
603 | rtl_write_dword(rtlpriv, REG_EDCA_VI_PARAM, 0x005EA324); | ||
604 | rtl_write_dword(rtlpriv, REG_EDCA_VO_PARAM, 0x002FA226); | ||
605 | /* PIFS */ | ||
606 | rtl_write_byte(rtlpriv, REG_PIFS, 0x1C); | ||
607 | /* AGGR BREAK TIME Register */ | ||
608 | rtl_write_byte(rtlpriv, REG_AGGR_BREAK_TIME, 0x16); | ||
609 | rtl_write_word(rtlpriv, REG_NAV_PROT_LEN, 0x0040); | ||
610 | rtl_write_byte(rtlpriv, REG_BCNDMATIM, 0x02); | ||
611 | rtl_write_byte(rtlpriv, REG_ATIMWND, 0x02); | ||
612 | } | ||
613 | |||
614 | void rtl92c_init_ampdu_aggregation(struct ieee80211_hw *hw) | ||
615 | { | ||
616 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
617 | |||
618 | rtl_write_dword(rtlpriv, REG_AGGLEN_LMT, 0x99997631); | ||
619 | rtl_write_byte(rtlpriv, REG_AGGR_BREAK_TIME, 0x16); | ||
620 | /* init AMPDU aggregation number, tuning for Tx's TP, */ | ||
621 | rtl_write_word(rtlpriv, 0x4CA, 0x0708); | ||
622 | } | ||
623 | |||
624 | void rtl92c_init_beacon_max_error(struct ieee80211_hw *hw, bool infra_mode) | ||
625 | { | ||
626 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
627 | |||
628 | rtl_write_byte(rtlpriv, REG_BCN_MAX_ERR, 0xFF); | ||
629 | } | ||
630 | |||
631 | void rtl92c_init_rdg_setting(struct ieee80211_hw *hw) | ||
632 | { | ||
633 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
634 | |||
635 | rtl_write_byte(rtlpriv, REG_RD_CTRL, 0xFF); | ||
636 | rtl_write_word(rtlpriv, REG_RD_NAV_NXT, 0x200); | ||
637 | rtl_write_byte(rtlpriv, REG_RD_RESP_PKT_TH, 0x05); | ||
638 | } | ||
639 | |||
640 | void rtl92c_init_retry_function(struct ieee80211_hw *hw) | ||
641 | { | ||
642 | u8 value8; | ||
643 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
644 | |||
645 | value8 = rtl_read_byte(rtlpriv, REG_FWHW_TXQ_CTRL); | ||
646 | value8 |= EN_AMPDU_RTY_NEW; | ||
647 | rtl_write_byte(rtlpriv, REG_FWHW_TXQ_CTRL, value8); | ||
648 | /* Set ACK timeout */ | ||
649 | rtl_write_byte(rtlpriv, REG_ACKTO, 0x40); | ||
650 | } | ||
651 | |||
652 | void rtl92c_init_beacon_parameters(struct ieee80211_hw *hw, | ||
653 | enum version_8192c version) | ||
654 | { | ||
655 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
656 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
657 | |||
658 | rtl_write_word(rtlpriv, REG_TBTT_PROHIBIT, 0x6404);/* ms */ | ||
659 | rtl_write_byte(rtlpriv, REG_DRVERLYINT, DRIVER_EARLY_INT_TIME);/*ms*/ | ||
660 | rtl_write_byte(rtlpriv, REG_BCNDMATIM, BCN_DMA_ATIME_INT_TIME); | ||
661 | if (IS_NORMAL_CHIP(rtlhal->version)) | ||
662 | rtl_write_word(rtlpriv, REG_BCNTCFG, 0x660F); | ||
663 | else | ||
664 | rtl_write_word(rtlpriv, REG_BCNTCFG, 0x66FF); | ||
665 | } | ||
666 | |||
667 | void rtl92c_disable_fast_edca(struct ieee80211_hw *hw) | ||
668 | { | ||
669 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
670 | |||
671 | rtl_write_word(rtlpriv, REG_FAST_EDCA_CTRL, 0); | ||
672 | } | ||
673 | |||
674 | void rtl92c_set_min_space(struct ieee80211_hw *hw, bool is2T) | ||
675 | { | ||
676 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
677 | u8 value = is2T ? MAX_MSS_DENSITY_2T : MAX_MSS_DENSITY_1T; | ||
678 | |||
679 | rtl_write_byte(rtlpriv, REG_AMPDU_MIN_SPACE, value); | ||
680 | } | ||
681 | |||
682 | u16 rtl92c_get_mgt_filter(struct ieee80211_hw *hw) | ||
683 | { | ||
684 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
685 | |||
686 | return rtl_read_word(rtlpriv, REG_RXFLTMAP0); | ||
687 | } | ||
688 | |||
689 | void rtl92c_set_mgt_filter(struct ieee80211_hw *hw, u16 filter) | ||
690 | { | ||
691 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
692 | |||
693 | rtl_write_word(rtlpriv, REG_RXFLTMAP0, filter); | ||
694 | } | ||
695 | |||
696 | u16 rtl92c_get_ctrl_filter(struct ieee80211_hw *hw) | ||
697 | { | ||
698 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
699 | |||
700 | return rtl_read_word(rtlpriv, REG_RXFLTMAP1); | ||
701 | } | ||
702 | |||
703 | void rtl92c_set_ctrl_filter(struct ieee80211_hw *hw, u16 filter) | ||
704 | { | ||
705 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
706 | |||
707 | rtl_write_word(rtlpriv, REG_RXFLTMAP1, filter); | ||
708 | } | ||
709 | |||
710 | u16 rtl92c_get_data_filter(struct ieee80211_hw *hw) | ||
711 | { | ||
712 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
713 | |||
714 | return rtl_read_word(rtlpriv, REG_RXFLTMAP2); | ||
715 | } | ||
716 | |||
717 | void rtl92c_set_data_filter(struct ieee80211_hw *hw, u16 filter) | ||
718 | { | ||
719 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
720 | |||
721 | rtl_write_word(rtlpriv, REG_RXFLTMAP2, filter); | ||
722 | } | ||
723 | /*==============================================================*/ | ||
724 | |||
725 | static u8 _rtl92c_query_rxpwrpercentage(char antpower) | ||
726 | { | ||
727 | if ((antpower <= -100) || (antpower >= 20)) | ||
728 | return 0; | ||
729 | else if (antpower >= 0) | ||
730 | return 100; | ||
731 | else | ||
732 | return 100 + antpower; | ||
733 | } | ||
734 | |||
735 | static u8 _rtl92c_evm_db_to_percentage(char value) | ||
736 | { | ||
737 | char ret_val; | ||
738 | |||
739 | ret_val = value; | ||
740 | if (ret_val >= 0) | ||
741 | ret_val = 0; | ||
742 | if (ret_val <= -33) | ||
743 | ret_val = -33; | ||
744 | ret_val = 0 - ret_val; | ||
745 | ret_val *= 3; | ||
746 | if (ret_val == 99) | ||
747 | ret_val = 100; | ||
748 | return ret_val; | ||
749 | } | ||
750 | |||
751 | static long _rtl92c_translate_todbm(struct ieee80211_hw *hw, | ||
752 | u8 signal_strength_index) | ||
753 | { | ||
754 | long signal_power; | ||
755 | |||
756 | signal_power = (long)((signal_strength_index + 1) >> 1); | ||
757 | signal_power -= 95; | ||
758 | return signal_power; | ||
759 | } | ||
760 | |||
761 | static long _rtl92c_signal_scale_mapping(struct ieee80211_hw *hw, | ||
762 | long currsig) | ||
763 | { | ||
764 | long retsig; | ||
765 | |||
766 | if (currsig >= 61 && currsig <= 100) | ||
767 | retsig = 90 + ((currsig - 60) / 4); | ||
768 | else if (currsig >= 41 && currsig <= 60) | ||
769 | retsig = 78 + ((currsig - 40) / 2); | ||
770 | else if (currsig >= 31 && currsig <= 40) | ||
771 | retsig = 66 + (currsig - 30); | ||
772 | else if (currsig >= 21 && currsig <= 30) | ||
773 | retsig = 54 + (currsig - 20); | ||
774 | else if (currsig >= 5 && currsig <= 20) | ||
775 | retsig = 42 + (((currsig - 5) * 2) / 3); | ||
776 | else if (currsig == 4) | ||
777 | retsig = 36; | ||
778 | else if (currsig == 3) | ||
779 | retsig = 27; | ||
780 | else if (currsig == 2) | ||
781 | retsig = 18; | ||
782 | else if (currsig == 1) | ||
783 | retsig = 9; | ||
784 | else | ||
785 | retsig = currsig; | ||
786 | return retsig; | ||
787 | } | ||
788 | |||
789 | static void _rtl92c_query_rxphystatus(struct ieee80211_hw *hw, | ||
790 | struct rtl_stats *pstats, | ||
791 | struct rx_desc_92c *pdesc, | ||
792 | struct rx_fwinfo_92c *p_drvinfo, | ||
793 | bool packet_match_bssid, | ||
794 | bool packet_toself, | ||
795 | bool packet_beacon) | ||
796 | { | ||
797 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
798 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
799 | struct phy_sts_cck_8192s_t *cck_buf; | ||
800 | s8 rx_pwr_all = 0, rx_pwr[4]; | ||
801 | u8 rf_rx_num = 0, evm, pwdb_all; | ||
802 | u8 i, max_spatial_stream; | ||
803 | u32 rssi, total_rssi = 0; | ||
804 | bool in_powersavemode = false; | ||
805 | bool is_cck_rate; | ||
806 | |||
807 | is_cck_rate = RX_HAL_IS_CCK_RATE(pdesc); | ||
808 | pstats->packet_matchbssid = packet_match_bssid; | ||
809 | pstats->packet_toself = packet_toself; | ||
810 | pstats->is_cck = is_cck_rate; | ||
811 | pstats->packet_beacon = packet_beacon; | ||
812 | pstats->is_cck = is_cck_rate; | ||
813 | pstats->RX_SIGQ[0] = -1; | ||
814 | pstats->RX_SIGQ[1] = -1; | ||
815 | if (is_cck_rate) { | ||
816 | u8 report, cck_highpwr; | ||
817 | cck_buf = (struct phy_sts_cck_8192s_t *)p_drvinfo; | ||
818 | if (!in_powersavemode) | ||
819 | cck_highpwr = rtlphy->cck_high_power; | ||
820 | else | ||
821 | cck_highpwr = false; | ||
822 | if (!cck_highpwr) { | ||
823 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
824 | report = cck_buf->cck_agc_rpt & 0xc0; | ||
825 | report = report >> 6; | ||
826 | switch (report) { | ||
827 | case 0x3: | ||
828 | rx_pwr_all = -46 - (cck_agc_rpt & 0x3e); | ||
829 | break; | ||
830 | case 0x2: | ||
831 | rx_pwr_all = -26 - (cck_agc_rpt & 0x3e); | ||
832 | break; | ||
833 | case 0x1: | ||
834 | rx_pwr_all = -12 - (cck_agc_rpt & 0x3e); | ||
835 | break; | ||
836 | case 0x0: | ||
837 | rx_pwr_all = 16 - (cck_agc_rpt & 0x3e); | ||
838 | break; | ||
839 | } | ||
840 | } else { | ||
841 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
842 | report = p_drvinfo->cfosho[0] & 0x60; | ||
843 | report = report >> 5; | ||
844 | switch (report) { | ||
845 | case 0x3: | ||
846 | rx_pwr_all = -46 - ((cck_agc_rpt & 0x1f) << 1); | ||
847 | break; | ||
848 | case 0x2: | ||
849 | rx_pwr_all = -26 - ((cck_agc_rpt & 0x1f) << 1); | ||
850 | break; | ||
851 | case 0x1: | ||
852 | rx_pwr_all = -12 - ((cck_agc_rpt & 0x1f) << 1); | ||
853 | break; | ||
854 | case 0x0: | ||
855 | rx_pwr_all = 16 - ((cck_agc_rpt & 0x1f) << 1); | ||
856 | break; | ||
857 | } | ||
858 | } | ||
859 | pwdb_all = _rtl92c_query_rxpwrpercentage(rx_pwr_all); | ||
860 | pstats->rx_pwdb_all = pwdb_all; | ||
861 | pstats->recvsignalpower = rx_pwr_all; | ||
862 | if (packet_match_bssid) { | ||
863 | u8 sq; | ||
864 | if (pstats->rx_pwdb_all > 40) | ||
865 | sq = 100; | ||
866 | else { | ||
867 | sq = cck_buf->sq_rpt; | ||
868 | if (sq > 64) | ||
869 | sq = 0; | ||
870 | else if (sq < 20) | ||
871 | sq = 100; | ||
872 | else | ||
873 | sq = ((64 - sq) * 100) / 44; | ||
874 | } | ||
875 | pstats->signalquality = sq; | ||
876 | pstats->RX_SIGQ[0] = sq; | ||
877 | pstats->RX_SIGQ[1] = -1; | ||
878 | } | ||
879 | } else { | ||
880 | rtlpriv->dm.rfpath_rxenable[0] = | ||
881 | rtlpriv->dm.rfpath_rxenable[1] = true; | ||
882 | for (i = RF90_PATH_A; i < RF90_PATH_MAX; i++) { | ||
883 | if (rtlpriv->dm.rfpath_rxenable[i]) | ||
884 | rf_rx_num++; | ||
885 | rx_pwr[i] = | ||
886 | ((p_drvinfo->gain_trsw[i] & 0x3f) * 2) - 110; | ||
887 | rssi = _rtl92c_query_rxpwrpercentage(rx_pwr[i]); | ||
888 | total_rssi += rssi; | ||
889 | rtlpriv->stats.rx_snr_db[i] = | ||
890 | (long)(p_drvinfo->rxsnr[i] / 2); | ||
891 | |||
892 | if (packet_match_bssid) | ||
893 | pstats->rx_mimo_signalstrength[i] = (u8) rssi; | ||
894 | } | ||
895 | rx_pwr_all = ((p_drvinfo->pwdb_all >> 1) & 0x7f) - 110; | ||
896 | pwdb_all = _rtl92c_query_rxpwrpercentage(rx_pwr_all); | ||
897 | pstats->rx_pwdb_all = pwdb_all; | ||
898 | pstats->rxpower = rx_pwr_all; | ||
899 | pstats->recvsignalpower = rx_pwr_all; | ||
900 | if (GET_RX_DESC_RX_MCS(pdesc) && | ||
901 | GET_RX_DESC_RX_MCS(pdesc) >= DESC92C_RATEMCS8 && | ||
902 | GET_RX_DESC_RX_MCS(pdesc) <= DESC92C_RATEMCS15) | ||
903 | max_spatial_stream = 2; | ||
904 | else | ||
905 | max_spatial_stream = 1; | ||
906 | for (i = 0; i < max_spatial_stream; i++) { | ||
907 | evm = _rtl92c_evm_db_to_percentage(p_drvinfo->rxevm[i]); | ||
908 | if (packet_match_bssid) { | ||
909 | if (i == 0) | ||
910 | pstats->signalquality = | ||
911 | (u8) (evm & 0xff); | ||
912 | pstats->RX_SIGQ[i] = | ||
913 | (u8) (evm & 0xff); | ||
914 | } | ||
915 | } | ||
916 | } | ||
917 | if (is_cck_rate) | ||
918 | pstats->signalstrength = | ||
919 | (u8) (_rtl92c_signal_scale_mapping(hw, pwdb_all)); | ||
920 | else if (rf_rx_num != 0) | ||
921 | pstats->signalstrength = | ||
922 | (u8) (_rtl92c_signal_scale_mapping | ||
923 | (hw, total_rssi /= rf_rx_num)); | ||
924 | } | ||
925 | |||
926 | static void _rtl92c_process_ui_rssi(struct ieee80211_hw *hw, | ||
927 | struct rtl_stats *pstats) | ||
928 | { | ||
929 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
930 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
931 | u8 rfpath; | ||
932 | u32 last_rssi, tmpval; | ||
933 | |||
934 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
935 | rtlpriv->stats.rssi_calculate_cnt++; | ||
936 | if (rtlpriv->stats.ui_rssi.total_num++ >= | ||
937 | PHY_RSSI_SLID_WIN_MAX) { | ||
938 | rtlpriv->stats.ui_rssi.total_num = | ||
939 | PHY_RSSI_SLID_WIN_MAX; | ||
940 | last_rssi = | ||
941 | rtlpriv->stats.ui_rssi.elements[rtlpriv-> | ||
942 | stats.ui_rssi.index]; | ||
943 | rtlpriv->stats.ui_rssi.total_val -= last_rssi; | ||
944 | } | ||
945 | rtlpriv->stats.ui_rssi.total_val += pstats->signalstrength; | ||
946 | rtlpriv->stats.ui_rssi.elements[rtlpriv->stats.ui_rssi. | ||
947 | index++] = pstats->signalstrength; | ||
948 | if (rtlpriv->stats.ui_rssi.index >= PHY_RSSI_SLID_WIN_MAX) | ||
949 | rtlpriv->stats.ui_rssi.index = 0; | ||
950 | tmpval = rtlpriv->stats.ui_rssi.total_val / | ||
951 | rtlpriv->stats.ui_rssi.total_num; | ||
952 | rtlpriv->stats.signal_strength = | ||
953 | _rtl92c_translate_todbm(hw, (u8) tmpval); | ||
954 | pstats->rssi = rtlpriv->stats.signal_strength; | ||
955 | } | ||
956 | if (!pstats->is_cck && pstats->packet_toself) { | ||
957 | for (rfpath = RF90_PATH_A; rfpath < rtlphy->num_total_rfpath; | ||
958 | rfpath++) { | ||
959 | if (!rtl8192_phy_check_is_legal_rfpath(hw, rfpath)) | ||
960 | continue; | ||
961 | if (rtlpriv->stats.rx_rssi_percentage[rfpath] == 0) { | ||
962 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
963 | pstats->rx_mimo_signalstrength[rfpath]; | ||
964 | } | ||
965 | if (pstats->rx_mimo_signalstrength[rfpath] > | ||
966 | rtlpriv->stats.rx_rssi_percentage[rfpath]) { | ||
967 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
968 | ((rtlpriv->stats. | ||
969 | rx_rssi_percentage[rfpath] * | ||
970 | (RX_SMOOTH_FACTOR - 1)) + | ||
971 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
972 | (RX_SMOOTH_FACTOR); | ||
973 | |||
974 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
975 | rtlpriv->stats.rx_rssi_percentage[rfpath] + | ||
976 | 1; | ||
977 | } else { | ||
978 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
979 | ((rtlpriv->stats. | ||
980 | rx_rssi_percentage[rfpath] * | ||
981 | (RX_SMOOTH_FACTOR - 1)) + | ||
982 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
983 | (RX_SMOOTH_FACTOR); | ||
984 | } | ||
985 | } | ||
986 | } | ||
987 | } | ||
988 | |||
989 | static void _rtl92c_update_rxsignalstatistics(struct ieee80211_hw *hw, | ||
990 | struct rtl_stats *pstats) | ||
991 | { | ||
992 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
993 | int weighting = 0; | ||
994 | |||
995 | if (rtlpriv->stats.recv_signal_power == 0) | ||
996 | rtlpriv->stats.recv_signal_power = pstats->recvsignalpower; | ||
997 | if (pstats->recvsignalpower > rtlpriv->stats.recv_signal_power) | ||
998 | weighting = 5; | ||
999 | else if (pstats->recvsignalpower < rtlpriv->stats.recv_signal_power) | ||
1000 | weighting = (-5); | ||
1001 | rtlpriv->stats.recv_signal_power = | ||
1002 | (rtlpriv->stats.recv_signal_power * 5 + | ||
1003 | pstats->recvsignalpower + weighting) / 6; | ||
1004 | } | ||
1005 | |||
1006 | static void _rtl92c_process_pwdb(struct ieee80211_hw *hw, | ||
1007 | struct rtl_stats *pstats) | ||
1008 | { | ||
1009 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1010 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1011 | long undecorated_smoothed_pwdb = 0; | ||
1012 | |||
1013 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
1014 | return; | ||
1015 | } else { | ||
1016 | undecorated_smoothed_pwdb = | ||
1017 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
1018 | } | ||
1019 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
1020 | if (undecorated_smoothed_pwdb < 0) | ||
1021 | undecorated_smoothed_pwdb = pstats->rx_pwdb_all; | ||
1022 | if (pstats->rx_pwdb_all > (u32) undecorated_smoothed_pwdb) { | ||
1023 | undecorated_smoothed_pwdb = | ||
1024 | (((undecorated_smoothed_pwdb) * | ||
1025 | (RX_SMOOTH_FACTOR - 1)) + | ||
1026 | (pstats->rx_pwdb_all)) / (RX_SMOOTH_FACTOR); | ||
1027 | undecorated_smoothed_pwdb = undecorated_smoothed_pwdb | ||
1028 | + 1; | ||
1029 | } else { | ||
1030 | undecorated_smoothed_pwdb = | ||
1031 | (((undecorated_smoothed_pwdb) * | ||
1032 | (RX_SMOOTH_FACTOR - 1)) + | ||
1033 | (pstats->rx_pwdb_all)) / (RX_SMOOTH_FACTOR); | ||
1034 | } | ||
1035 | rtlpriv->dm.undecorated_smoothed_pwdb = | ||
1036 | undecorated_smoothed_pwdb; | ||
1037 | _rtl92c_update_rxsignalstatistics(hw, pstats); | ||
1038 | } | ||
1039 | } | ||
1040 | |||
1041 | static void _rtl92c_process_LINK_Q(struct ieee80211_hw *hw, | ||
1042 | struct rtl_stats *pstats) | ||
1043 | { | ||
1044 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1045 | u32 last_evm = 0, n_stream, tmpval; | ||
1046 | |||
1047 | if (pstats->signalquality != 0) { | ||
1048 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
1049 | if (rtlpriv->stats.LINK_Q.total_num++ >= | ||
1050 | PHY_LINKQUALITY_SLID_WIN_MAX) { | ||
1051 | rtlpriv->stats.LINK_Q.total_num = | ||
1052 | PHY_LINKQUALITY_SLID_WIN_MAX; | ||
1053 | last_evm = | ||
1054 | rtlpriv->stats.LINK_Q.elements | ||
1055 | [rtlpriv->stats.LINK_Q.index]; | ||
1056 | rtlpriv->stats.LINK_Q.total_val -= | ||
1057 | last_evm; | ||
1058 | } | ||
1059 | rtlpriv->stats.LINK_Q.total_val += | ||
1060 | pstats->signalquality; | ||
1061 | rtlpriv->stats.LINK_Q.elements | ||
1062 | [rtlpriv->stats.LINK_Q.index++] = | ||
1063 | pstats->signalquality; | ||
1064 | if (rtlpriv->stats.LINK_Q.index >= | ||
1065 | PHY_LINKQUALITY_SLID_WIN_MAX) | ||
1066 | rtlpriv->stats.LINK_Q.index = 0; | ||
1067 | tmpval = rtlpriv->stats.LINK_Q.total_val / | ||
1068 | rtlpriv->stats.LINK_Q.total_num; | ||
1069 | rtlpriv->stats.signal_quality = tmpval; | ||
1070 | rtlpriv->stats.last_sigstrength_inpercent = tmpval; | ||
1071 | for (n_stream = 0; n_stream < 2; | ||
1072 | n_stream++) { | ||
1073 | if (pstats->RX_SIGQ[n_stream] != -1) { | ||
1074 | if (!rtlpriv->stats.RX_EVM[n_stream]) { | ||
1075 | rtlpriv->stats.RX_EVM[n_stream] | ||
1076 | = pstats->RX_SIGQ[n_stream]; | ||
1077 | } | ||
1078 | rtlpriv->stats.RX_EVM[n_stream] = | ||
1079 | ((rtlpriv->stats.RX_EVM | ||
1080 | [n_stream] * | ||
1081 | (RX_SMOOTH_FACTOR - 1)) + | ||
1082 | (pstats->RX_SIGQ | ||
1083 | [n_stream] * 1)) / | ||
1084 | (RX_SMOOTH_FACTOR); | ||
1085 | } | ||
1086 | } | ||
1087 | } | ||
1088 | } else { | ||
1089 | ; | ||
1090 | } | ||
1091 | } | ||
1092 | |||
1093 | static void _rtl92c_process_phyinfo(struct ieee80211_hw *hw, | ||
1094 | u8 *buffer, | ||
1095 | struct rtl_stats *pcurrent_stats) | ||
1096 | { | ||
1097 | if (!pcurrent_stats->packet_matchbssid && | ||
1098 | !pcurrent_stats->packet_beacon) | ||
1099 | return; | ||
1100 | _rtl92c_process_ui_rssi(hw, pcurrent_stats); | ||
1101 | _rtl92c_process_pwdb(hw, pcurrent_stats); | ||
1102 | _rtl92c_process_LINK_Q(hw, pcurrent_stats); | ||
1103 | } | ||
1104 | |||
1105 | void rtl92c_translate_rx_signal_stuff(struct ieee80211_hw *hw, | ||
1106 | struct sk_buff *skb, | ||
1107 | struct rtl_stats *pstats, | ||
1108 | struct rx_desc_92c *pdesc, | ||
1109 | struct rx_fwinfo_92c *p_drvinfo) | ||
1110 | { | ||
1111 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1112 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1113 | struct ieee80211_hdr *hdr; | ||
1114 | u8 *tmp_buf; | ||
1115 | u8 *praddr; | ||
1116 | u8 *psaddr; | ||
1117 | __le16 fc; | ||
1118 | u16 type, cpu_fc; | ||
1119 | bool packet_matchbssid, packet_toself, packet_beacon; | ||
1120 | |||
1121 | tmp_buf = skb->data + pstats->rx_drvinfo_size + pstats->rx_bufshift; | ||
1122 | hdr = (struct ieee80211_hdr *)tmp_buf; | ||
1123 | fc = hdr->frame_control; | ||
1124 | cpu_fc = le16_to_cpu(fc); | ||
1125 | type = WLAN_FC_GET_TYPE(fc); | ||
1126 | praddr = hdr->addr1; | ||
1127 | psaddr = hdr->addr2; | ||
1128 | packet_matchbssid = | ||
1129 | ((IEEE80211_FTYPE_CTL != type) && | ||
1130 | (!compare_ether_addr(mac->bssid, | ||
1131 | (cpu_fc & IEEE80211_FCTL_TODS) ? | ||
1132 | hdr->addr1 : (cpu_fc & IEEE80211_FCTL_FROMDS) ? | ||
1133 | hdr->addr2 : hdr->addr3)) && | ||
1134 | (!pstats->hwerror) && (!pstats->crc) && (!pstats->icv)); | ||
1135 | |||
1136 | packet_toself = packet_matchbssid && | ||
1137 | (!compare_ether_addr(praddr, rtlefuse->dev_addr)); | ||
1138 | if (ieee80211_is_beacon(fc)) | ||
1139 | packet_beacon = true; | ||
1140 | _rtl92c_query_rxphystatus(hw, pstats, pdesc, p_drvinfo, | ||
1141 | packet_matchbssid, packet_toself, | ||
1142 | packet_beacon); | ||
1143 | _rtl92c_process_phyinfo(hw, tmp_buf, pstats); | ||
1144 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/mac.h b/drivers/net/wireless/rtlwifi/rtl8192cu/mac.h new file mode 100644 index 000000000000..298fdb724aa5 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/mac.h | |||
@@ -0,0 +1,180 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92C_MAC_H__ | ||
31 | #define __RTL92C_MAC_H__ | ||
32 | |||
33 | #define LLT_LAST_ENTRY_OF_TX_PKT_BUFFER 255 | ||
34 | #define DRIVER_EARLY_INT_TIME 0x05 | ||
35 | #define BCN_DMA_ATIME_INT_TIME 0x02 | ||
36 | |||
37 | void rtl92c_read_chip_version(struct ieee80211_hw *hw); | ||
38 | bool rtl92c_llt_write(struct ieee80211_hw *hw, u32 address, u32 data); | ||
39 | bool rtl92c_init_llt_table(struct ieee80211_hw *hw, u32 boundary); | ||
40 | void rtl92c_set_key(struct ieee80211_hw *hw, u32 key_index, | ||
41 | u8 *p_macaddr, bool is_group, u8 enc_algo, | ||
42 | bool is_wepkey, bool clear_all); | ||
43 | void rtl92c_enable_interrupt(struct ieee80211_hw *hw); | ||
44 | void rtl92c_disable_interrupt(struct ieee80211_hw *hw); | ||
45 | void rtl92c_set_qos(struct ieee80211_hw *hw, int aci); | ||
46 | |||
47 | |||
48 | /*--------------------------------------------------------------- | ||
49 | * Hardware init functions | ||
50 | *---------------------------------------------------------------*/ | ||
51 | void rtl92c_set_mac_addr(struct ieee80211_hw *hw, const u8 *addr); | ||
52 | void rtl92c_init_interrupt(struct ieee80211_hw *hw); | ||
53 | void rtl92c_init_driver_info_size(struct ieee80211_hw *hw, u8 size); | ||
54 | |||
55 | int rtl92c_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type); | ||
56 | void rtl92c_init_network_type(struct ieee80211_hw *hw); | ||
57 | void rtl92c_init_adaptive_ctrl(struct ieee80211_hw *hw); | ||
58 | void rtl92c_init_rate_fallback(struct ieee80211_hw *hw); | ||
59 | |||
60 | void rtl92c_init_edca_param(struct ieee80211_hw *hw, | ||
61 | u16 queue, | ||
62 | u16 txop, | ||
63 | u8 ecwmax, | ||
64 | u8 ecwmin, | ||
65 | u8 aifs); | ||
66 | |||
67 | void rtl92c_init_edca(struct ieee80211_hw *hw); | ||
68 | void rtl92c_init_ampdu_aggregation(struct ieee80211_hw *hw); | ||
69 | void rtl92c_init_beacon_max_error(struct ieee80211_hw *hw, bool infra_mode); | ||
70 | void rtl92c_init_rdg_setting(struct ieee80211_hw *hw); | ||
71 | void rtl92c_init_retry_function(struct ieee80211_hw *hw); | ||
72 | |||
73 | void rtl92c_init_beacon_parameters(struct ieee80211_hw *hw, | ||
74 | enum version_8192c version); | ||
75 | |||
76 | void rtl92c_disable_fast_edca(struct ieee80211_hw *hw); | ||
77 | void rtl92c_set_min_space(struct ieee80211_hw *hw, bool is2T); | ||
78 | |||
79 | /* For filter */ | ||
80 | u16 rtl92c_get_mgt_filter(struct ieee80211_hw *hw); | ||
81 | void rtl92c_set_mgt_filter(struct ieee80211_hw *hw, u16 filter); | ||
82 | u16 rtl92c_get_ctrl_filter(struct ieee80211_hw *hw); | ||
83 | void rtl92c_set_ctrl_filter(struct ieee80211_hw *hw, u16 filter); | ||
84 | u16 rtl92c_get_data_filter(struct ieee80211_hw *hw); | ||
85 | void rtl92c_set_data_filter(struct ieee80211_hw *hw, u16 filter); | ||
86 | |||
87 | |||
88 | u32 rtl92c_get_txdma_status(struct ieee80211_hw *hw); | ||
89 | |||
90 | #define RX_HAL_IS_CCK_RATE(_pdesc)\ | ||
91 | (GET_RX_DESC_RX_MCS(_pdesc) == DESC92C_RATE1M ||\ | ||
92 | GET_RX_DESC_RX_MCS(_pdesc) == DESC92C_RATE2M ||\ | ||
93 | GET_RX_DESC_RX_MCS(_pdesc) == DESC92C_RATE5_5M ||\ | ||
94 | GET_RX_DESC_RX_MCS(_pdesc) == DESC92C_RATE11M) | ||
95 | |||
96 | struct rx_fwinfo_92c { | ||
97 | u8 gain_trsw[4]; | ||
98 | u8 pwdb_all; | ||
99 | u8 cfosho[4]; | ||
100 | u8 cfotail[4]; | ||
101 | char rxevm[2]; | ||
102 | char rxsnr[4]; | ||
103 | u8 pdsnr[2]; | ||
104 | u8 csi_current[2]; | ||
105 | u8 csi_target[2]; | ||
106 | u8 sigevm; | ||
107 | u8 max_ex_pwr; | ||
108 | u8 ex_intf_flag:1; | ||
109 | u8 sgi_en:1; | ||
110 | u8 rxsc:2; | ||
111 | u8 reserve:4; | ||
112 | } __packed; | ||
113 | |||
114 | struct rx_desc_92c { | ||
115 | u32 length:14; | ||
116 | u32 crc32:1; | ||
117 | u32 icverror:1; | ||
118 | u32 drv_infosize:4; | ||
119 | u32 security:3; | ||
120 | u32 qos:1; | ||
121 | u32 shift:2; | ||
122 | u32 phystatus:1; | ||
123 | u32 swdec:1; | ||
124 | u32 lastseg:1; | ||
125 | u32 firstseg:1; | ||
126 | u32 eor:1; | ||
127 | u32 own:1; | ||
128 | u32 macid:5; /* word 1 */ | ||
129 | u32 tid:4; | ||
130 | u32 hwrsvd:5; | ||
131 | u32 paggr:1; | ||
132 | u32 faggr:1; | ||
133 | u32 a1_fit:4; | ||
134 | u32 a2_fit:4; | ||
135 | u32 pam:1; | ||
136 | u32 pwr:1; | ||
137 | u32 moredata:1; | ||
138 | u32 morefrag:1; | ||
139 | u32 type:2; | ||
140 | u32 mc:1; | ||
141 | u32 bc:1; | ||
142 | u32 seq:12; /* word 2 */ | ||
143 | u32 frag:4; | ||
144 | u32 nextpktlen:14; | ||
145 | u32 nextind:1; | ||
146 | u32 rsvd:1; | ||
147 | u32 rxmcs:6; /* word 3 */ | ||
148 | u32 rxht:1; | ||
149 | u32 amsdu:1; | ||
150 | u32 splcp:1; | ||
151 | u32 bandwidth:1; | ||
152 | u32 htc:1; | ||
153 | u32 tcpchk_rpt:1; | ||
154 | u32 ipcchk_rpt:1; | ||
155 | u32 tcpchk_valid:1; | ||
156 | u32 hwpcerr:1; | ||
157 | u32 hwpcind:1; | ||
158 | u32 iv0:16; | ||
159 | u32 iv1; /* word 4 */ | ||
160 | u32 tsfl; /* word 5 */ | ||
161 | u32 bufferaddress; /* word 6 */ | ||
162 | u32 bufferaddress64; /* word 7 */ | ||
163 | } __packed; | ||
164 | |||
165 | enum rtl_desc_qsel rtl92c_map_hwqueue_to_fwqueue(u16 fc, | ||
166 | unsigned int | ||
167 | skb_queue); | ||
168 | void rtl92c_translate_rx_signal_stuff(struct ieee80211_hw *hw, | ||
169 | struct sk_buff *skb, | ||
170 | struct rtl_stats *pstats, | ||
171 | struct rx_desc_92c *pdesc, | ||
172 | struct rx_fwinfo_92c *p_drvinfo); | ||
173 | |||
174 | /*--------------------------------------------------------------- | ||
175 | * Card disable functions | ||
176 | *---------------------------------------------------------------*/ | ||
177 | |||
178 | |||
179 | |||
180 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/phy.c b/drivers/net/wireless/rtlwifi/rtl8192cu/phy.c new file mode 100644 index 000000000000..9a3d0239e27e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/phy.c | |||
@@ -0,0 +1,607 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../ps.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "phy.h" | ||
36 | #include "rf.h" | ||
37 | #include "dm.h" | ||
38 | #include "table.h" | ||
39 | |||
40 | u32 rtl92cu_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
41 | enum radio_path rfpath, u32 regaddr, u32 bitmask) | ||
42 | { | ||
43 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
44 | u32 original_value, readback_value, bitshift; | ||
45 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
46 | |||
47 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), " | ||
48 | "rfpath(%#x), bitmask(%#x)\n", | ||
49 | regaddr, rfpath, bitmask)); | ||
50 | if (rtlphy->rf_mode != RF_OP_BY_FW) { | ||
51 | original_value = _rtl92c_phy_rf_serial_read(hw, | ||
52 | rfpath, regaddr); | ||
53 | } else { | ||
54 | original_value = _rtl92c_phy_fw_rf_serial_read(hw, | ||
55 | rfpath, regaddr); | ||
56 | } | ||
57 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
58 | readback_value = (original_value & bitmask) >> bitshift; | ||
59 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
60 | ("regaddr(%#x), rfpath(%#x), " | ||
61 | "bitmask(%#x), original_value(%#x)\n", | ||
62 | regaddr, rfpath, bitmask, original_value)); | ||
63 | return readback_value; | ||
64 | } | ||
65 | |||
66 | void rtl92cu_phy_set_rf_reg(struct ieee80211_hw *hw, | ||
67 | enum radio_path rfpath, | ||
68 | u32 regaddr, u32 bitmask, u32 data) | ||
69 | { | ||
70 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
71 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
72 | u32 original_value, bitshift; | ||
73 | |||
74 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
75 | ("regaddr(%#x), bitmask(%#x), data(%#x), rfpath(%#x)\n", | ||
76 | regaddr, bitmask, data, rfpath)); | ||
77 | if (rtlphy->rf_mode != RF_OP_BY_FW) { | ||
78 | if (bitmask != RFREG_OFFSET_MASK) { | ||
79 | original_value = _rtl92c_phy_rf_serial_read(hw, | ||
80 | rfpath, | ||
81 | regaddr); | ||
82 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
83 | data = | ||
84 | ((original_value & (~bitmask)) | | ||
85 | (data << bitshift)); | ||
86 | } | ||
87 | _rtl92c_phy_rf_serial_write(hw, rfpath, regaddr, data); | ||
88 | } else { | ||
89 | if (bitmask != RFREG_OFFSET_MASK) { | ||
90 | original_value = _rtl92c_phy_fw_rf_serial_read(hw, | ||
91 | rfpath, | ||
92 | regaddr); | ||
93 | bitshift = _rtl92c_phy_calculate_bit_shift(bitmask); | ||
94 | data = | ||
95 | ((original_value & (~bitmask)) | | ||
96 | (data << bitshift)); | ||
97 | } | ||
98 | _rtl92c_phy_fw_rf_serial_write(hw, rfpath, regaddr, data); | ||
99 | } | ||
100 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), " | ||
101 | "bitmask(%#x), data(%#x), rfpath(%#x)\n", | ||
102 | regaddr, bitmask, data, rfpath)); | ||
103 | } | ||
104 | |||
105 | bool rtl92cu_phy_mac_config(struct ieee80211_hw *hw) | ||
106 | { | ||
107 | bool rtstatus; | ||
108 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
109 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
110 | bool is92c = IS_92C_SERIAL(rtlhal->version); | ||
111 | |||
112 | rtstatus = _rtl92cu_phy_config_mac_with_headerfile(hw); | ||
113 | if (is92c && IS_HARDWARE_TYPE_8192CE(rtlhal)) | ||
114 | rtl_write_byte(rtlpriv, 0x14, 0x71); | ||
115 | return rtstatus; | ||
116 | } | ||
117 | |||
118 | bool rtl92cu_phy_bb_config(struct ieee80211_hw *hw) | ||
119 | { | ||
120 | bool rtstatus = true; | ||
121 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
122 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
123 | u16 regval; | ||
124 | u8 b_reg_hwparafile = 1; | ||
125 | |||
126 | _rtl92c_phy_init_bb_rf_register_definition(hw); | ||
127 | regval = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
128 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, regval | BIT(13) | | ||
129 | BIT(0) | BIT(1)); | ||
130 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL, 0x83); | ||
131 | rtl_write_byte(rtlpriv, REG_AFE_PLL_CTRL + 1, 0xdb); | ||
132 | rtl_write_byte(rtlpriv, REG_RF_CTRL, RF_EN | RF_RSTB | RF_SDMRSTB); | ||
133 | if (IS_HARDWARE_TYPE_8192CE(rtlhal)) { | ||
134 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, FEN_PPLL | FEN_PCIEA | | ||
135 | FEN_DIO_PCIE | FEN_BB_GLB_RSTn | FEN_BBRSTB); | ||
136 | } else if (IS_HARDWARE_TYPE_8192CU(rtlhal)) { | ||
137 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, FEN_USBA | FEN_USBD | | ||
138 | FEN_BB_GLB_RSTn | FEN_BBRSTB); | ||
139 | rtl_write_byte(rtlpriv, REG_LDOHCI12_CTRL, 0x0f); | ||
140 | } | ||
141 | rtl_write_byte(rtlpriv, REG_AFE_XTAL_CTRL + 1, 0x80); | ||
142 | if (b_reg_hwparafile == 1) | ||
143 | rtstatus = _rtl92c_phy_bb8192c_config_parafile(hw); | ||
144 | return rtstatus; | ||
145 | } | ||
146 | |||
147 | bool _rtl92cu_phy_config_mac_with_headerfile(struct ieee80211_hw *hw) | ||
148 | { | ||
149 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
150 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
151 | u32 i; | ||
152 | u32 arraylength; | ||
153 | u32 *ptrarray; | ||
154 | |||
155 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Read Rtl819XMACPHY_Array\n")); | ||
156 | arraylength = rtlphy->hwparam_tables[MAC_REG].length ; | ||
157 | ptrarray = rtlphy->hwparam_tables[MAC_REG].pdata; | ||
158 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
159 | ("Img:RTL8192CEMAC_2T_ARRAY\n")); | ||
160 | for (i = 0; i < arraylength; i = i + 2) | ||
161 | rtl_write_byte(rtlpriv, ptrarray[i], (u8) ptrarray[i + 1]); | ||
162 | return true; | ||
163 | } | ||
164 | |||
165 | bool _rtl92cu_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
166 | u8 configtype) | ||
167 | { | ||
168 | int i; | ||
169 | u32 *phy_regarray_table; | ||
170 | u32 *agctab_array_table; | ||
171 | u16 phy_reg_arraylen, agctab_arraylen; | ||
172 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
173 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
174 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
175 | |||
176 | if (IS_92C_SERIAL(rtlhal->version)) { | ||
177 | agctab_arraylen = rtlphy->hwparam_tables[AGCTAB_2T].length; | ||
178 | agctab_array_table = rtlphy->hwparam_tables[AGCTAB_2T].pdata; | ||
179 | phy_reg_arraylen = rtlphy->hwparam_tables[PHY_REG_2T].length; | ||
180 | phy_regarray_table = rtlphy->hwparam_tables[PHY_REG_2T].pdata; | ||
181 | } else { | ||
182 | agctab_arraylen = rtlphy->hwparam_tables[AGCTAB_1T].length; | ||
183 | agctab_array_table = rtlphy->hwparam_tables[AGCTAB_1T].pdata; | ||
184 | phy_reg_arraylen = rtlphy->hwparam_tables[PHY_REG_1T].length; | ||
185 | phy_regarray_table = rtlphy->hwparam_tables[PHY_REG_1T].pdata; | ||
186 | } | ||
187 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
188 | for (i = 0; i < phy_reg_arraylen; i = i + 2) { | ||
189 | if (phy_regarray_table[i] == 0xfe) | ||
190 | mdelay(50); | ||
191 | else if (phy_regarray_table[i] == 0xfd) | ||
192 | mdelay(5); | ||
193 | else if (phy_regarray_table[i] == 0xfc) | ||
194 | mdelay(1); | ||
195 | else if (phy_regarray_table[i] == 0xfb) | ||
196 | udelay(50); | ||
197 | else if (phy_regarray_table[i] == 0xfa) | ||
198 | udelay(5); | ||
199 | else if (phy_regarray_table[i] == 0xf9) | ||
200 | udelay(1); | ||
201 | rtl_set_bbreg(hw, phy_regarray_table[i], MASKDWORD, | ||
202 | phy_regarray_table[i + 1]); | ||
203 | udelay(1); | ||
204 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
205 | ("The phy_regarray_table[0] is %x" | ||
206 | " Rtl819XPHY_REGArray[1] is %x\n", | ||
207 | phy_regarray_table[i], | ||
208 | phy_regarray_table[i + 1])); | ||
209 | } | ||
210 | } else if (configtype == BASEBAND_CONFIG_AGC_TAB) { | ||
211 | for (i = 0; i < agctab_arraylen; i = i + 2) { | ||
212 | rtl_set_bbreg(hw, agctab_array_table[i], MASKDWORD, | ||
213 | agctab_array_table[i + 1]); | ||
214 | udelay(1); | ||
215 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
216 | ("The agctab_array_table[0] is " | ||
217 | "%x Rtl819XPHY_REGArray[1] is %x\n", | ||
218 | agctab_array_table[i], | ||
219 | agctab_array_table[i + 1])); | ||
220 | } | ||
221 | } | ||
222 | return true; | ||
223 | } | ||
224 | |||
225 | bool _rtl92cu_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
226 | u8 configtype) | ||
227 | { | ||
228 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
229 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
230 | int i; | ||
231 | u32 *phy_regarray_table_pg; | ||
232 | u16 phy_regarray_pg_len; | ||
233 | |||
234 | rtlphy->pwrgroup_cnt = 0; | ||
235 | phy_regarray_pg_len = rtlphy->hwparam_tables[PHY_REG_PG].length; | ||
236 | phy_regarray_table_pg = rtlphy->hwparam_tables[PHY_REG_PG].pdata; | ||
237 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
238 | for (i = 0; i < phy_regarray_pg_len; i = i + 3) { | ||
239 | if (phy_regarray_table_pg[i] == 0xfe) | ||
240 | mdelay(50); | ||
241 | else if (phy_regarray_table_pg[i] == 0xfd) | ||
242 | mdelay(5); | ||
243 | else if (phy_regarray_table_pg[i] == 0xfc) | ||
244 | mdelay(1); | ||
245 | else if (phy_regarray_table_pg[i] == 0xfb) | ||
246 | udelay(50); | ||
247 | else if (phy_regarray_table_pg[i] == 0xfa) | ||
248 | udelay(5); | ||
249 | else if (phy_regarray_table_pg[i] == 0xf9) | ||
250 | udelay(1); | ||
251 | _rtl92c_store_pwrIndex_diffrate_offset(hw, | ||
252 | phy_regarray_table_pg[i], | ||
253 | phy_regarray_table_pg[i + 1], | ||
254 | phy_regarray_table_pg[i + 2]); | ||
255 | } | ||
256 | } else { | ||
257 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, | ||
258 | ("configtype != BaseBand_Config_PHY_REG\n")); | ||
259 | } | ||
260 | return true; | ||
261 | } | ||
262 | |||
263 | bool rtl92cu_phy_config_rf_with_headerfile(struct ieee80211_hw *hw, | ||
264 | enum radio_path rfpath) | ||
265 | { | ||
266 | int i; | ||
267 | u32 *radioa_array_table; | ||
268 | u32 *radiob_array_table; | ||
269 | u16 radioa_arraylen, radiob_arraylen; | ||
270 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
271 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
272 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
273 | |||
274 | if (IS_92C_SERIAL(rtlhal->version)) { | ||
275 | radioa_arraylen = rtlphy->hwparam_tables[RADIOA_2T].length; | ||
276 | radioa_array_table = rtlphy->hwparam_tables[RADIOA_2T].pdata; | ||
277 | radiob_arraylen = rtlphy->hwparam_tables[RADIOB_2T].length; | ||
278 | radiob_array_table = rtlphy->hwparam_tables[RADIOB_2T].pdata; | ||
279 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
280 | ("Radio_A:RTL8192CERADIOA_2TARRAY\n")); | ||
281 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
282 | ("Radio_B:RTL8192CE_RADIOB_2TARRAY\n")); | ||
283 | } else { | ||
284 | radioa_arraylen = rtlphy->hwparam_tables[RADIOA_1T].length; | ||
285 | radioa_array_table = rtlphy->hwparam_tables[RADIOA_1T].pdata; | ||
286 | radiob_arraylen = rtlphy->hwparam_tables[RADIOB_1T].length; | ||
287 | radiob_array_table = rtlphy->hwparam_tables[RADIOB_1T].pdata; | ||
288 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
289 | ("Radio_A:RTL8192CE_RADIOA_1TARRAY\n")); | ||
290 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
291 | ("Radio_B:RTL8192CE_RADIOB_1TARRAY\n")); | ||
292 | } | ||
293 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("Radio No %x\n", rfpath)); | ||
294 | switch (rfpath) { | ||
295 | case RF90_PATH_A: | ||
296 | for (i = 0; i < radioa_arraylen; i = i + 2) { | ||
297 | if (radioa_array_table[i] == 0xfe) | ||
298 | mdelay(50); | ||
299 | else if (radioa_array_table[i] == 0xfd) | ||
300 | mdelay(5); | ||
301 | else if (radioa_array_table[i] == 0xfc) | ||
302 | mdelay(1); | ||
303 | else if (radioa_array_table[i] == 0xfb) | ||
304 | udelay(50); | ||
305 | else if (radioa_array_table[i] == 0xfa) | ||
306 | udelay(5); | ||
307 | else if (radioa_array_table[i] == 0xf9) | ||
308 | udelay(1); | ||
309 | else { | ||
310 | rtl_set_rfreg(hw, rfpath, radioa_array_table[i], | ||
311 | RFREG_OFFSET_MASK, | ||
312 | radioa_array_table[i + 1]); | ||
313 | udelay(1); | ||
314 | } | ||
315 | } | ||
316 | break; | ||
317 | case RF90_PATH_B: | ||
318 | for (i = 0; i < radiob_arraylen; i = i + 2) { | ||
319 | if (radiob_array_table[i] == 0xfe) { | ||
320 | mdelay(50); | ||
321 | } else if (radiob_array_table[i] == 0xfd) | ||
322 | mdelay(5); | ||
323 | else if (radiob_array_table[i] == 0xfc) | ||
324 | mdelay(1); | ||
325 | else if (radiob_array_table[i] == 0xfb) | ||
326 | udelay(50); | ||
327 | else if (radiob_array_table[i] == 0xfa) | ||
328 | udelay(5); | ||
329 | else if (radiob_array_table[i] == 0xf9) | ||
330 | udelay(1); | ||
331 | else { | ||
332 | rtl_set_rfreg(hw, rfpath, radiob_array_table[i], | ||
333 | RFREG_OFFSET_MASK, | ||
334 | radiob_array_table[i + 1]); | ||
335 | udelay(1); | ||
336 | } | ||
337 | } | ||
338 | break; | ||
339 | case RF90_PATH_C: | ||
340 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
341 | ("switch case not process\n")); | ||
342 | break; | ||
343 | case RF90_PATH_D: | ||
344 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
345 | ("switch case not process\n")); | ||
346 | break; | ||
347 | } | ||
348 | return true; | ||
349 | } | ||
350 | |||
351 | void rtl92cu_phy_set_bw_mode_callback(struct ieee80211_hw *hw) | ||
352 | { | ||
353 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
354 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
355 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
356 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
357 | u8 reg_bw_opmode; | ||
358 | u8 reg_prsr_rsc; | ||
359 | |||
360 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, | ||
361 | ("Switch to %s bandwidth\n", | ||
362 | rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20 ? | ||
363 | "20MHz" : "40MHz")) | ||
364 | if (is_hal_stop(rtlhal)) { | ||
365 | rtlphy->set_bwmode_inprogress = false; | ||
366 | return; | ||
367 | } | ||
368 | reg_bw_opmode = rtl_read_byte(rtlpriv, REG_BWOPMODE); | ||
369 | reg_prsr_rsc = rtl_read_byte(rtlpriv, REG_RRSR + 2); | ||
370 | switch (rtlphy->current_chan_bw) { | ||
371 | case HT_CHANNEL_WIDTH_20: | ||
372 | reg_bw_opmode |= BW_OPMODE_20MHZ; | ||
373 | rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode); | ||
374 | break; | ||
375 | case HT_CHANNEL_WIDTH_20_40: | ||
376 | reg_bw_opmode &= ~BW_OPMODE_20MHZ; | ||
377 | rtl_write_byte(rtlpriv, REG_BWOPMODE, reg_bw_opmode); | ||
378 | reg_prsr_rsc = | ||
379 | (reg_prsr_rsc & 0x90) | (mac->cur_40_prime_sc << 5); | ||
380 | rtl_write_byte(rtlpriv, REG_RRSR + 2, reg_prsr_rsc); | ||
381 | break; | ||
382 | default: | ||
383 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
384 | ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw)); | ||
385 | break; | ||
386 | } | ||
387 | switch (rtlphy->current_chan_bw) { | ||
388 | case HT_CHANNEL_WIDTH_20: | ||
389 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x0); | ||
390 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x0); | ||
391 | rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 1); | ||
392 | break; | ||
393 | case HT_CHANNEL_WIDTH_20_40: | ||
394 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x1); | ||
395 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x1); | ||
396 | rtl_set_bbreg(hw, RCCK0_SYSTEM, BCCK_SIDEBAND, | ||
397 | (mac->cur_40_prime_sc >> 1)); | ||
398 | rtl_set_bbreg(hw, ROFDM1_LSTF, 0xC00, mac->cur_40_prime_sc); | ||
399 | rtl_set_bbreg(hw, RFPGA0_ANALOGPARAMETER2, BIT(10), 0); | ||
400 | rtl_set_bbreg(hw, 0x818, (BIT(26) | BIT(27)), | ||
401 | (mac->cur_40_prime_sc == | ||
402 | HAL_PRIME_CHNL_OFFSET_LOWER) ? 2 : 1); | ||
403 | break; | ||
404 | default: | ||
405 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
406 | ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw)); | ||
407 | break; | ||
408 | } | ||
409 | rtl92cu_phy_rf6052_set_bandwidth(hw, rtlphy->current_chan_bw); | ||
410 | rtlphy->set_bwmode_inprogress = false; | ||
411 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n")); | ||
412 | } | ||
413 | |||
414 | void rtl92cu_bb_block_on(struct ieee80211_hw *hw) | ||
415 | { | ||
416 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
417 | |||
418 | mutex_lock(&rtlpriv->io.bb_mutex); | ||
419 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BCCKEN, 0x1); | ||
420 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BOFDMEN, 0x1); | ||
421 | mutex_unlock(&rtlpriv->io.bb_mutex); | ||
422 | } | ||
423 | |||
424 | void _rtl92cu_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t) | ||
425 | { | ||
426 | u8 tmpreg; | ||
427 | u32 rf_a_mode = 0, rf_b_mode = 0, lc_cal; | ||
428 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
429 | |||
430 | tmpreg = rtl_read_byte(rtlpriv, 0xd03); | ||
431 | |||
432 | if ((tmpreg & 0x70) != 0) | ||
433 | rtl_write_byte(rtlpriv, 0xd03, tmpreg & 0x8F); | ||
434 | else | ||
435 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0xFF); | ||
436 | |||
437 | if ((tmpreg & 0x70) != 0) { | ||
438 | rf_a_mode = rtl_get_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS); | ||
439 | if (is2t) | ||
440 | rf_b_mode = rtl_get_rfreg(hw, RF90_PATH_B, 0x00, | ||
441 | MASK12BITS); | ||
442 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS, | ||
443 | (rf_a_mode & 0x8FFFF) | 0x10000); | ||
444 | if (is2t) | ||
445 | rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS, | ||
446 | (rf_b_mode & 0x8FFFF) | 0x10000); | ||
447 | } | ||
448 | lc_cal = rtl_get_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS); | ||
449 | rtl_set_rfreg(hw, RF90_PATH_A, 0x18, MASK12BITS, lc_cal | 0x08000); | ||
450 | mdelay(100); | ||
451 | if ((tmpreg & 0x70) != 0) { | ||
452 | rtl_write_byte(rtlpriv, 0xd03, tmpreg); | ||
453 | rtl_set_rfreg(hw, RF90_PATH_A, 0x00, MASK12BITS, rf_a_mode); | ||
454 | if (is2t) | ||
455 | rtl_set_rfreg(hw, RF90_PATH_B, 0x00, MASK12BITS, | ||
456 | rf_b_mode); | ||
457 | } else { | ||
458 | rtl_write_byte(rtlpriv, REG_TXPAUSE, 0x00); | ||
459 | } | ||
460 | } | ||
461 | |||
462 | static bool _rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
463 | enum rf_pwrstate rfpwr_state) | ||
464 | { | ||
465 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
466 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
467 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
468 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
469 | bool bresult = true; | ||
470 | u8 i, queue_id; | ||
471 | struct rtl8192_tx_ring *ring = NULL; | ||
472 | |||
473 | ppsc->set_rfpowerstate_inprogress = true; | ||
474 | switch (rfpwr_state) { | ||
475 | case ERFON: | ||
476 | if ((ppsc->rfpwr_state == ERFOFF) && | ||
477 | RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC)) { | ||
478 | bool rtstatus; | ||
479 | u32 InitializeCount = 0; | ||
480 | |||
481 | do { | ||
482 | InitializeCount++; | ||
483 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
484 | ("IPS Set eRf nic enable\n")); | ||
485 | rtstatus = rtl_ps_enable_nic(hw); | ||
486 | } while ((rtstatus != true) | ||
487 | && (InitializeCount < 10)); | ||
488 | RT_CLEAR_PS_LEVEL(ppsc, | ||
489 | RT_RF_OFF_LEVL_HALT_NIC); | ||
490 | } else { | ||
491 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
492 | ("Set ERFON sleeped:%d ms\n", | ||
493 | jiffies_to_msecs(jiffies - | ||
494 | ppsc-> | ||
495 | last_sleep_jiffies))); | ||
496 | ppsc->last_awake_jiffies = jiffies; | ||
497 | rtl92ce_phy_set_rf_on(hw); | ||
498 | } | ||
499 | if (mac->link_state == MAC80211_LINKED) { | ||
500 | rtlpriv->cfg->ops->led_control(hw, | ||
501 | LED_CTL_LINK); | ||
502 | } else { | ||
503 | rtlpriv->cfg->ops->led_control(hw, | ||
504 | LED_CTL_NO_LINK); | ||
505 | } | ||
506 | break; | ||
507 | case ERFOFF: | ||
508 | for (queue_id = 0, i = 0; | ||
509 | queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) { | ||
510 | ring = &pcipriv->dev.tx_ring[queue_id]; | ||
511 | if (skb_queue_len(&ring->queue) == 0 || | ||
512 | queue_id == BEACON_QUEUE) { | ||
513 | queue_id++; | ||
514 | continue; | ||
515 | } else { | ||
516 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
517 | ("eRf Off/Sleep: %d times " | ||
518 | "TcbBusyQueue[%d] " | ||
519 | "=%d before doze!\n", (i + 1), | ||
520 | queue_id, | ||
521 | skb_queue_len(&ring->queue))); | ||
522 | udelay(10); | ||
523 | i++; | ||
524 | } | ||
525 | if (i >= MAX_DOZE_WAITING_TIMES_9x) { | ||
526 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
527 | ("\nERFOFF: %d times " | ||
528 | "TcbBusyQueue[%d] = %d !\n", | ||
529 | MAX_DOZE_WAITING_TIMES_9x, | ||
530 | queue_id, | ||
531 | skb_queue_len(&ring->queue))); | ||
532 | break; | ||
533 | } | ||
534 | } | ||
535 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC) { | ||
536 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
537 | ("IPS Set eRf nic disable\n")); | ||
538 | rtl_ps_disable_nic(hw); | ||
539 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
540 | } else { | ||
541 | if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS) { | ||
542 | rtlpriv->cfg->ops->led_control(hw, | ||
543 | LED_CTL_NO_LINK); | ||
544 | } else { | ||
545 | rtlpriv->cfg->ops->led_control(hw, | ||
546 | LED_CTL_POWER_OFF); | ||
547 | } | ||
548 | } | ||
549 | break; | ||
550 | case ERFSLEEP: | ||
551 | if (ppsc->rfpwr_state == ERFOFF) | ||
552 | break; | ||
553 | for (queue_id = 0, i = 0; | ||
554 | queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) { | ||
555 | ring = &pcipriv->dev.tx_ring[queue_id]; | ||
556 | if (skb_queue_len(&ring->queue) == 0) { | ||
557 | queue_id++; | ||
558 | continue; | ||
559 | } else { | ||
560 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
561 | ("eRf Off/Sleep: %d times " | ||
562 | "TcbBusyQueue[%d] =%d before " | ||
563 | "doze!\n", (i + 1), queue_id, | ||
564 | skb_queue_len(&ring->queue))); | ||
565 | udelay(10); | ||
566 | i++; | ||
567 | } | ||
568 | if (i >= MAX_DOZE_WAITING_TIMES_9x) { | ||
569 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
570 | ("\n ERFSLEEP: %d times " | ||
571 | "TcbBusyQueue[%d] = %d !\n", | ||
572 | MAX_DOZE_WAITING_TIMES_9x, | ||
573 | queue_id, | ||
574 | skb_queue_len(&ring->queue))); | ||
575 | break; | ||
576 | } | ||
577 | } | ||
578 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
579 | ("Set ERFSLEEP awaked:%d ms\n", | ||
580 | jiffies_to_msecs(jiffies - | ||
581 | ppsc->last_awake_jiffies))); | ||
582 | ppsc->last_sleep_jiffies = jiffies; | ||
583 | _rtl92c_phy_set_rf_sleep(hw); | ||
584 | break; | ||
585 | default: | ||
586 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
587 | ("switch case not process\n")); | ||
588 | bresult = false; | ||
589 | break; | ||
590 | } | ||
591 | if (bresult) | ||
592 | ppsc->rfpwr_state = rfpwr_state; | ||
593 | ppsc->set_rfpowerstate_inprogress = false; | ||
594 | return bresult; | ||
595 | } | ||
596 | |||
597 | bool rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
598 | enum rf_pwrstate rfpwr_state) | ||
599 | { | ||
600 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
601 | bool bresult = false; | ||
602 | |||
603 | if (rfpwr_state == ppsc->rfpwr_state) | ||
604 | return bresult; | ||
605 | bresult = _rtl92cu_phy_set_rf_power_state(hw, rfpwr_state); | ||
606 | return bresult; | ||
607 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/phy.h b/drivers/net/wireless/rtlwifi/rtl8192cu/phy.h new file mode 100644 index 000000000000..ff81a61729d7 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/phy.h | |||
@@ -0,0 +1,50 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../rtl8192ce/phy.h" | ||
31 | |||
32 | void rtl92cu_bb_block_on(struct ieee80211_hw *hw); | ||
33 | bool rtl8192_phy_check_is_legal_rfpath(struct ieee80211_hw *hw, u32 rfpath); | ||
34 | void rtl92c_phy_set_io(struct ieee80211_hw *hw); | ||
35 | bool _rtl92cu_phy_config_mac_with_headerfile(struct ieee80211_hw *hw); | ||
36 | bool rtl92cu_phy_bb_config(struct ieee80211_hw *hw); | ||
37 | u32 rtl92cu_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
38 | enum radio_path rfpath, u32 regaddr, u32 bitmask); | ||
39 | void rtl92cu_phy_set_rf_reg(struct ieee80211_hw *hw, | ||
40 | enum radio_path rfpath, | ||
41 | u32 regaddr, u32 bitmask, u32 data); | ||
42 | bool rtl92cu_phy_mac_config(struct ieee80211_hw *hw); | ||
43 | bool _rtl92cu_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
44 | u8 configtype); | ||
45 | void _rtl92cu_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t); | ||
46 | bool _rtl92cu_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
47 | u8 configtype); | ||
48 | void rtl92cu_phy_set_bw_mode_callback(struct ieee80211_hw *hw); | ||
49 | bool rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
50 | enum rf_pwrstate rfpwr_state); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/reg.h b/drivers/net/wireless/rtlwifi/rtl8192cu/reg.h new file mode 100644 index 000000000000..7f1be614c998 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/reg.h | |||
@@ -0,0 +1,30 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../rtl8192ce/reg.h" | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/rf.c b/drivers/net/wireless/rtlwifi/rtl8192cu/rf.c new file mode 100644 index 000000000000..c7576ec4744e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/rf.c | |||
@@ -0,0 +1,493 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "reg.h" | ||
32 | #include "def.h" | ||
33 | #include "phy.h" | ||
34 | #include "rf.h" | ||
35 | #include "dm.h" | ||
36 | |||
37 | static bool _rtl92c_phy_rf6052_config_parafile(struct ieee80211_hw *hw); | ||
38 | |||
39 | void rtl92cu_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, u8 bandwidth) | ||
40 | { | ||
41 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
42 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
43 | |||
44 | switch (bandwidth) { | ||
45 | case HT_CHANNEL_WIDTH_20: | ||
46 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
47 | 0xfffff3ff) | 0x0400); | ||
48 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
49 | rtlphy->rfreg_chnlval[0]); | ||
50 | break; | ||
51 | case HT_CHANNEL_WIDTH_20_40: | ||
52 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
53 | 0xfffff3ff)); | ||
54 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
55 | rtlphy->rfreg_chnlval[0]); | ||
56 | break; | ||
57 | default: | ||
58 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
59 | ("unknown bandwidth: %#X\n", bandwidth)); | ||
60 | break; | ||
61 | } | ||
62 | } | ||
63 | |||
64 | void rtl92cu_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
65 | u8 *ppowerlevel) | ||
66 | { | ||
67 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
68 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
69 | struct rtl_hal *rtlhal = rtl_hal(rtlpriv); | ||
70 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
71 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
72 | u32 tx_agc[2] = { 0, 0 }, tmpval = 0; | ||
73 | bool turbo_scanoff = false; | ||
74 | u8 idx1, idx2; | ||
75 | u8 *ptr; | ||
76 | |||
77 | if (rtlhal->interface == INTF_PCI) { | ||
78 | if (rtlefuse->eeprom_regulatory != 0) | ||
79 | turbo_scanoff = true; | ||
80 | } else { | ||
81 | if ((rtlefuse->eeprom_regulatory != 0) || | ||
82 | (rtlefuse->external_pa)) | ||
83 | turbo_scanoff = true; | ||
84 | } | ||
85 | if (mac->act_scanning == true) { | ||
86 | tx_agc[RF90_PATH_A] = 0x3f3f3f3f; | ||
87 | tx_agc[RF90_PATH_B] = 0x3f3f3f3f; | ||
88 | if (turbo_scanoff) { | ||
89 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
90 | tx_agc[idx1] = ppowerlevel[idx1] | | ||
91 | (ppowerlevel[idx1] << 8) | | ||
92 | (ppowerlevel[idx1] << 16) | | ||
93 | (ppowerlevel[idx1] << 24); | ||
94 | if (rtlhal->interface == INTF_USB) { | ||
95 | if (tx_agc[idx1] > 0x20 && | ||
96 | rtlefuse->external_pa) | ||
97 | tx_agc[idx1] = 0x20; | ||
98 | } | ||
99 | } | ||
100 | } | ||
101 | } else { | ||
102 | if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
103 | TXHIGHPWRLEVEL_LEVEL1) { | ||
104 | tx_agc[RF90_PATH_A] = 0x10101010; | ||
105 | tx_agc[RF90_PATH_B] = 0x10101010; | ||
106 | } else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
107 | TXHIGHPWRLEVEL_LEVEL1) { | ||
108 | tx_agc[RF90_PATH_A] = 0x00000000; | ||
109 | tx_agc[RF90_PATH_B] = 0x00000000; | ||
110 | } else{ | ||
111 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
112 | tx_agc[idx1] = ppowerlevel[idx1] | | ||
113 | (ppowerlevel[idx1] << 8) | | ||
114 | (ppowerlevel[idx1] << 16) | | ||
115 | (ppowerlevel[idx1] << 24); | ||
116 | } | ||
117 | if (rtlefuse->eeprom_regulatory == 0) { | ||
118 | tmpval = (rtlphy->mcs_txpwrlevel_origoffset | ||
119 | [0][6]) + | ||
120 | (rtlphy->mcs_txpwrlevel_origoffset | ||
121 | [0][7] << 8); | ||
122 | tx_agc[RF90_PATH_A] += tmpval; | ||
123 | tmpval = (rtlphy->mcs_txpwrlevel_origoffset | ||
124 | [0][14]) + | ||
125 | (rtlphy->mcs_txpwrlevel_origoffset | ||
126 | [0][15] << 24); | ||
127 | tx_agc[RF90_PATH_B] += tmpval; | ||
128 | } | ||
129 | } | ||
130 | } | ||
131 | for (idx1 = RF90_PATH_A; idx1 <= RF90_PATH_B; idx1++) { | ||
132 | ptr = (u8 *) (&(tx_agc[idx1])); | ||
133 | for (idx2 = 0; idx2 < 4; idx2++) { | ||
134 | if (*ptr > RF6052_MAX_TX_PWR) | ||
135 | *ptr = RF6052_MAX_TX_PWR; | ||
136 | ptr++; | ||
137 | } | ||
138 | } | ||
139 | tmpval = tx_agc[RF90_PATH_A] & 0xff; | ||
140 | rtl_set_bbreg(hw, RTXAGC_A_CCK1_MCS32, MASKBYTE1, tmpval); | ||
141 | |||
142 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
143 | ("CCK PWR 1M (rf-A) = 0x%x (reg 0x%x)\n", tmpval, | ||
144 | RTXAGC_A_CCK1_MCS32)); | ||
145 | |||
146 | tmpval = tx_agc[RF90_PATH_A] >> 8; | ||
147 | if (mac->mode == WIRELESS_MODE_B) | ||
148 | tmpval = tmpval & 0xff00ffff; | ||
149 | rtl_set_bbreg(hw, RTXAGC_B_CCK11_A_CCK2_11, 0xffffff00, tmpval); | ||
150 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
151 | ("CCK PWR 2~11M (rf-A) = 0x%x (reg 0x%x)\n", tmpval, | ||
152 | RTXAGC_B_CCK11_A_CCK2_11)); | ||
153 | tmpval = tx_agc[RF90_PATH_B] >> 24; | ||
154 | rtl_set_bbreg(hw, RTXAGC_B_CCK11_A_CCK2_11, MASKBYTE0, tmpval); | ||
155 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
156 | ("CCK PWR 11M (rf-B) = 0x%x (reg 0x%x)\n", tmpval, | ||
157 | RTXAGC_B_CCK11_A_CCK2_11)); | ||
158 | tmpval = tx_agc[RF90_PATH_B] & 0x00ffffff; | ||
159 | rtl_set_bbreg(hw, RTXAGC_B_CCK1_55_MCS32, 0xffffff00, tmpval); | ||
160 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
161 | ("CCK PWR 1~5.5M (rf-B) = 0x%x (reg 0x%x)\n", tmpval, | ||
162 | RTXAGC_B_CCK1_55_MCS32)); | ||
163 | } | ||
164 | |||
165 | static void rtl92c_phy_get_power_base(struct ieee80211_hw *hw, | ||
166 | u8 *ppowerlevel, u8 channel, | ||
167 | u32 *ofdmbase, u32 *mcsbase) | ||
168 | { | ||
169 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
170 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
171 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
172 | u32 powerBase0, powerBase1; | ||
173 | u8 legacy_pwrdiff = 0, ht20_pwrdiff = 0; | ||
174 | u8 i, powerlevel[2]; | ||
175 | |||
176 | for (i = 0; i < 2; i++) { | ||
177 | powerlevel[i] = ppowerlevel[i]; | ||
178 | legacy_pwrdiff = rtlefuse->txpwr_legacyhtdiff[i][channel - 1]; | ||
179 | powerBase0 = powerlevel[i] + legacy_pwrdiff; | ||
180 | powerBase0 = (powerBase0 << 24) | (powerBase0 << 16) | | ||
181 | (powerBase0 << 8) | powerBase0; | ||
182 | *(ofdmbase + i) = powerBase0; | ||
183 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
184 | (" [OFDM power base index rf(%c) = 0x%x]\n", | ||
185 | ((i == 0) ? 'A' : 'B'), *(ofdmbase + i))); | ||
186 | } | ||
187 | for (i = 0; i < 2; i++) { | ||
188 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20) { | ||
189 | ht20_pwrdiff = rtlefuse->txpwr_ht20diff[i][channel - 1]; | ||
190 | powerlevel[i] += ht20_pwrdiff; | ||
191 | } | ||
192 | powerBase1 = powerlevel[i]; | ||
193 | powerBase1 = (powerBase1 << 24) | | ||
194 | (powerBase1 << 16) | (powerBase1 << 8) | powerBase1; | ||
195 | *(mcsbase + i) = powerBase1; | ||
196 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
197 | (" [MCS power base index rf(%c) = 0x%x]\n", | ||
198 | ((i == 0) ? 'A' : 'B'), *(mcsbase + i))); | ||
199 | } | ||
200 | } | ||
201 | |||
202 | static void _rtl92c_get_txpower_writeval_by_regulatory(struct ieee80211_hw *hw, | ||
203 | u8 channel, u8 index, | ||
204 | u32 *powerBase0, | ||
205 | u32 *powerBase1, | ||
206 | u32 *p_outwriteval) | ||
207 | { | ||
208 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
209 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
210 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
211 | u8 i, chnlgroup = 0, pwr_diff_limit[4]; | ||
212 | u32 writeVal, customer_limit, rf; | ||
213 | |||
214 | for (rf = 0; rf < 2; rf++) { | ||
215 | switch (rtlefuse->eeprom_regulatory) { | ||
216 | case 0: | ||
217 | chnlgroup = 0; | ||
218 | writeVal = rtlphy->mcs_txpwrlevel_origoffset | ||
219 | [chnlgroup][index + (rf ? 8 : 0)] | ||
220 | + ((index < 2) ? powerBase0[rf] : powerBase1[rf]); | ||
221 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
222 | ("RTK better performance,writeVal(%c) = 0x%x\n", | ||
223 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
224 | break; | ||
225 | case 1: | ||
226 | if (rtlphy->pwrgroup_cnt == 1) | ||
227 | chnlgroup = 0; | ||
228 | if (rtlphy->pwrgroup_cnt >= 3) { | ||
229 | if (channel <= 3) | ||
230 | chnlgroup = 0; | ||
231 | else if (channel >= 4 && channel <= 9) | ||
232 | chnlgroup = 1; | ||
233 | else if (channel > 9) | ||
234 | chnlgroup = 2; | ||
235 | if (rtlphy->current_chan_bw == | ||
236 | HT_CHANNEL_WIDTH_20) | ||
237 | chnlgroup++; | ||
238 | else | ||
239 | chnlgroup += 4; | ||
240 | } | ||
241 | writeVal = rtlphy->mcs_txpwrlevel_origoffset | ||
242 | [chnlgroup][index + | ||
243 | (rf ? 8 : 0)] + | ||
244 | ((index < 2) ? powerBase0[rf] : | ||
245 | powerBase1[rf]); | ||
246 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
247 | ("Realtek regulatory, 20MHz, " | ||
248 | "writeVal(%c) = 0x%x\n", | ||
249 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
250 | break; | ||
251 | case 2: | ||
252 | writeVal = ((index < 2) ? powerBase0[rf] : | ||
253 | powerBase1[rf]); | ||
254 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
255 | ("Better regulatory,writeVal(%c) = 0x%x\n", | ||
256 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
257 | break; | ||
258 | case 3: | ||
259 | chnlgroup = 0; | ||
260 | if (rtlphy->current_chan_bw == | ||
261 | HT_CHANNEL_WIDTH_20_40) { | ||
262 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
263 | ("customer's limit, 40MHzrf(%c) = " | ||
264 | "0x%x\n", ((rf == 0) ? 'A' : 'B'), | ||
265 | rtlefuse->pwrgroup_ht40[rf] | ||
266 | [channel - 1])); | ||
267 | } else { | ||
268 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
269 | ("customer's limit, 20MHz rf(%c) = " | ||
270 | "0x%x\n", ((rf == 0) ? 'A' : 'B'), | ||
271 | rtlefuse->pwrgroup_ht20[rf] | ||
272 | [channel - 1])); | ||
273 | } | ||
274 | for (i = 0; i < 4; i++) { | ||
275 | pwr_diff_limit[i] = | ||
276 | (u8) ((rtlphy->mcs_txpwrlevel_origoffset | ||
277 | [chnlgroup][index + (rf ? 8 : 0)] | ||
278 | & (0x7f << (i * 8))) >> (i * 8)); | ||
279 | if (rtlphy->current_chan_bw == | ||
280 | HT_CHANNEL_WIDTH_20_40) { | ||
281 | if (pwr_diff_limit[i] > | ||
282 | rtlefuse->pwrgroup_ht40[rf] | ||
283 | [channel - 1]) | ||
284 | pwr_diff_limit[i] = rtlefuse-> | ||
285 | pwrgroup_ht40[rf] | ||
286 | [channel - 1]; | ||
287 | } else { | ||
288 | if (pwr_diff_limit[i] > | ||
289 | rtlefuse->pwrgroup_ht20[rf] | ||
290 | [channel - 1]) | ||
291 | pwr_diff_limit[i] = | ||
292 | rtlefuse->pwrgroup_ht20[rf] | ||
293 | [channel - 1]; | ||
294 | } | ||
295 | } | ||
296 | customer_limit = (pwr_diff_limit[3] << 24) | | ||
297 | (pwr_diff_limit[2] << 16) | | ||
298 | (pwr_diff_limit[1] << 8) | (pwr_diff_limit[0]); | ||
299 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
300 | ("Customer's limit rf(%c) = 0x%x\n", | ||
301 | ((rf == 0) ? 'A' : 'B'), customer_limit)); | ||
302 | writeVal = customer_limit + ((index < 2) ? | ||
303 | powerBase0[rf] : powerBase1[rf]); | ||
304 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
305 | ("Customer, writeVal rf(%c)= 0x%x\n", | ||
306 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
307 | break; | ||
308 | default: | ||
309 | chnlgroup = 0; | ||
310 | writeVal = rtlphy->mcs_txpwrlevel_origoffset[chnlgroup] | ||
311 | [index + (rf ? 8 : 0)] + ((index < 2) ? | ||
312 | powerBase0[rf] : powerBase1[rf]); | ||
313 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, ("RTK better " | ||
314 | "performance, writeValrf(%c) = 0x%x\n", | ||
315 | ((rf == 0) ? 'A' : 'B'), writeVal)); | ||
316 | break; | ||
317 | } | ||
318 | if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
319 | TXHIGHPWRLEVEL_LEVEL1) | ||
320 | writeVal = 0x14141414; | ||
321 | else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
322 | TXHIGHPWRLEVEL_LEVEL2) | ||
323 | writeVal = 0x00000000; | ||
324 | if (rtlpriv->dm.dynamic_txhighpower_lvl == TXHIGHPWRLEVEL_BT1) | ||
325 | writeVal = writeVal - 0x06060606; | ||
326 | else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
327 | TXHIGHPWRLEVEL_BT2) | ||
328 | writeVal = writeVal; | ||
329 | *(p_outwriteval + rf) = writeVal; | ||
330 | } | ||
331 | } | ||
332 | |||
333 | static void _rtl92c_write_ofdm_power_reg(struct ieee80211_hw *hw, | ||
334 | u8 index, u32 *pValue) | ||
335 | { | ||
336 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
337 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
338 | u16 regoffset_a[6] = { | ||
339 | RTXAGC_A_RATE18_06, RTXAGC_A_RATE54_24, | ||
340 | RTXAGC_A_MCS03_MCS00, RTXAGC_A_MCS07_MCS04, | ||
341 | RTXAGC_A_MCS11_MCS08, RTXAGC_A_MCS15_MCS12 | ||
342 | }; | ||
343 | u16 regoffset_b[6] = { | ||
344 | RTXAGC_B_RATE18_06, RTXAGC_B_RATE54_24, | ||
345 | RTXAGC_B_MCS03_MCS00, RTXAGC_B_MCS07_MCS04, | ||
346 | RTXAGC_B_MCS11_MCS08, RTXAGC_B_MCS15_MCS12 | ||
347 | }; | ||
348 | u8 i, rf, pwr_val[4]; | ||
349 | u32 writeVal; | ||
350 | u16 regoffset; | ||
351 | |||
352 | for (rf = 0; rf < 2; rf++) { | ||
353 | writeVal = pValue[rf]; | ||
354 | for (i = 0; i < 4; i++) { | ||
355 | pwr_val[i] = (u8)((writeVal & (0x7f << (i * 8))) >> | ||
356 | (i * 8)); | ||
357 | if (pwr_val[i] > RF6052_MAX_TX_PWR) | ||
358 | pwr_val[i] = RF6052_MAX_TX_PWR; | ||
359 | } | ||
360 | writeVal = (pwr_val[3] << 24) | (pwr_val[2] << 16) | | ||
361 | (pwr_val[1] << 8) | pwr_val[0]; | ||
362 | if (rf == 0) | ||
363 | regoffset = regoffset_a[index]; | ||
364 | else | ||
365 | regoffset = regoffset_b[index]; | ||
366 | rtl_set_bbreg(hw, regoffset, MASKDWORD, writeVal); | ||
367 | RTPRINT(rtlpriv, FPHY, PHY_TXPWR, | ||
368 | ("Set 0x%x = %08x\n", regoffset, writeVal)); | ||
369 | if (((get_rf_type(rtlphy) == RF_2T2R) && | ||
370 | (regoffset == RTXAGC_A_MCS15_MCS12 || | ||
371 | regoffset == RTXAGC_B_MCS15_MCS12)) || | ||
372 | ((get_rf_type(rtlphy) != RF_2T2R) && | ||
373 | (regoffset == RTXAGC_A_MCS07_MCS04 || | ||
374 | regoffset == RTXAGC_B_MCS07_MCS04))) { | ||
375 | writeVal = pwr_val[3]; | ||
376 | if (regoffset == RTXAGC_A_MCS15_MCS12 || | ||
377 | regoffset == RTXAGC_A_MCS07_MCS04) | ||
378 | regoffset = 0xc90; | ||
379 | if (regoffset == RTXAGC_B_MCS15_MCS12 || | ||
380 | regoffset == RTXAGC_B_MCS07_MCS04) | ||
381 | regoffset = 0xc98; | ||
382 | for (i = 0; i < 3; i++) { | ||
383 | writeVal = (writeVal > 6) ? (writeVal - 6) : 0; | ||
384 | rtl_write_byte(rtlpriv, (u32)(regoffset + i), | ||
385 | (u8)writeVal); | ||
386 | } | ||
387 | } | ||
388 | } | ||
389 | } | ||
390 | |||
391 | void rtl92cu_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
392 | u8 *ppowerlevel, u8 channel) | ||
393 | { | ||
394 | u32 writeVal[2], powerBase0[2], powerBase1[2]; | ||
395 | u8 index = 0; | ||
396 | |||
397 | rtl92c_phy_get_power_base(hw, ppowerlevel, | ||
398 | channel, &powerBase0[0], &powerBase1[0]); | ||
399 | for (index = 0; index < 6; index++) { | ||
400 | _rtl92c_get_txpower_writeval_by_regulatory(hw, | ||
401 | channel, index, | ||
402 | &powerBase0[0], | ||
403 | &powerBase1[0], | ||
404 | &writeVal[0]); | ||
405 | _rtl92c_write_ofdm_power_reg(hw, index, &writeVal[0]); | ||
406 | } | ||
407 | } | ||
408 | |||
409 | bool rtl92cu_phy_rf6052_config(struct ieee80211_hw *hw) | ||
410 | { | ||
411 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
412 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
413 | bool rtstatus = true; | ||
414 | u8 b_reg_hwparafile = 1; | ||
415 | |||
416 | if (rtlphy->rf_type == RF_1T1R) | ||
417 | rtlphy->num_total_rfpath = 1; | ||
418 | else | ||
419 | rtlphy->num_total_rfpath = 2; | ||
420 | if (b_reg_hwparafile == 1) | ||
421 | rtstatus = _rtl92c_phy_rf6052_config_parafile(hw); | ||
422 | return rtstatus; | ||
423 | } | ||
424 | |||
425 | static bool _rtl92c_phy_rf6052_config_parafile(struct ieee80211_hw *hw) | ||
426 | { | ||
427 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
428 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
429 | u32 u4_regvalue = 0; | ||
430 | u8 rfpath; | ||
431 | bool rtstatus = true; | ||
432 | struct bb_reg_def *pphyreg; | ||
433 | |||
434 | for (rfpath = 0; rfpath < rtlphy->num_total_rfpath; rfpath++) { | ||
435 | pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
436 | switch (rfpath) { | ||
437 | case RF90_PATH_A: | ||
438 | case RF90_PATH_C: | ||
439 | u4_regvalue = rtl_get_bbreg(hw, pphyreg->rfintfs, | ||
440 | BRFSI_RFENV); | ||
441 | break; | ||
442 | case RF90_PATH_B: | ||
443 | case RF90_PATH_D: | ||
444 | u4_regvalue = rtl_get_bbreg(hw, pphyreg->rfintfs, | ||
445 | BRFSI_RFENV << 16); | ||
446 | break; | ||
447 | } | ||
448 | rtl_set_bbreg(hw, pphyreg->rfintfe, BRFSI_RFENV << 16, 0x1); | ||
449 | udelay(1); | ||
450 | rtl_set_bbreg(hw, pphyreg->rfintfo, BRFSI_RFENV, 0x1); | ||
451 | udelay(1); | ||
452 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, | ||
453 | B3WIREADDREAALENGTH, 0x0); | ||
454 | udelay(1); | ||
455 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, B3WIREDATALENGTH, 0x0); | ||
456 | udelay(1); | ||
457 | switch (rfpath) { | ||
458 | case RF90_PATH_A: | ||
459 | rtstatus = rtl92cu_phy_config_rf_with_headerfile(hw, | ||
460 | (enum radio_path) rfpath); | ||
461 | break; | ||
462 | case RF90_PATH_B: | ||
463 | rtstatus = rtl92cu_phy_config_rf_with_headerfile(hw, | ||
464 | (enum radio_path) rfpath); | ||
465 | break; | ||
466 | case RF90_PATH_C: | ||
467 | break; | ||
468 | case RF90_PATH_D: | ||
469 | break; | ||
470 | } | ||
471 | switch (rfpath) { | ||
472 | case RF90_PATH_A: | ||
473 | case RF90_PATH_C: | ||
474 | rtl_set_bbreg(hw, pphyreg->rfintfs, | ||
475 | BRFSI_RFENV, u4_regvalue); | ||
476 | break; | ||
477 | case RF90_PATH_B: | ||
478 | case RF90_PATH_D: | ||
479 | rtl_set_bbreg(hw, pphyreg->rfintfs, | ||
480 | BRFSI_RFENV << 16, u4_regvalue); | ||
481 | break; | ||
482 | } | ||
483 | if (rtstatus != true) { | ||
484 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
485 | ("Radio[%d] Fail!!", rfpath)); | ||
486 | goto phy_rf_cfg_fail; | ||
487 | } | ||
488 | } | ||
489 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, ("<---\n")); | ||
490 | return rtstatus; | ||
491 | phy_rf_cfg_fail: | ||
492 | return rtstatus; | ||
493 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/rf.h b/drivers/net/wireless/rtlwifi/rtl8192cu/rf.h new file mode 100644 index 000000000000..500a2094b6bb --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/rf.h | |||
@@ -0,0 +1,51 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CU_RF_H__ | ||
31 | #define __RTL92CU_RF_H__ | ||
32 | |||
33 | #define RF6052_MAX_TX_PWR 0x3F | ||
34 | #define RF6052_MAX_REG 0x3F | ||
35 | #define RF6052_MAX_PATH 2 | ||
36 | |||
37 | extern void rtl92cu_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, | ||
38 | u8 bandwidth); | ||
39 | extern void rtl92c_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
40 | u8 *ppowerlevel); | ||
41 | extern void rtl92c_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
42 | u8 *ppowerlevel, u8 channel); | ||
43 | bool rtl92cu_phy_rf6052_config(struct ieee80211_hw *hw); | ||
44 | bool rtl92cu_phy_config_rf_with_headerfile(struct ieee80211_hw *hw, | ||
45 | enum radio_path rfpath); | ||
46 | void rtl92cu_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
47 | u8 *ppowerlevel); | ||
48 | void rtl92cu_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
49 | u8 *ppowerlevel, u8 channel); | ||
50 | |||
51 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/sw.c b/drivers/net/wireless/rtlwifi/rtl8192cu/sw.c new file mode 100644 index 000000000000..942f7a3969a7 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/sw.c | |||
@@ -0,0 +1,357 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../core.h" | ||
32 | #include "../usb.h" | ||
33 | #include "../efuse.h" | ||
34 | #include "reg.h" | ||
35 | #include "def.h" | ||
36 | #include "phy.h" | ||
37 | #include "mac.h" | ||
38 | #include "dm.h" | ||
39 | #include "rf.h" | ||
40 | #include "sw.h" | ||
41 | #include "trx.h" | ||
42 | #include "led.h" | ||
43 | #include "hw.h" | ||
44 | #include <linux/vmalloc.h> | ||
45 | |||
46 | MODULE_AUTHOR("Georgia <georgia@realtek.com>"); | ||
47 | MODULE_AUTHOR("Ziv Huang <ziv_huang@realtek.com>"); | ||
48 | MODULE_AUTHOR("Larry Finger <Larry.Finger@lwfinger.net>"); | ||
49 | MODULE_LICENSE("GPL"); | ||
50 | MODULE_DESCRIPTION("Realtek 8192C/8188C 802.11n USB wireless"); | ||
51 | MODULE_FIRMWARE("rtlwifi/rtl8192cufw.bin"); | ||
52 | |||
53 | static int rtl92cu_init_sw_vars(struct ieee80211_hw *hw) | ||
54 | { | ||
55 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
56 | const struct firmware *firmware; | ||
57 | int err; | ||
58 | |||
59 | rtlpriv->dm.dm_initialgain_enable = 1; | ||
60 | rtlpriv->dm.dm_flag = 0; | ||
61 | rtlpriv->dm.disable_framebursting = 0; | ||
62 | rtlpriv->dm.thermalvalue = 0; | ||
63 | rtlpriv->rtlhal.pfirmware = vmalloc(0x4000); | ||
64 | if (!rtlpriv->rtlhal.pfirmware) { | ||
65 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
66 | ("Can't alloc buffer for fw.\n")); | ||
67 | return 1; | ||
68 | } | ||
69 | /* request fw */ | ||
70 | err = request_firmware(&firmware, rtlpriv->cfg->fw_name, | ||
71 | rtlpriv->io.dev); | ||
72 | if (err) { | ||
73 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
74 | ("Failed to request firmware!\n")); | ||
75 | return 1; | ||
76 | } | ||
77 | if (firmware->size > 0x4000) { | ||
78 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
79 | ("Firmware is too big!\n")); | ||
80 | release_firmware(firmware); | ||
81 | return 1; | ||
82 | } | ||
83 | memcpy(rtlpriv->rtlhal.pfirmware, firmware->data, firmware->size); | ||
84 | rtlpriv->rtlhal.fwsize = firmware->size; | ||
85 | release_firmware(firmware); | ||
86 | |||
87 | return 0; | ||
88 | } | ||
89 | |||
90 | static void rtl92cu_deinit_sw_vars(struct ieee80211_hw *hw) | ||
91 | { | ||
92 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
93 | |||
94 | if (rtlpriv->rtlhal.pfirmware) { | ||
95 | vfree(rtlpriv->rtlhal.pfirmware); | ||
96 | rtlpriv->rtlhal.pfirmware = NULL; | ||
97 | } | ||
98 | } | ||
99 | |||
100 | static struct rtl_hal_ops rtl8192cu_hal_ops = { | ||
101 | .init_sw_vars = rtl92cu_init_sw_vars, | ||
102 | .deinit_sw_vars = rtl92cu_deinit_sw_vars, | ||
103 | .read_chip_version = rtl92c_read_chip_version, | ||
104 | .read_eeprom_info = rtl92cu_read_eeprom_info, | ||
105 | .enable_interrupt = rtl92c_enable_interrupt, | ||
106 | .disable_interrupt = rtl92c_disable_interrupt, | ||
107 | .hw_init = rtl92cu_hw_init, | ||
108 | .hw_disable = rtl92cu_card_disable, | ||
109 | .set_network_type = rtl92cu_set_network_type, | ||
110 | .set_chk_bssid = rtl92cu_set_check_bssid, | ||
111 | .set_qos = rtl92c_set_qos, | ||
112 | .set_bcn_reg = rtl92cu_set_beacon_related_registers, | ||
113 | .set_bcn_intv = rtl92cu_set_beacon_interval, | ||
114 | .update_interrupt_mask = rtl92cu_update_interrupt_mask, | ||
115 | .get_hw_reg = rtl92cu_get_hw_reg, | ||
116 | .set_hw_reg = rtl92cu_set_hw_reg, | ||
117 | .update_rate_tbl = rtl92cu_update_hal_rate_table, | ||
118 | .update_rate_mask = rtl92cu_update_hal_rate_mask, | ||
119 | .fill_tx_desc = rtl92cu_tx_fill_desc, | ||
120 | .fill_fake_txdesc = rtl92cu_fill_fake_txdesc, | ||
121 | .fill_tx_cmddesc = rtl92cu_tx_fill_cmddesc, | ||
122 | .cmd_send_packet = rtl92cu_cmd_send_packet, | ||
123 | .query_rx_desc = rtl92cu_rx_query_desc, | ||
124 | .set_channel_access = rtl92cu_update_channel_access_setting, | ||
125 | .radio_onoff_checking = rtl92cu_gpio_radio_on_off_checking, | ||
126 | .set_bw_mode = rtl92c_phy_set_bw_mode, | ||
127 | .switch_channel = rtl92c_phy_sw_chnl, | ||
128 | .dm_watchdog = rtl92c_dm_watchdog, | ||
129 | .scan_operation_backup = rtl92c_phy_scan_operation_backup, | ||
130 | .set_rf_power_state = rtl92cu_phy_set_rf_power_state, | ||
131 | .led_control = rtl92cu_led_control, | ||
132 | .enable_hw_sec = rtl92cu_enable_hw_security_config, | ||
133 | .set_key = rtl92c_set_key, | ||
134 | .init_sw_leds = rtl92cu_init_sw_leds, | ||
135 | .deinit_sw_leds = rtl92cu_deinit_sw_leds, | ||
136 | .get_bbreg = rtl92c_phy_query_bb_reg, | ||
137 | .set_bbreg = rtl92c_phy_set_bb_reg, | ||
138 | .get_rfreg = rtl92cu_phy_query_rf_reg, | ||
139 | .set_rfreg = rtl92cu_phy_set_rf_reg, | ||
140 | .phy_rf6052_config = rtl92cu_phy_rf6052_config, | ||
141 | .phy_rf6052_set_cck_txpower = rtl92cu_phy_rf6052_set_cck_txpower, | ||
142 | .phy_rf6052_set_ofdm_txpower = rtl92cu_phy_rf6052_set_ofdm_txpower, | ||
143 | .config_bb_with_headerfile = _rtl92cu_phy_config_bb_with_headerfile, | ||
144 | .config_bb_with_pgheaderfile = _rtl92cu_phy_config_bb_with_pgheaderfile, | ||
145 | .phy_lc_calibrate = _rtl92cu_phy_lc_calibrate, | ||
146 | .phy_set_bw_mode_callback = rtl92cu_phy_set_bw_mode_callback, | ||
147 | .dm_dynamic_txpower = rtl92cu_dm_dynamic_txpower, | ||
148 | }; | ||
149 | |||
150 | static struct rtl_mod_params rtl92cu_mod_params = { | ||
151 | .sw_crypto = 0, | ||
152 | }; | ||
153 | |||
154 | static struct rtl_hal_usbint_cfg rtl92cu_interface_cfg = { | ||
155 | /* rx */ | ||
156 | .in_ep_num = RTL92C_USB_BULK_IN_NUM, | ||
157 | .rx_urb_num = RTL92C_NUM_RX_URBS, | ||
158 | .rx_max_size = RTL92C_SIZE_MAX_RX_BUFFER, | ||
159 | .usb_rx_hdl = rtl8192cu_rx_hdl, | ||
160 | .usb_rx_segregate_hdl = NULL, /* rtl8192c_rx_segregate_hdl; */ | ||
161 | /* tx */ | ||
162 | .usb_tx_cleanup = rtl8192c_tx_cleanup, | ||
163 | .usb_tx_post_hdl = rtl8192c_tx_post_hdl, | ||
164 | .usb_tx_aggregate_hdl = rtl8192c_tx_aggregate_hdl, | ||
165 | /* endpoint mapping */ | ||
166 | .usb_endpoint_mapping = rtl8192cu_endpoint_mapping, | ||
167 | .usb_mq_to_hwq = rtl8192cu_mq_to_hwq, | ||
168 | }; | ||
169 | |||
170 | static struct rtl_hal_cfg rtl92cu_hal_cfg = { | ||
171 | .name = "rtl92c_usb", | ||
172 | .fw_name = "rtlwifi/rtl8192cufw.bin", | ||
173 | .ops = &rtl8192cu_hal_ops, | ||
174 | .mod_params = &rtl92cu_mod_params, | ||
175 | .usb_interface_cfg = &rtl92cu_interface_cfg, | ||
176 | |||
177 | .maps[SYS_ISO_CTRL] = REG_SYS_ISO_CTRL, | ||
178 | .maps[SYS_FUNC_EN] = REG_SYS_FUNC_EN, | ||
179 | .maps[SYS_CLK] = REG_SYS_CLKR, | ||
180 | .maps[MAC_RCR_AM] = AM, | ||
181 | .maps[MAC_RCR_AB] = AB, | ||
182 | .maps[MAC_RCR_ACRC32] = ACRC32, | ||
183 | .maps[MAC_RCR_ACF] = ACF, | ||
184 | .maps[MAC_RCR_AAP] = AAP, | ||
185 | |||
186 | .maps[EFUSE_TEST] = REG_EFUSE_TEST, | ||
187 | .maps[EFUSE_CTRL] = REG_EFUSE_CTRL, | ||
188 | .maps[EFUSE_CLK] = 0, | ||
189 | .maps[EFUSE_CLK_CTRL] = REG_EFUSE_CTRL, | ||
190 | .maps[EFUSE_PWC_EV12V] = PWC_EV12V, | ||
191 | .maps[EFUSE_FEN_ELDR] = FEN_ELDR, | ||
192 | .maps[EFUSE_LOADER_CLK_EN] = LOADER_CLK_EN, | ||
193 | .maps[EFUSE_ANA8M] = EFUSE_ANA8M, | ||
194 | .maps[EFUSE_HWSET_MAX_SIZE] = HWSET_MAX_SIZE, | ||
195 | .maps[EFUSE_MAX_SECTION_MAP] = EFUSE_MAX_SECTION, | ||
196 | .maps[EFUSE_REAL_CONTENT_SIZE] = EFUSE_REAL_CONTENT_LEN, | ||
197 | |||
198 | .maps[RWCAM] = REG_CAMCMD, | ||
199 | .maps[WCAMI] = REG_CAMWRITE, | ||
200 | .maps[RCAMO] = REG_CAMREAD, | ||
201 | .maps[CAMDBG] = REG_CAMDBG, | ||
202 | .maps[SECR] = REG_SECCFG, | ||
203 | .maps[SEC_CAM_NONE] = CAM_NONE, | ||
204 | .maps[SEC_CAM_WEP40] = CAM_WEP40, | ||
205 | .maps[SEC_CAM_TKIP] = CAM_TKIP, | ||
206 | .maps[SEC_CAM_AES] = CAM_AES, | ||
207 | .maps[SEC_CAM_WEP104] = CAM_WEP104, | ||
208 | |||
209 | .maps[RTL_IMR_BCNDMAINT6] = IMR_BCNDMAINT6, | ||
210 | .maps[RTL_IMR_BCNDMAINT5] = IMR_BCNDMAINT5, | ||
211 | .maps[RTL_IMR_BCNDMAINT4] = IMR_BCNDMAINT4, | ||
212 | .maps[RTL_IMR_BCNDMAINT3] = IMR_BCNDMAINT3, | ||
213 | .maps[RTL_IMR_BCNDMAINT2] = IMR_BCNDMAINT2, | ||
214 | .maps[RTL_IMR_BCNDMAINT1] = IMR_BCNDMAINT1, | ||
215 | .maps[RTL_IMR_BCNDOK8] = IMR_BCNDOK8, | ||
216 | .maps[RTL_IMR_BCNDOK7] = IMR_BCNDOK7, | ||
217 | .maps[RTL_IMR_BCNDOK6] = IMR_BCNDOK6, | ||
218 | .maps[RTL_IMR_BCNDOK5] = IMR_BCNDOK5, | ||
219 | .maps[RTL_IMR_BCNDOK4] = IMR_BCNDOK4, | ||
220 | .maps[RTL_IMR_BCNDOK3] = IMR_BCNDOK3, | ||
221 | .maps[RTL_IMR_BCNDOK2] = IMR_BCNDOK2, | ||
222 | .maps[RTL_IMR_BCNDOK1] = IMR_BCNDOK1, | ||
223 | .maps[RTL_IMR_TIMEOUT2] = IMR_TIMEOUT2, | ||
224 | .maps[RTL_IMR_TIMEOUT1] = IMR_TIMEOUT1, | ||
225 | |||
226 | .maps[RTL_IMR_TXFOVW] = IMR_TXFOVW, | ||
227 | .maps[RTL_IMR_PSTIMEOUT] = IMR_PSTIMEOUT, | ||
228 | .maps[RTL_IMR_BcnInt] = IMR_BCNINT, | ||
229 | .maps[RTL_IMR_RXFOVW] = IMR_RXFOVW, | ||
230 | .maps[RTL_IMR_RDU] = IMR_RDU, | ||
231 | .maps[RTL_IMR_ATIMEND] = IMR_ATIMEND, | ||
232 | .maps[RTL_IMR_BDOK] = IMR_BDOK, | ||
233 | .maps[RTL_IMR_MGNTDOK] = IMR_MGNTDOK, | ||
234 | .maps[RTL_IMR_TBDER] = IMR_TBDER, | ||
235 | .maps[RTL_IMR_HIGHDOK] = IMR_HIGHDOK, | ||
236 | .maps[RTL_IMR_TBDOK] = IMR_TBDOK, | ||
237 | .maps[RTL_IMR_BKDOK] = IMR_BKDOK, | ||
238 | .maps[RTL_IMR_BEDOK] = IMR_BEDOK, | ||
239 | .maps[RTL_IMR_VIDOK] = IMR_VIDOK, | ||
240 | .maps[RTL_IMR_VODOK] = IMR_VODOK, | ||
241 | .maps[RTL_IMR_ROK] = IMR_ROK, | ||
242 | .maps[RTL_IBSS_INT_MASKS] = (IMR_BCNINT | IMR_TBDOK | IMR_TBDER), | ||
243 | |||
244 | .maps[RTL_RC_CCK_RATE1M] = DESC92C_RATE1M, | ||
245 | .maps[RTL_RC_CCK_RATE2M] = DESC92C_RATE2M, | ||
246 | .maps[RTL_RC_CCK_RATE5_5M] = DESC92C_RATE5_5M, | ||
247 | .maps[RTL_RC_CCK_RATE11M] = DESC92C_RATE11M, | ||
248 | .maps[RTL_RC_OFDM_RATE6M] = DESC92C_RATE6M, | ||
249 | .maps[RTL_RC_OFDM_RATE9M] = DESC92C_RATE9M, | ||
250 | .maps[RTL_RC_OFDM_RATE12M] = DESC92C_RATE12M, | ||
251 | .maps[RTL_RC_OFDM_RATE18M] = DESC92C_RATE18M, | ||
252 | .maps[RTL_RC_OFDM_RATE24M] = DESC92C_RATE24M, | ||
253 | .maps[RTL_RC_OFDM_RATE36M] = DESC92C_RATE36M, | ||
254 | .maps[RTL_RC_OFDM_RATE48M] = DESC92C_RATE48M, | ||
255 | .maps[RTL_RC_OFDM_RATE54M] = DESC92C_RATE54M, | ||
256 | .maps[RTL_RC_HT_RATEMCS7] = DESC92C_RATEMCS7, | ||
257 | .maps[RTL_RC_HT_RATEMCS15] = DESC92C_RATEMCS15, | ||
258 | }; | ||
259 | |||
260 | #define USB_VENDER_ID_REALTEK 0x0bda | ||
261 | |||
262 | /* 2010-10-19 DID_USB_V3.4 */ | ||
263 | static struct usb_device_id rtl8192c_usb_ids[] = { | ||
264 | |||
265 | /*=== Realtek demoboard ===*/ | ||
266 | /* Default ID */ | ||
267 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8191, rtl92cu_hal_cfg)}, | ||
268 | |||
269 | /****** 8188CU ********/ | ||
270 | /* 8188CE-VAU USB minCard */ | ||
271 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8170, rtl92cu_hal_cfg)}, | ||
272 | /* 8188cu 1*1 dongle */ | ||
273 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8176, rtl92cu_hal_cfg)}, | ||
274 | /* 8188cu 1*1 dongle, (b/g mode only) */ | ||
275 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8177, rtl92cu_hal_cfg)}, | ||
276 | /* 8188cu Slim Solo */ | ||
277 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817a, rtl92cu_hal_cfg)}, | ||
278 | /* 8188cu Slim Combo */ | ||
279 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817b, rtl92cu_hal_cfg)}, | ||
280 | /* 8188RU High-power USB Dongle */ | ||
281 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817d, rtl92cu_hal_cfg)}, | ||
282 | /* 8188CE-VAU USB minCard (b/g mode only) */ | ||
283 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817e, rtl92cu_hal_cfg)}, | ||
284 | /* 8188 Combo for BC4 */ | ||
285 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8754, rtl92cu_hal_cfg)}, | ||
286 | |||
287 | /****** 8192CU ********/ | ||
288 | /* 8191cu 1*2 */ | ||
289 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x8177, rtl92cu_hal_cfg)}, | ||
290 | /* 8192cu 2*2 */ | ||
291 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817b, rtl92cu_hal_cfg)}, | ||
292 | /* 8192CE-VAU USB minCard */ | ||
293 | {RTL_USB_DEVICE(USB_VENDER_ID_REALTEK, 0x817c, rtl92cu_hal_cfg)}, | ||
294 | |||
295 | /*=== Customer ID ===*/ | ||
296 | /****** 8188CU ********/ | ||
297 | {RTL_USB_DEVICE(0x050d, 0x1102, rtl92cu_hal_cfg)}, /*Belkin - Edimax*/ | ||
298 | {RTL_USB_DEVICE(0x06f8, 0xe033, rtl92cu_hal_cfg)}, /*Hercules - Edimax*/ | ||
299 | {RTL_USB_DEVICE(0x07b8, 0x8188, rtl92cu_hal_cfg)}, /*Abocom - Abocom*/ | ||
300 | {RTL_USB_DEVICE(0x07b8, 0x8189, rtl92cu_hal_cfg)}, /*Funai - Abocom*/ | ||
301 | {RTL_USB_DEVICE(0x0846, 0x9041, rtl92cu_hal_cfg)}, /*NetGear WNA1000M*/ | ||
302 | {RTL_USB_DEVICE(0x0Df6, 0x0052, rtl92cu_hal_cfg)}, /*Sitecom - Edimax*/ | ||
303 | {RTL_USB_DEVICE(0x0eb0, 0x9071, rtl92cu_hal_cfg)}, /*NO Brand - Etop*/ | ||
304 | /* HP - Lite-On ,8188CUS Slim Combo */ | ||
305 | {RTL_USB_DEVICE(0x103c, 0x1629, rtl92cu_hal_cfg)}, | ||
306 | {RTL_USB_DEVICE(0x2001, 0x3308, rtl92cu_hal_cfg)}, /*D-Link - Alpha*/ | ||
307 | {RTL_USB_DEVICE(0x2019, 0xab2a, rtl92cu_hal_cfg)}, /*Planex - Abocom*/ | ||
308 | {RTL_USB_DEVICE(0x2019, 0xed17, rtl92cu_hal_cfg)}, /*PCI - Edimax*/ | ||
309 | {RTL_USB_DEVICE(0x20f4, 0x648b, rtl92cu_hal_cfg)}, /*TRENDnet - Cameo*/ | ||
310 | {RTL_USB_DEVICE(0x7392, 0x7811, rtl92cu_hal_cfg)}, /*Edimax - Edimax*/ | ||
311 | {RTL_USB_DEVICE(0x3358, 0x13d3, rtl92cu_hal_cfg)}, /*Azwave 8188CE-VAU*/ | ||
312 | /* Russian customer -Azwave (8188CE-VAU b/g mode only) */ | ||
313 | {RTL_USB_DEVICE(0x3359, 0x13d3, rtl92cu_hal_cfg)}, | ||
314 | |||
315 | /****** 8192CU ********/ | ||
316 | {RTL_USB_DEVICE(0x0586, 0x341f, rtl92cu_hal_cfg)}, /*Zyxel -Abocom*/ | ||
317 | {RTL_USB_DEVICE(0x07aa, 0x0056, rtl92cu_hal_cfg)}, /*ATKK-Gemtek*/ | ||
318 | {RTL_USB_DEVICE(0x07b8, 0x8178, rtl92cu_hal_cfg)}, /*Funai -Abocom*/ | ||
319 | {RTL_USB_DEVICE(0x07b8, 0x8178, rtl92cu_hal_cfg)}, /*Abocom -Abocom*/ | ||
320 | {RTL_USB_DEVICE(0x2001, 0x3307, rtl92cu_hal_cfg)}, /*D-Link-Cameo*/ | ||
321 | {RTL_USB_DEVICE(0x2001, 0x3309, rtl92cu_hal_cfg)}, /*D-Link-Alpha*/ | ||
322 | {RTL_USB_DEVICE(0x2001, 0x330a, rtl92cu_hal_cfg)}, /*D-Link-Alpha*/ | ||
323 | {RTL_USB_DEVICE(0x2019, 0xab2b, rtl92cu_hal_cfg)}, /*Planex -Abocom*/ | ||
324 | {RTL_USB_DEVICE(0x7392, 0x7822, rtl92cu_hal_cfg)}, /*Edimax -Edimax*/ | ||
325 | {} | ||
326 | }; | ||
327 | |||
328 | MODULE_DEVICE_TABLE(usb, rtl8192c_usb_ids); | ||
329 | |||
330 | static struct usb_driver rtl8192cu_driver = { | ||
331 | .name = "rtl8192cu", | ||
332 | .probe = rtl_usb_probe, | ||
333 | .disconnect = rtl_usb_disconnect, | ||
334 | .id_table = rtl8192c_usb_ids, | ||
335 | |||
336 | #ifdef CONFIG_PM | ||
337 | /* .suspend = rtl_usb_suspend, */ | ||
338 | /* .resume = rtl_usb_resume, */ | ||
339 | /* .reset_resume = rtl8192c_resume, */ | ||
340 | #endif /* CONFIG_PM */ | ||
341 | #ifdef CONFIG_AUTOSUSPEND | ||
342 | .supports_autosuspend = 1, | ||
343 | #endif | ||
344 | }; | ||
345 | |||
346 | static int __init rtl8192cu_init(void) | ||
347 | { | ||
348 | return usb_register(&rtl8192cu_driver); | ||
349 | } | ||
350 | |||
351 | static void __exit rtl8192cu_exit(void) | ||
352 | { | ||
353 | usb_deregister(&rtl8192cu_driver); | ||
354 | } | ||
355 | |||
356 | module_init(rtl8192cu_init); | ||
357 | module_exit(rtl8192cu_exit); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/sw.h b/drivers/net/wireless/rtlwifi/rtl8192cu/sw.h new file mode 100644 index 000000000000..43b1177924ab --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/sw.h | |||
@@ -0,0 +1,53 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CU_SW_H__ | ||
31 | #define __RTL92CU_SW_H__ | ||
32 | |||
33 | #define EFUSE_MAX_SECTION 16 | ||
34 | |||
35 | void rtl92cu_phy_rf6052_set_cck_txpower(struct ieee80211_hw *hw, | ||
36 | u8 *powerlevel); | ||
37 | void rtl92cu_phy_rf6052_set_ofdm_txpower(struct ieee80211_hw *hw, | ||
38 | u8 *ppowerlevel, u8 channel); | ||
39 | bool _rtl92cu_phy_config_bb_with_headerfile(struct ieee80211_hw *hw, | ||
40 | u8 configtype); | ||
41 | bool _rtl92cu_phy_config_bb_with_pgheaderfile(struct ieee80211_hw *hw, | ||
42 | u8 configtype); | ||
43 | void _rtl92cu_phy_lc_calibrate(struct ieee80211_hw *hw, bool is2t); | ||
44 | void rtl92cu_phy_set_rf_reg(struct ieee80211_hw *hw, | ||
45 | enum radio_path rfpath, | ||
46 | u32 regaddr, u32 bitmask, u32 data); | ||
47 | bool rtl92cu_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
48 | enum rf_pwrstate rfpwr_state); | ||
49 | u32 rtl92cu_phy_query_rf_reg(struct ieee80211_hw *hw, | ||
50 | enum radio_path rfpath, u32 regaddr, u32 bitmask); | ||
51 | void rtl92cu_phy_set_bw_mode_callback(struct ieee80211_hw *hw); | ||
52 | |||
53 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/table.c b/drivers/net/wireless/rtlwifi/rtl8192cu/table.c new file mode 100644 index 000000000000..d57ef5e88a9e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/table.c | |||
@@ -0,0 +1,1888 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "table.h" | ||
31 | |||
32 | u32 RTL8192CUPHY_REG_2TARRAY[RTL8192CUPHY_REG_2TARRAY_LENGTH] = { | ||
33 | 0x024, 0x0011800f, | ||
34 | 0x028, 0x00ffdb83, | ||
35 | 0x800, 0x80040002, | ||
36 | 0x804, 0x00000003, | ||
37 | 0x808, 0x0000fc00, | ||
38 | 0x80c, 0x0000000a, | ||
39 | 0x810, 0x10005388, | ||
40 | 0x814, 0x020c3d10, | ||
41 | 0x818, 0x02200385, | ||
42 | 0x81c, 0x00000000, | ||
43 | 0x820, 0x01000100, | ||
44 | 0x824, 0x00390004, | ||
45 | 0x828, 0x01000100, | ||
46 | 0x82c, 0x00390004, | ||
47 | 0x830, 0x27272727, | ||
48 | 0x834, 0x27272727, | ||
49 | 0x838, 0x27272727, | ||
50 | 0x83c, 0x27272727, | ||
51 | 0x840, 0x00010000, | ||
52 | 0x844, 0x00010000, | ||
53 | 0x848, 0x27272727, | ||
54 | 0x84c, 0x27272727, | ||
55 | 0x850, 0x00000000, | ||
56 | 0x854, 0x00000000, | ||
57 | 0x858, 0x569a569a, | ||
58 | 0x85c, 0x0c1b25a4, | ||
59 | 0x860, 0x66e60230, | ||
60 | 0x864, 0x061f0130, | ||
61 | 0x868, 0x27272727, | ||
62 | 0x86c, 0x2b2b2b27, | ||
63 | 0x870, 0x07000700, | ||
64 | 0x874, 0x22184000, | ||
65 | 0x878, 0x08080808, | ||
66 | 0x87c, 0x00000000, | ||
67 | 0x880, 0xc0083070, | ||
68 | 0x884, 0x000004d5, | ||
69 | 0x888, 0x00000000, | ||
70 | 0x88c, 0xcc0000c0, | ||
71 | 0x890, 0x00000800, | ||
72 | 0x894, 0xfffffffe, | ||
73 | 0x898, 0x40302010, | ||
74 | 0x89c, 0x00706050, | ||
75 | 0x900, 0x00000000, | ||
76 | 0x904, 0x00000023, | ||
77 | 0x908, 0x00000000, | ||
78 | 0x90c, 0x81121313, | ||
79 | 0xa00, 0x00d047c8, | ||
80 | 0xa04, 0x80ff000c, | ||
81 | 0xa08, 0x8c838300, | ||
82 | 0xa0c, 0x2e68120f, | ||
83 | 0xa10, 0x9500bb78, | ||
84 | 0xa14, 0x11144028, | ||
85 | 0xa18, 0x00881117, | ||
86 | 0xa1c, 0x89140f00, | ||
87 | 0xa20, 0x1a1b0000, | ||
88 | 0xa24, 0x090e1317, | ||
89 | 0xa28, 0x00000204, | ||
90 | 0xa2c, 0x00d30000, | ||
91 | 0xa70, 0x101fbf00, | ||
92 | 0xa74, 0x00000007, | ||
93 | 0xc00, 0x48071d40, | ||
94 | 0xc04, 0x03a05633, | ||
95 | 0xc08, 0x000000e4, | ||
96 | 0xc0c, 0x6c6c6c6c, | ||
97 | 0xc10, 0x08800000, | ||
98 | 0xc14, 0x40000100, | ||
99 | 0xc18, 0x08800000, | ||
100 | 0xc1c, 0x40000100, | ||
101 | 0xc20, 0x00000000, | ||
102 | 0xc24, 0x00000000, | ||
103 | 0xc28, 0x00000000, | ||
104 | 0xc2c, 0x00000000, | ||
105 | 0xc30, 0x69e9ac44, | ||
106 | 0xc34, 0x469652cf, | ||
107 | 0xc38, 0x49795994, | ||
108 | 0xc3c, 0x0a97971c, | ||
109 | 0xc40, 0x1f7c403f, | ||
110 | 0xc44, 0x000100b7, | ||
111 | 0xc48, 0xec020107, | ||
112 | 0xc4c, 0x007f037f, | ||
113 | 0xc50, 0x6954341e, | ||
114 | 0xc54, 0x43bc0094, | ||
115 | 0xc58, 0x6954341e, | ||
116 | 0xc5c, 0x433c0094, | ||
117 | 0xc60, 0x00000000, | ||
118 | 0xc64, 0x5116848b, | ||
119 | 0xc68, 0x47c00bff, | ||
120 | 0xc6c, 0x00000036, | ||
121 | 0xc70, 0x2c7f000d, | ||
122 | 0xc74, 0x0186115b, | ||
123 | 0xc78, 0x0000001f, | ||
124 | 0xc7c, 0x00b99612, | ||
125 | 0xc80, 0x40000100, | ||
126 | 0xc84, 0x20f60000, | ||
127 | 0xc88, 0x40000100, | ||
128 | 0xc8c, 0x20200000, | ||
129 | 0xc90, 0x00121820, | ||
130 | 0xc94, 0x00000000, | ||
131 | 0xc98, 0x00121820, | ||
132 | 0xc9c, 0x00007f7f, | ||
133 | 0xca0, 0x00000000, | ||
134 | 0xca4, 0x00000080, | ||
135 | 0xca8, 0x00000000, | ||
136 | 0xcac, 0x00000000, | ||
137 | 0xcb0, 0x00000000, | ||
138 | 0xcb4, 0x00000000, | ||
139 | 0xcb8, 0x00000000, | ||
140 | 0xcbc, 0x28000000, | ||
141 | 0xcc0, 0x00000000, | ||
142 | 0xcc4, 0x00000000, | ||
143 | 0xcc8, 0x00000000, | ||
144 | 0xccc, 0x00000000, | ||
145 | 0xcd0, 0x00000000, | ||
146 | 0xcd4, 0x00000000, | ||
147 | 0xcd8, 0x64b22427, | ||
148 | 0xcdc, 0x00766932, | ||
149 | 0xce0, 0x00222222, | ||
150 | 0xce4, 0x00000000, | ||
151 | 0xce8, 0x37644302, | ||
152 | 0xcec, 0x2f97d40c, | ||
153 | 0xd00, 0x00080740, | ||
154 | 0xd04, 0x00020403, | ||
155 | 0xd08, 0x0000907f, | ||
156 | 0xd0c, 0x20010201, | ||
157 | 0xd10, 0xa0633333, | ||
158 | 0xd14, 0x3333bc43, | ||
159 | 0xd18, 0x7a8f5b6b, | ||
160 | 0xd2c, 0xcc979975, | ||
161 | 0xd30, 0x00000000, | ||
162 | 0xd34, 0x80608000, | ||
163 | 0xd38, 0x00000000, | ||
164 | 0xd3c, 0x00027293, | ||
165 | 0xd40, 0x00000000, | ||
166 | 0xd44, 0x00000000, | ||
167 | 0xd48, 0x00000000, | ||
168 | 0xd4c, 0x00000000, | ||
169 | 0xd50, 0x6437140a, | ||
170 | 0xd54, 0x00000000, | ||
171 | 0xd58, 0x00000000, | ||
172 | 0xd5c, 0x30032064, | ||
173 | 0xd60, 0x4653de68, | ||
174 | 0xd64, 0x04518a3c, | ||
175 | 0xd68, 0x00002101, | ||
176 | 0xd6c, 0x2a201c16, | ||
177 | 0xd70, 0x1812362e, | ||
178 | 0xd74, 0x322c2220, | ||
179 | 0xd78, 0x000e3c24, | ||
180 | 0xe00, 0x2a2a2a2a, | ||
181 | 0xe04, 0x2a2a2a2a, | ||
182 | 0xe08, 0x03902a2a, | ||
183 | 0xe10, 0x2a2a2a2a, | ||
184 | 0xe14, 0x2a2a2a2a, | ||
185 | 0xe18, 0x2a2a2a2a, | ||
186 | 0xe1c, 0x2a2a2a2a, | ||
187 | 0xe28, 0x00000000, | ||
188 | 0xe30, 0x1000dc1f, | ||
189 | 0xe34, 0x10008c1f, | ||
190 | 0xe38, 0x02140102, | ||
191 | 0xe3c, 0x681604c2, | ||
192 | 0xe40, 0x01007c00, | ||
193 | 0xe44, 0x01004800, | ||
194 | 0xe48, 0xfb000000, | ||
195 | 0xe4c, 0x000028d1, | ||
196 | 0xe50, 0x1000dc1f, | ||
197 | 0xe54, 0x10008c1f, | ||
198 | 0xe58, 0x02140102, | ||
199 | 0xe5c, 0x28160d05, | ||
200 | 0xe60, 0x00000010, | ||
201 | 0xe68, 0x001b25a4, | ||
202 | 0xe6c, 0x63db25a4, | ||
203 | 0xe70, 0x63db25a4, | ||
204 | 0xe74, 0x0c1b25a4, | ||
205 | 0xe78, 0x0c1b25a4, | ||
206 | 0xe7c, 0x0c1b25a4, | ||
207 | 0xe80, 0x0c1b25a4, | ||
208 | 0xe84, 0x63db25a4, | ||
209 | 0xe88, 0x0c1b25a4, | ||
210 | 0xe8c, 0x63db25a4, | ||
211 | 0xed0, 0x63db25a4, | ||
212 | 0xed4, 0x63db25a4, | ||
213 | 0xed8, 0x63db25a4, | ||
214 | 0xedc, 0x001b25a4, | ||
215 | 0xee0, 0x001b25a4, | ||
216 | 0xeec, 0x6fdb25a4, | ||
217 | 0xf14, 0x00000003, | ||
218 | 0xf4c, 0x00000000, | ||
219 | 0xf00, 0x00000300, | ||
220 | }; | ||
221 | |||
222 | u32 RTL8192CUPHY_REG_1TARRAY[RTL8192CUPHY_REG_1TARRAY_LENGTH] = { | ||
223 | 0x024, 0x0011800f, | ||
224 | 0x028, 0x00ffdb83, | ||
225 | 0x800, 0x80040000, | ||
226 | 0x804, 0x00000001, | ||
227 | 0x808, 0x0000fc00, | ||
228 | 0x80c, 0x0000000a, | ||
229 | 0x810, 0x10005388, | ||
230 | 0x814, 0x020c3d10, | ||
231 | 0x818, 0x02200385, | ||
232 | 0x81c, 0x00000000, | ||
233 | 0x820, 0x01000100, | ||
234 | 0x824, 0x00390004, | ||
235 | 0x828, 0x00000000, | ||
236 | 0x82c, 0x00000000, | ||
237 | 0x830, 0x00000000, | ||
238 | 0x834, 0x00000000, | ||
239 | 0x838, 0x00000000, | ||
240 | 0x83c, 0x00000000, | ||
241 | 0x840, 0x00010000, | ||
242 | 0x844, 0x00000000, | ||
243 | 0x848, 0x00000000, | ||
244 | 0x84c, 0x00000000, | ||
245 | 0x850, 0x00000000, | ||
246 | 0x854, 0x00000000, | ||
247 | 0x858, 0x569a569a, | ||
248 | 0x85c, 0x001b25a4, | ||
249 | 0x860, 0x66e60230, | ||
250 | 0x864, 0x061f0130, | ||
251 | 0x868, 0x00000000, | ||
252 | 0x86c, 0x32323200, | ||
253 | 0x870, 0x07000700, | ||
254 | 0x874, 0x22004000, | ||
255 | 0x878, 0x00000808, | ||
256 | 0x87c, 0x00000000, | ||
257 | 0x880, 0xc0083070, | ||
258 | 0x884, 0x000004d5, | ||
259 | 0x888, 0x00000000, | ||
260 | 0x88c, 0xccc000c0, | ||
261 | 0x890, 0x00000800, | ||
262 | 0x894, 0xfffffffe, | ||
263 | 0x898, 0x40302010, | ||
264 | 0x89c, 0x00706050, | ||
265 | 0x900, 0x00000000, | ||
266 | 0x904, 0x00000023, | ||
267 | 0x908, 0x00000000, | ||
268 | 0x90c, 0x81121111, | ||
269 | 0xa00, 0x00d047c8, | ||
270 | 0xa04, 0x80ff000c, | ||
271 | 0xa08, 0x8c838300, | ||
272 | 0xa0c, 0x2e68120f, | ||
273 | 0xa10, 0x9500bb78, | ||
274 | 0xa14, 0x11144028, | ||
275 | 0xa18, 0x00881117, | ||
276 | 0xa1c, 0x89140f00, | ||
277 | 0xa20, 0x1a1b0000, | ||
278 | 0xa24, 0x090e1317, | ||
279 | 0xa28, 0x00000204, | ||
280 | 0xa2c, 0x00d30000, | ||
281 | 0xa70, 0x101fbf00, | ||
282 | 0xa74, 0x00000007, | ||
283 | 0xc00, 0x48071d40, | ||
284 | 0xc04, 0x03a05611, | ||
285 | 0xc08, 0x000000e4, | ||
286 | 0xc0c, 0x6c6c6c6c, | ||
287 | 0xc10, 0x08800000, | ||
288 | 0xc14, 0x40000100, | ||
289 | 0xc18, 0x08800000, | ||
290 | 0xc1c, 0x40000100, | ||
291 | 0xc20, 0x00000000, | ||
292 | 0xc24, 0x00000000, | ||
293 | 0xc28, 0x00000000, | ||
294 | 0xc2c, 0x00000000, | ||
295 | 0xc30, 0x69e9ac44, | ||
296 | 0xc34, 0x469652cf, | ||
297 | 0xc38, 0x49795994, | ||
298 | 0xc3c, 0x0a97971c, | ||
299 | 0xc40, 0x1f7c403f, | ||
300 | 0xc44, 0x000100b7, | ||
301 | 0xc48, 0xec020107, | ||
302 | 0xc4c, 0x007f037f, | ||
303 | 0xc50, 0x6954341e, | ||
304 | 0xc54, 0x43bc0094, | ||
305 | 0xc58, 0x6954341e, | ||
306 | 0xc5c, 0x433c0094, | ||
307 | 0xc60, 0x00000000, | ||
308 | 0xc64, 0x5116848b, | ||
309 | 0xc68, 0x47c00bff, | ||
310 | 0xc6c, 0x00000036, | ||
311 | 0xc70, 0x2c7f000d, | ||
312 | 0xc74, 0x018610db, | ||
313 | 0xc78, 0x0000001f, | ||
314 | 0xc7c, 0x00b91612, | ||
315 | 0xc80, 0x40000100, | ||
316 | 0xc84, 0x20f60000, | ||
317 | 0xc88, 0x40000100, | ||
318 | 0xc8c, 0x20200000, | ||
319 | 0xc90, 0x00121820, | ||
320 | 0xc94, 0x00000000, | ||
321 | 0xc98, 0x00121820, | ||
322 | 0xc9c, 0x00007f7f, | ||
323 | 0xca0, 0x00000000, | ||
324 | 0xca4, 0x00000080, | ||
325 | 0xca8, 0x00000000, | ||
326 | 0xcac, 0x00000000, | ||
327 | 0xcb0, 0x00000000, | ||
328 | 0xcb4, 0x00000000, | ||
329 | 0xcb8, 0x00000000, | ||
330 | 0xcbc, 0x28000000, | ||
331 | 0xcc0, 0x00000000, | ||
332 | 0xcc4, 0x00000000, | ||
333 | 0xcc8, 0x00000000, | ||
334 | 0xccc, 0x00000000, | ||
335 | 0xcd0, 0x00000000, | ||
336 | 0xcd4, 0x00000000, | ||
337 | 0xcd8, 0x64b22427, | ||
338 | 0xcdc, 0x00766932, | ||
339 | 0xce0, 0x00222222, | ||
340 | 0xce4, 0x00000000, | ||
341 | 0xce8, 0x37644302, | ||
342 | 0xcec, 0x2f97d40c, | ||
343 | 0xd00, 0x00080740, | ||
344 | 0xd04, 0x00020401, | ||
345 | 0xd08, 0x0000907f, | ||
346 | 0xd0c, 0x20010201, | ||
347 | 0xd10, 0xa0633333, | ||
348 | 0xd14, 0x3333bc43, | ||
349 | 0xd18, 0x7a8f5b6b, | ||
350 | 0xd2c, 0xcc979975, | ||
351 | 0xd30, 0x00000000, | ||
352 | 0xd34, 0x80608000, | ||
353 | 0xd38, 0x00000000, | ||
354 | 0xd3c, 0x00027293, | ||
355 | 0xd40, 0x00000000, | ||
356 | 0xd44, 0x00000000, | ||
357 | 0xd48, 0x00000000, | ||
358 | 0xd4c, 0x00000000, | ||
359 | 0xd50, 0x6437140a, | ||
360 | 0xd54, 0x00000000, | ||
361 | 0xd58, 0x00000000, | ||
362 | 0xd5c, 0x30032064, | ||
363 | 0xd60, 0x4653de68, | ||
364 | 0xd64, 0x04518a3c, | ||
365 | 0xd68, 0x00002101, | ||
366 | 0xd6c, 0x2a201c16, | ||
367 | 0xd70, 0x1812362e, | ||
368 | 0xd74, 0x322c2220, | ||
369 | 0xd78, 0x000e3c24, | ||
370 | 0xe00, 0x2a2a2a2a, | ||
371 | 0xe04, 0x2a2a2a2a, | ||
372 | 0xe08, 0x03902a2a, | ||
373 | 0xe10, 0x2a2a2a2a, | ||
374 | 0xe14, 0x2a2a2a2a, | ||
375 | 0xe18, 0x2a2a2a2a, | ||
376 | 0xe1c, 0x2a2a2a2a, | ||
377 | 0xe28, 0x00000000, | ||
378 | 0xe30, 0x1000dc1f, | ||
379 | 0xe34, 0x10008c1f, | ||
380 | 0xe38, 0x02140102, | ||
381 | 0xe3c, 0x681604c2, | ||
382 | 0xe40, 0x01007c00, | ||
383 | 0xe44, 0x01004800, | ||
384 | 0xe48, 0xfb000000, | ||
385 | 0xe4c, 0x000028d1, | ||
386 | 0xe50, 0x1000dc1f, | ||
387 | 0xe54, 0x10008c1f, | ||
388 | 0xe58, 0x02140102, | ||
389 | 0xe5c, 0x28160d05, | ||
390 | 0xe60, 0x00000008, | ||
391 | 0xe68, 0x001b25a4, | ||
392 | 0xe6c, 0x631b25a0, | ||
393 | 0xe70, 0x631b25a0, | ||
394 | 0xe74, 0x081b25a0, | ||
395 | 0xe78, 0x081b25a0, | ||
396 | 0xe7c, 0x081b25a0, | ||
397 | 0xe80, 0x081b25a0, | ||
398 | 0xe84, 0x631b25a0, | ||
399 | 0xe88, 0x081b25a0, | ||
400 | 0xe8c, 0x631b25a0, | ||
401 | 0xed0, 0x631b25a0, | ||
402 | 0xed4, 0x631b25a0, | ||
403 | 0xed8, 0x631b25a0, | ||
404 | 0xedc, 0x001b25a0, | ||
405 | 0xee0, 0x001b25a0, | ||
406 | 0xeec, 0x6b1b25a0, | ||
407 | 0xf14, 0x00000003, | ||
408 | 0xf4c, 0x00000000, | ||
409 | 0xf00, 0x00000300, | ||
410 | }; | ||
411 | |||
412 | u32 RTL8192CUPHY_REG_ARRAY_PG[RTL8192CUPHY_REG_ARRAY_PGLENGTH] = { | ||
413 | 0xe00, 0xffffffff, 0x07090c0c, | ||
414 | 0xe04, 0xffffffff, 0x01020405, | ||
415 | 0xe08, 0x0000ff00, 0x00000000, | ||
416 | 0x86c, 0xffffff00, 0x00000000, | ||
417 | 0xe10, 0xffffffff, 0x0b0c0c0e, | ||
418 | 0xe14, 0xffffffff, 0x01030506, | ||
419 | 0xe18, 0xffffffff, 0x0b0c0d0e, | ||
420 | 0xe1c, 0xffffffff, 0x01030509, | ||
421 | 0x830, 0xffffffff, 0x07090c0c, | ||
422 | 0x834, 0xffffffff, 0x01020405, | ||
423 | 0x838, 0xffffff00, 0x00000000, | ||
424 | 0x86c, 0x000000ff, 0x00000000, | ||
425 | 0x83c, 0xffffffff, 0x0b0c0d0e, | ||
426 | 0x848, 0xffffffff, 0x01030509, | ||
427 | 0x84c, 0xffffffff, 0x0b0c0d0e, | ||
428 | 0x868, 0xffffffff, 0x01030509, | ||
429 | 0xe00, 0xffffffff, 0x00000000, | ||
430 | 0xe04, 0xffffffff, 0x00000000, | ||
431 | 0xe08, 0x0000ff00, 0x00000000, | ||
432 | 0x86c, 0xffffff00, 0x00000000, | ||
433 | 0xe10, 0xffffffff, 0x00000000, | ||
434 | 0xe14, 0xffffffff, 0x00000000, | ||
435 | 0xe18, 0xffffffff, 0x00000000, | ||
436 | 0xe1c, 0xffffffff, 0x00000000, | ||
437 | 0x830, 0xffffffff, 0x00000000, | ||
438 | 0x834, 0xffffffff, 0x00000000, | ||
439 | 0x838, 0xffffff00, 0x00000000, | ||
440 | 0x86c, 0x000000ff, 0x00000000, | ||
441 | 0x83c, 0xffffffff, 0x00000000, | ||
442 | 0x848, 0xffffffff, 0x00000000, | ||
443 | 0x84c, 0xffffffff, 0x00000000, | ||
444 | 0x868, 0xffffffff, 0x00000000, | ||
445 | 0xe00, 0xffffffff, 0x04040404, | ||
446 | 0xe04, 0xffffffff, 0x00020204, | ||
447 | 0xe08, 0x0000ff00, 0x00000000, | ||
448 | 0x86c, 0xffffff00, 0x00000000, | ||
449 | 0xe10, 0xffffffff, 0x06060606, | ||
450 | 0xe14, 0xffffffff, 0x00020406, | ||
451 | 0xe18, 0xffffffff, 0x00000000, | ||
452 | 0xe1c, 0xffffffff, 0x00000000, | ||
453 | 0x830, 0xffffffff, 0x04040404, | ||
454 | 0x834, 0xffffffff, 0x00020204, | ||
455 | 0x838, 0xffffff00, 0x00000000, | ||
456 | 0x86c, 0x000000ff, 0x00000000, | ||
457 | 0x83c, 0xffffffff, 0x06060606, | ||
458 | 0x848, 0xffffffff, 0x00020406, | ||
459 | 0x84c, 0xffffffff, 0x00000000, | ||
460 | 0x868, 0xffffffff, 0x00000000, | ||
461 | 0xe00, 0xffffffff, 0x00000000, | ||
462 | 0xe04, 0xffffffff, 0x00000000, | ||
463 | 0xe08, 0x0000ff00, 0x00000000, | ||
464 | 0x86c, 0xffffff00, 0x00000000, | ||
465 | 0xe10, 0xffffffff, 0x00000000, | ||
466 | 0xe14, 0xffffffff, 0x00000000, | ||
467 | 0xe18, 0xffffffff, 0x00000000, | ||
468 | 0xe1c, 0xffffffff, 0x00000000, | ||
469 | 0x830, 0xffffffff, 0x00000000, | ||
470 | 0x834, 0xffffffff, 0x00000000, | ||
471 | 0x838, 0xffffff00, 0x00000000, | ||
472 | 0x86c, 0x000000ff, 0x00000000, | ||
473 | 0x83c, 0xffffffff, 0x00000000, | ||
474 | 0x848, 0xffffffff, 0x00000000, | ||
475 | 0x84c, 0xffffffff, 0x00000000, | ||
476 | 0x868, 0xffffffff, 0x00000000, | ||
477 | 0xe00, 0xffffffff, 0x00000000, | ||
478 | 0xe04, 0xffffffff, 0x00000000, | ||
479 | 0xe08, 0x0000ff00, 0x00000000, | ||
480 | 0x86c, 0xffffff00, 0x00000000, | ||
481 | 0xe10, 0xffffffff, 0x00000000, | ||
482 | 0xe14, 0xffffffff, 0x00000000, | ||
483 | 0xe18, 0xffffffff, 0x00000000, | ||
484 | 0xe1c, 0xffffffff, 0x00000000, | ||
485 | 0x830, 0xffffffff, 0x00000000, | ||
486 | 0x834, 0xffffffff, 0x00000000, | ||
487 | 0x838, 0xffffff00, 0x00000000, | ||
488 | 0x86c, 0x000000ff, 0x00000000, | ||
489 | 0x83c, 0xffffffff, 0x00000000, | ||
490 | 0x848, 0xffffffff, 0x00000000, | ||
491 | 0x84c, 0xffffffff, 0x00000000, | ||
492 | 0x868, 0xffffffff, 0x00000000, | ||
493 | 0xe00, 0xffffffff, 0x04040404, | ||
494 | 0xe04, 0xffffffff, 0x00020204, | ||
495 | 0xe08, 0x0000ff00, 0x00000000, | ||
496 | 0x86c, 0xffffff00, 0x00000000, | ||
497 | 0xe10, 0xffffffff, 0x00000000, | ||
498 | 0xe14, 0xffffffff, 0x00000000, | ||
499 | 0xe18, 0xffffffff, 0x00000000, | ||
500 | 0xe1c, 0xffffffff, 0x00000000, | ||
501 | 0x830, 0xffffffff, 0x04040404, | ||
502 | 0x834, 0xffffffff, 0x00020204, | ||
503 | 0x838, 0xffffff00, 0x00000000, | ||
504 | 0x86c, 0x000000ff, 0x00000000, | ||
505 | 0x83c, 0xffffffff, 0x00000000, | ||
506 | 0x848, 0xffffffff, 0x00000000, | ||
507 | 0x84c, 0xffffffff, 0x00000000, | ||
508 | 0x868, 0xffffffff, 0x00000000, | ||
509 | 0xe00, 0xffffffff, 0x00000000, | ||
510 | 0xe04, 0xffffffff, 0x00000000, | ||
511 | 0xe08, 0x0000ff00, 0x00000000, | ||
512 | 0x86c, 0xffffff00, 0x00000000, | ||
513 | 0xe10, 0xffffffff, 0x00000000, | ||
514 | 0xe14, 0xffffffff, 0x00000000, | ||
515 | 0xe18, 0xffffffff, 0x00000000, | ||
516 | 0xe1c, 0xffffffff, 0x00000000, | ||
517 | 0x830, 0xffffffff, 0x00000000, | ||
518 | 0x834, 0xffffffff, 0x00000000, | ||
519 | 0x838, 0xffffff00, 0x00000000, | ||
520 | 0x86c, 0x000000ff, 0x00000000, | ||
521 | 0x83c, 0xffffffff, 0x00000000, | ||
522 | 0x848, 0xffffffff, 0x00000000, | ||
523 | 0x84c, 0xffffffff, 0x00000000, | ||
524 | 0x868, 0xffffffff, 0x00000000, | ||
525 | }; | ||
526 | |||
527 | u32 RTL8192CURADIOA_2TARRAY[RTL8192CURADIOA_2TARRAYLENGTH] = { | ||
528 | 0x000, 0x00030159, | ||
529 | 0x001, 0x00031284, | ||
530 | 0x002, 0x00098000, | ||
531 | 0x003, 0x00018c63, | ||
532 | 0x004, 0x000210e7, | ||
533 | 0x009, 0x0002044f, | ||
534 | 0x00a, 0x0001adb1, | ||
535 | 0x00b, 0x00054867, | ||
536 | 0x00c, 0x0008992e, | ||
537 | 0x00d, 0x0000e52c, | ||
538 | 0x00e, 0x00039ce7, | ||
539 | 0x00f, 0x00000451, | ||
540 | 0x019, 0x00000000, | ||
541 | 0x01a, 0x00010255, | ||
542 | 0x01b, 0x00060a00, | ||
543 | 0x01c, 0x000fc378, | ||
544 | 0x01d, 0x000a1250, | ||
545 | 0x01e, 0x0004445f, | ||
546 | 0x01f, 0x00080001, | ||
547 | 0x020, 0x0000b614, | ||
548 | 0x021, 0x0006c000, | ||
549 | 0x022, 0x00000000, | ||
550 | 0x023, 0x00001558, | ||
551 | 0x024, 0x00000060, | ||
552 | 0x025, 0x00000483, | ||
553 | 0x026, 0x0004f000, | ||
554 | 0x027, 0x000ec7d9, | ||
555 | 0x028, 0x000577c0, | ||
556 | 0x029, 0x00004783, | ||
557 | 0x02a, 0x00000001, | ||
558 | 0x02b, 0x00021334, | ||
559 | 0x02a, 0x00000000, | ||
560 | 0x02b, 0x00000054, | ||
561 | 0x02a, 0x00000001, | ||
562 | 0x02b, 0x00000808, | ||
563 | 0x02b, 0x00053333, | ||
564 | 0x02c, 0x0000000c, | ||
565 | 0x02a, 0x00000002, | ||
566 | 0x02b, 0x00000808, | ||
567 | 0x02b, 0x0005b333, | ||
568 | 0x02c, 0x0000000d, | ||
569 | 0x02a, 0x00000003, | ||
570 | 0x02b, 0x00000808, | ||
571 | 0x02b, 0x00063333, | ||
572 | 0x02c, 0x0000000d, | ||
573 | 0x02a, 0x00000004, | ||
574 | 0x02b, 0x00000808, | ||
575 | 0x02b, 0x0006b333, | ||
576 | 0x02c, 0x0000000d, | ||
577 | 0x02a, 0x00000005, | ||
578 | 0x02b, 0x00000808, | ||
579 | 0x02b, 0x00073333, | ||
580 | 0x02c, 0x0000000d, | ||
581 | 0x02a, 0x00000006, | ||
582 | 0x02b, 0x00000709, | ||
583 | 0x02b, 0x0005b333, | ||
584 | 0x02c, 0x0000000d, | ||
585 | 0x02a, 0x00000007, | ||
586 | 0x02b, 0x00000709, | ||
587 | 0x02b, 0x00063333, | ||
588 | 0x02c, 0x0000000d, | ||
589 | 0x02a, 0x00000008, | ||
590 | 0x02b, 0x0000060a, | ||
591 | 0x02b, 0x0004b333, | ||
592 | 0x02c, 0x0000000d, | ||
593 | 0x02a, 0x00000009, | ||
594 | 0x02b, 0x0000060a, | ||
595 | 0x02b, 0x00053333, | ||
596 | 0x02c, 0x0000000d, | ||
597 | 0x02a, 0x0000000a, | ||
598 | 0x02b, 0x0000060a, | ||
599 | 0x02b, 0x0005b333, | ||
600 | 0x02c, 0x0000000d, | ||
601 | 0x02a, 0x0000000b, | ||
602 | 0x02b, 0x0000060a, | ||
603 | 0x02b, 0x00063333, | ||
604 | 0x02c, 0x0000000d, | ||
605 | 0x02a, 0x0000000c, | ||
606 | 0x02b, 0x0000060a, | ||
607 | 0x02b, 0x0006b333, | ||
608 | 0x02c, 0x0000000d, | ||
609 | 0x02a, 0x0000000d, | ||
610 | 0x02b, 0x0000060a, | ||
611 | 0x02b, 0x00073333, | ||
612 | 0x02c, 0x0000000d, | ||
613 | 0x02a, 0x0000000e, | ||
614 | 0x02b, 0x0000050b, | ||
615 | 0x02b, 0x00066666, | ||
616 | 0x02c, 0x0000001a, | ||
617 | 0x02a, 0x000e0000, | ||
618 | 0x010, 0x0004000f, | ||
619 | 0x011, 0x000e31fc, | ||
620 | 0x010, 0x0006000f, | ||
621 | 0x011, 0x000ff9f8, | ||
622 | 0x010, 0x0002000f, | ||
623 | 0x011, 0x000203f9, | ||
624 | 0x010, 0x0003000f, | ||
625 | 0x011, 0x000ff500, | ||
626 | 0x010, 0x00000000, | ||
627 | 0x011, 0x00000000, | ||
628 | 0x010, 0x0008000f, | ||
629 | 0x011, 0x0003f100, | ||
630 | 0x010, 0x0009000f, | ||
631 | 0x011, 0x00023100, | ||
632 | 0x012, 0x00032000, | ||
633 | 0x012, 0x00071000, | ||
634 | 0x012, 0x000b0000, | ||
635 | 0x012, 0x000fc000, | ||
636 | 0x013, 0x000287af, | ||
637 | 0x013, 0x000244b7, | ||
638 | 0x013, 0x000204ab, | ||
639 | 0x013, 0x0001c49f, | ||
640 | 0x013, 0x00018493, | ||
641 | 0x013, 0x00014297, | ||
642 | 0x013, 0x00010295, | ||
643 | 0x013, 0x0000c298, | ||
644 | 0x013, 0x0000819c, | ||
645 | 0x013, 0x000040a8, | ||
646 | 0x013, 0x0000001c, | ||
647 | 0x014, 0x0001944c, | ||
648 | 0x014, 0x00059444, | ||
649 | 0x014, 0x0009944c, | ||
650 | 0x014, 0x000d9444, | ||
651 | 0x015, 0x0000f424, | ||
652 | 0x015, 0x0004f424, | ||
653 | 0x015, 0x0008f424, | ||
654 | 0x015, 0x000cf424, | ||
655 | 0x016, 0x000e0330, | ||
656 | 0x016, 0x000a0330, | ||
657 | 0x016, 0x00060330, | ||
658 | 0x016, 0x00020330, | ||
659 | 0x000, 0x00010159, | ||
660 | 0x018, 0x0000f401, | ||
661 | 0x0fe, 0x00000000, | ||
662 | 0x0fe, 0x00000000, | ||
663 | 0x01f, 0x00080003, | ||
664 | 0x0fe, 0x00000000, | ||
665 | 0x0fe, 0x00000000, | ||
666 | 0x01e, 0x00044457, | ||
667 | 0x01f, 0x00080000, | ||
668 | 0x000, 0x00030159, | ||
669 | }; | ||
670 | |||
671 | u32 RTL8192CU_RADIOB_2TARRAY[RTL8192CURADIOB_2TARRAYLENGTH] = { | ||
672 | 0x000, 0x00030159, | ||
673 | 0x001, 0x00031284, | ||
674 | 0x002, 0x00098000, | ||
675 | 0x003, 0x00018c63, | ||
676 | 0x004, 0x000210e7, | ||
677 | 0x009, 0x0002044f, | ||
678 | 0x00a, 0x0001adb1, | ||
679 | 0x00b, 0x00054867, | ||
680 | 0x00c, 0x0008992e, | ||
681 | 0x00d, 0x0000e52c, | ||
682 | 0x00e, 0x00039ce7, | ||
683 | 0x00f, 0x00000451, | ||
684 | 0x012, 0x00032000, | ||
685 | 0x012, 0x00071000, | ||
686 | 0x012, 0x000b0000, | ||
687 | 0x012, 0x000fc000, | ||
688 | 0x013, 0x000287af, | ||
689 | 0x013, 0x000244b7, | ||
690 | 0x013, 0x000204ab, | ||
691 | 0x013, 0x0001c49f, | ||
692 | 0x013, 0x00018493, | ||
693 | 0x013, 0x00014297, | ||
694 | 0x013, 0x00010295, | ||
695 | 0x013, 0x0000c298, | ||
696 | 0x013, 0x0000819c, | ||
697 | 0x013, 0x000040a8, | ||
698 | 0x013, 0x0000001c, | ||
699 | 0x014, 0x0001944c, | ||
700 | 0x014, 0x00059444, | ||
701 | 0x014, 0x0009944c, | ||
702 | 0x014, 0x000d9444, | ||
703 | 0x015, 0x0000f424, | ||
704 | 0x015, 0x0004f424, | ||
705 | 0x015, 0x0008f424, | ||
706 | 0x015, 0x000cf424, | ||
707 | 0x016, 0x000e0330, | ||
708 | 0x016, 0x000a0330, | ||
709 | 0x016, 0x00060330, | ||
710 | 0x016, 0x00020330, | ||
711 | }; | ||
712 | |||
713 | u32 RTL8192CU_RADIOA_1TARRAY[RTL8192CURADIOA_1TARRAYLENGTH] = { | ||
714 | 0x000, 0x00030159, | ||
715 | 0x001, 0x00031284, | ||
716 | 0x002, 0x00098000, | ||
717 | 0x003, 0x00018c63, | ||
718 | 0x004, 0x000210e7, | ||
719 | 0x009, 0x0002044f, | ||
720 | 0x00a, 0x0001adb1, | ||
721 | 0x00b, 0x00054867, | ||
722 | 0x00c, 0x0008992e, | ||
723 | 0x00d, 0x0000e52c, | ||
724 | 0x00e, 0x00039ce7, | ||
725 | 0x00f, 0x00000451, | ||
726 | 0x019, 0x00000000, | ||
727 | 0x01a, 0x00010255, | ||
728 | 0x01b, 0x00060a00, | ||
729 | 0x01c, 0x000fc378, | ||
730 | 0x01d, 0x000a1250, | ||
731 | 0x01e, 0x0004445f, | ||
732 | 0x01f, 0x00080001, | ||
733 | 0x020, 0x0000b614, | ||
734 | 0x021, 0x0006c000, | ||
735 | 0x022, 0x00000000, | ||
736 | 0x023, 0x00001558, | ||
737 | 0x024, 0x00000060, | ||
738 | 0x025, 0x00000483, | ||
739 | 0x026, 0x0004f000, | ||
740 | 0x027, 0x000ec7d9, | ||
741 | 0x028, 0x000577c0, | ||
742 | 0x029, 0x00004783, | ||
743 | 0x02a, 0x00000001, | ||
744 | 0x02b, 0x00021334, | ||
745 | 0x02a, 0x00000000, | ||
746 | 0x02b, 0x00000054, | ||
747 | 0x02a, 0x00000001, | ||
748 | 0x02b, 0x00000808, | ||
749 | 0x02b, 0x00053333, | ||
750 | 0x02c, 0x0000000c, | ||
751 | 0x02a, 0x00000002, | ||
752 | 0x02b, 0x00000808, | ||
753 | 0x02b, 0x0005b333, | ||
754 | 0x02c, 0x0000000d, | ||
755 | 0x02a, 0x00000003, | ||
756 | 0x02b, 0x00000808, | ||
757 | 0x02b, 0x00063333, | ||
758 | 0x02c, 0x0000000d, | ||
759 | 0x02a, 0x00000004, | ||
760 | 0x02b, 0x00000808, | ||
761 | 0x02b, 0x0006b333, | ||
762 | 0x02c, 0x0000000d, | ||
763 | 0x02a, 0x00000005, | ||
764 | 0x02b, 0x00000808, | ||
765 | 0x02b, 0x00073333, | ||
766 | 0x02c, 0x0000000d, | ||
767 | 0x02a, 0x00000006, | ||
768 | 0x02b, 0x00000709, | ||
769 | 0x02b, 0x0005b333, | ||
770 | 0x02c, 0x0000000d, | ||
771 | 0x02a, 0x00000007, | ||
772 | 0x02b, 0x00000709, | ||
773 | 0x02b, 0x00063333, | ||
774 | 0x02c, 0x0000000d, | ||
775 | 0x02a, 0x00000008, | ||
776 | 0x02b, 0x0000060a, | ||
777 | 0x02b, 0x0004b333, | ||
778 | 0x02c, 0x0000000d, | ||
779 | 0x02a, 0x00000009, | ||
780 | 0x02b, 0x0000060a, | ||
781 | 0x02b, 0x00053333, | ||
782 | 0x02c, 0x0000000d, | ||
783 | 0x02a, 0x0000000a, | ||
784 | 0x02b, 0x0000060a, | ||
785 | 0x02b, 0x0005b333, | ||
786 | 0x02c, 0x0000000d, | ||
787 | 0x02a, 0x0000000b, | ||
788 | 0x02b, 0x0000060a, | ||
789 | 0x02b, 0x00063333, | ||
790 | 0x02c, 0x0000000d, | ||
791 | 0x02a, 0x0000000c, | ||
792 | 0x02b, 0x0000060a, | ||
793 | 0x02b, 0x0006b333, | ||
794 | 0x02c, 0x0000000d, | ||
795 | 0x02a, 0x0000000d, | ||
796 | 0x02b, 0x0000060a, | ||
797 | 0x02b, 0x00073333, | ||
798 | 0x02c, 0x0000000d, | ||
799 | 0x02a, 0x0000000e, | ||
800 | 0x02b, 0x0000050b, | ||
801 | 0x02b, 0x00066666, | ||
802 | 0x02c, 0x0000001a, | ||
803 | 0x02a, 0x000e0000, | ||
804 | 0x010, 0x0004000f, | ||
805 | 0x011, 0x000e31fc, | ||
806 | 0x010, 0x0006000f, | ||
807 | 0x011, 0x000ff9f8, | ||
808 | 0x010, 0x0002000f, | ||
809 | 0x011, 0x000203f9, | ||
810 | 0x010, 0x0003000f, | ||
811 | 0x011, 0x000ff500, | ||
812 | 0x010, 0x00000000, | ||
813 | 0x011, 0x00000000, | ||
814 | 0x010, 0x0008000f, | ||
815 | 0x011, 0x0003f100, | ||
816 | 0x010, 0x0009000f, | ||
817 | 0x011, 0x00023100, | ||
818 | 0x012, 0x00032000, | ||
819 | 0x012, 0x00071000, | ||
820 | 0x012, 0x000b0000, | ||
821 | 0x012, 0x000fc000, | ||
822 | 0x013, 0x000287b3, | ||
823 | 0x013, 0x000244b7, | ||
824 | 0x013, 0x000204ab, | ||
825 | 0x013, 0x0001c49f, | ||
826 | 0x013, 0x00018493, | ||
827 | 0x013, 0x0001429b, | ||
828 | 0x013, 0x00010299, | ||
829 | 0x013, 0x0000c29c, | ||
830 | 0x013, 0x000081a0, | ||
831 | 0x013, 0x000040ac, | ||
832 | 0x013, 0x00000020, | ||
833 | 0x014, 0x0001944c, | ||
834 | 0x014, 0x00059444, | ||
835 | 0x014, 0x0009944c, | ||
836 | 0x014, 0x000d9444, | ||
837 | 0x015, 0x0000f405, | ||
838 | 0x015, 0x0004f405, | ||
839 | 0x015, 0x0008f405, | ||
840 | 0x015, 0x000cf405, | ||
841 | 0x016, 0x000e0330, | ||
842 | 0x016, 0x000a0330, | ||
843 | 0x016, 0x00060330, | ||
844 | 0x016, 0x00020330, | ||
845 | 0x000, 0x00010159, | ||
846 | 0x018, 0x0000f401, | ||
847 | 0x0fe, 0x00000000, | ||
848 | 0x0fe, 0x00000000, | ||
849 | 0x01f, 0x00080003, | ||
850 | 0x0fe, 0x00000000, | ||
851 | 0x0fe, 0x00000000, | ||
852 | 0x01e, 0x00044457, | ||
853 | 0x01f, 0x00080000, | ||
854 | 0x000, 0x00030159, | ||
855 | }; | ||
856 | |||
857 | u32 RTL8192CU_RADIOB_1TARRAY[RTL8192CURADIOB_1TARRAYLENGTH] = { | ||
858 | 0x0, | ||
859 | }; | ||
860 | |||
861 | u32 RTL8192CUMAC_2T_ARRAY[RTL8192CUMAC_2T_ARRAYLENGTH] = { | ||
862 | 0x420, 0x00000080, | ||
863 | 0x423, 0x00000000, | ||
864 | 0x430, 0x00000000, | ||
865 | 0x431, 0x00000000, | ||
866 | 0x432, 0x00000000, | ||
867 | 0x433, 0x00000001, | ||
868 | 0x434, 0x00000004, | ||
869 | 0x435, 0x00000005, | ||
870 | 0x436, 0x00000006, | ||
871 | 0x437, 0x00000007, | ||
872 | 0x438, 0x00000000, | ||
873 | 0x439, 0x00000000, | ||
874 | 0x43a, 0x00000000, | ||
875 | 0x43b, 0x00000001, | ||
876 | 0x43c, 0x00000004, | ||
877 | 0x43d, 0x00000005, | ||
878 | 0x43e, 0x00000006, | ||
879 | 0x43f, 0x00000007, | ||
880 | 0x440, 0x0000005d, | ||
881 | 0x441, 0x00000001, | ||
882 | 0x442, 0x00000000, | ||
883 | 0x444, 0x00000015, | ||
884 | 0x445, 0x000000f0, | ||
885 | 0x446, 0x0000000f, | ||
886 | 0x447, 0x00000000, | ||
887 | 0x458, 0x00000041, | ||
888 | 0x459, 0x000000a8, | ||
889 | 0x45a, 0x00000072, | ||
890 | 0x45b, 0x000000b9, | ||
891 | 0x460, 0x00000066, | ||
892 | 0x461, 0x00000066, | ||
893 | 0x462, 0x00000008, | ||
894 | 0x463, 0x00000003, | ||
895 | 0x4c8, 0x000000ff, | ||
896 | 0x4c9, 0x00000008, | ||
897 | 0x4cc, 0x000000ff, | ||
898 | 0x4cd, 0x000000ff, | ||
899 | 0x4ce, 0x00000001, | ||
900 | 0x500, 0x00000026, | ||
901 | 0x501, 0x000000a2, | ||
902 | 0x502, 0x0000002f, | ||
903 | 0x503, 0x00000000, | ||
904 | 0x504, 0x00000028, | ||
905 | 0x505, 0x000000a3, | ||
906 | 0x506, 0x0000005e, | ||
907 | 0x507, 0x00000000, | ||
908 | 0x508, 0x0000002b, | ||
909 | 0x509, 0x000000a4, | ||
910 | 0x50a, 0x0000005e, | ||
911 | 0x50b, 0x00000000, | ||
912 | 0x50c, 0x0000004f, | ||
913 | 0x50d, 0x000000a4, | ||
914 | 0x50e, 0x00000000, | ||
915 | 0x50f, 0x00000000, | ||
916 | 0x512, 0x0000001c, | ||
917 | 0x514, 0x0000000a, | ||
918 | 0x515, 0x00000010, | ||
919 | 0x516, 0x0000000a, | ||
920 | 0x517, 0x00000010, | ||
921 | 0x51a, 0x00000016, | ||
922 | 0x524, 0x0000000f, | ||
923 | 0x525, 0x0000004f, | ||
924 | 0x546, 0x00000040, | ||
925 | 0x547, 0x00000000, | ||
926 | 0x550, 0x00000010, | ||
927 | 0x551, 0x00000010, | ||
928 | 0x559, 0x00000002, | ||
929 | 0x55a, 0x00000002, | ||
930 | 0x55d, 0x000000ff, | ||
931 | 0x605, 0x00000030, | ||
932 | 0x608, 0x0000000e, | ||
933 | 0x609, 0x0000002a, | ||
934 | 0x652, 0x00000020, | ||
935 | 0x63c, 0x0000000a, | ||
936 | 0x63d, 0x0000000e, | ||
937 | 0x63e, 0x0000000a, | ||
938 | 0x63f, 0x0000000e, | ||
939 | 0x66e, 0x00000005, | ||
940 | 0x700, 0x00000021, | ||
941 | 0x701, 0x00000043, | ||
942 | 0x702, 0x00000065, | ||
943 | 0x703, 0x00000087, | ||
944 | 0x708, 0x00000021, | ||
945 | 0x709, 0x00000043, | ||
946 | 0x70a, 0x00000065, | ||
947 | 0x70b, 0x00000087, | ||
948 | }; | ||
949 | |||
950 | u32 RTL8192CUAGCTAB_2TARRAY[RTL8192CUAGCTAB_2TARRAYLENGTH] = { | ||
951 | 0xc78, 0x7b000001, | ||
952 | 0xc78, 0x7b010001, | ||
953 | 0xc78, 0x7b020001, | ||
954 | 0xc78, 0x7b030001, | ||
955 | 0xc78, 0x7b040001, | ||
956 | 0xc78, 0x7b050001, | ||
957 | 0xc78, 0x7a060001, | ||
958 | 0xc78, 0x79070001, | ||
959 | 0xc78, 0x78080001, | ||
960 | 0xc78, 0x77090001, | ||
961 | 0xc78, 0x760a0001, | ||
962 | 0xc78, 0x750b0001, | ||
963 | 0xc78, 0x740c0001, | ||
964 | 0xc78, 0x730d0001, | ||
965 | 0xc78, 0x720e0001, | ||
966 | 0xc78, 0x710f0001, | ||
967 | 0xc78, 0x70100001, | ||
968 | 0xc78, 0x6f110001, | ||
969 | 0xc78, 0x6e120001, | ||
970 | 0xc78, 0x6d130001, | ||
971 | 0xc78, 0x6c140001, | ||
972 | 0xc78, 0x6b150001, | ||
973 | 0xc78, 0x6a160001, | ||
974 | 0xc78, 0x69170001, | ||
975 | 0xc78, 0x68180001, | ||
976 | 0xc78, 0x67190001, | ||
977 | 0xc78, 0x661a0001, | ||
978 | 0xc78, 0x651b0001, | ||
979 | 0xc78, 0x641c0001, | ||
980 | 0xc78, 0x631d0001, | ||
981 | 0xc78, 0x621e0001, | ||
982 | 0xc78, 0x611f0001, | ||
983 | 0xc78, 0x60200001, | ||
984 | 0xc78, 0x49210001, | ||
985 | 0xc78, 0x48220001, | ||
986 | 0xc78, 0x47230001, | ||
987 | 0xc78, 0x46240001, | ||
988 | 0xc78, 0x45250001, | ||
989 | 0xc78, 0x44260001, | ||
990 | 0xc78, 0x43270001, | ||
991 | 0xc78, 0x42280001, | ||
992 | 0xc78, 0x41290001, | ||
993 | 0xc78, 0x402a0001, | ||
994 | 0xc78, 0x262b0001, | ||
995 | 0xc78, 0x252c0001, | ||
996 | 0xc78, 0x242d0001, | ||
997 | 0xc78, 0x232e0001, | ||
998 | 0xc78, 0x222f0001, | ||
999 | 0xc78, 0x21300001, | ||
1000 | 0xc78, 0x20310001, | ||
1001 | 0xc78, 0x06320001, | ||
1002 | 0xc78, 0x05330001, | ||
1003 | 0xc78, 0x04340001, | ||
1004 | 0xc78, 0x03350001, | ||
1005 | 0xc78, 0x02360001, | ||
1006 | 0xc78, 0x01370001, | ||
1007 | 0xc78, 0x00380001, | ||
1008 | 0xc78, 0x00390001, | ||
1009 | 0xc78, 0x003a0001, | ||
1010 | 0xc78, 0x003b0001, | ||
1011 | 0xc78, 0x003c0001, | ||
1012 | 0xc78, 0x003d0001, | ||
1013 | 0xc78, 0x003e0001, | ||
1014 | 0xc78, 0x003f0001, | ||
1015 | 0xc78, 0x7b400001, | ||
1016 | 0xc78, 0x7b410001, | ||
1017 | 0xc78, 0x7b420001, | ||
1018 | 0xc78, 0x7b430001, | ||
1019 | 0xc78, 0x7b440001, | ||
1020 | 0xc78, 0x7b450001, | ||
1021 | 0xc78, 0x7a460001, | ||
1022 | 0xc78, 0x79470001, | ||
1023 | 0xc78, 0x78480001, | ||
1024 | 0xc78, 0x77490001, | ||
1025 | 0xc78, 0x764a0001, | ||
1026 | 0xc78, 0x754b0001, | ||
1027 | 0xc78, 0x744c0001, | ||
1028 | 0xc78, 0x734d0001, | ||
1029 | 0xc78, 0x724e0001, | ||
1030 | 0xc78, 0x714f0001, | ||
1031 | 0xc78, 0x70500001, | ||
1032 | 0xc78, 0x6f510001, | ||
1033 | 0xc78, 0x6e520001, | ||
1034 | 0xc78, 0x6d530001, | ||
1035 | 0xc78, 0x6c540001, | ||
1036 | 0xc78, 0x6b550001, | ||
1037 | 0xc78, 0x6a560001, | ||
1038 | 0xc78, 0x69570001, | ||
1039 | 0xc78, 0x68580001, | ||
1040 | 0xc78, 0x67590001, | ||
1041 | 0xc78, 0x665a0001, | ||
1042 | 0xc78, 0x655b0001, | ||
1043 | 0xc78, 0x645c0001, | ||
1044 | 0xc78, 0x635d0001, | ||
1045 | 0xc78, 0x625e0001, | ||
1046 | 0xc78, 0x615f0001, | ||
1047 | 0xc78, 0x60600001, | ||
1048 | 0xc78, 0x49610001, | ||
1049 | 0xc78, 0x48620001, | ||
1050 | 0xc78, 0x47630001, | ||
1051 | 0xc78, 0x46640001, | ||
1052 | 0xc78, 0x45650001, | ||
1053 | 0xc78, 0x44660001, | ||
1054 | 0xc78, 0x43670001, | ||
1055 | 0xc78, 0x42680001, | ||
1056 | 0xc78, 0x41690001, | ||
1057 | 0xc78, 0x406a0001, | ||
1058 | 0xc78, 0x266b0001, | ||
1059 | 0xc78, 0x256c0001, | ||
1060 | 0xc78, 0x246d0001, | ||
1061 | 0xc78, 0x236e0001, | ||
1062 | 0xc78, 0x226f0001, | ||
1063 | 0xc78, 0x21700001, | ||
1064 | 0xc78, 0x20710001, | ||
1065 | 0xc78, 0x06720001, | ||
1066 | 0xc78, 0x05730001, | ||
1067 | 0xc78, 0x04740001, | ||
1068 | 0xc78, 0x03750001, | ||
1069 | 0xc78, 0x02760001, | ||
1070 | 0xc78, 0x01770001, | ||
1071 | 0xc78, 0x00780001, | ||
1072 | 0xc78, 0x00790001, | ||
1073 | 0xc78, 0x007a0001, | ||
1074 | 0xc78, 0x007b0001, | ||
1075 | 0xc78, 0x007c0001, | ||
1076 | 0xc78, 0x007d0001, | ||
1077 | 0xc78, 0x007e0001, | ||
1078 | 0xc78, 0x007f0001, | ||
1079 | 0xc78, 0x3800001e, | ||
1080 | 0xc78, 0x3801001e, | ||
1081 | 0xc78, 0x3802001e, | ||
1082 | 0xc78, 0x3803001e, | ||
1083 | 0xc78, 0x3804001e, | ||
1084 | 0xc78, 0x3805001e, | ||
1085 | 0xc78, 0x3806001e, | ||
1086 | 0xc78, 0x3807001e, | ||
1087 | 0xc78, 0x3808001e, | ||
1088 | 0xc78, 0x3c09001e, | ||
1089 | 0xc78, 0x3e0a001e, | ||
1090 | 0xc78, 0x400b001e, | ||
1091 | 0xc78, 0x440c001e, | ||
1092 | 0xc78, 0x480d001e, | ||
1093 | 0xc78, 0x4c0e001e, | ||
1094 | 0xc78, 0x500f001e, | ||
1095 | 0xc78, 0x5210001e, | ||
1096 | 0xc78, 0x5611001e, | ||
1097 | 0xc78, 0x5a12001e, | ||
1098 | 0xc78, 0x5e13001e, | ||
1099 | 0xc78, 0x6014001e, | ||
1100 | 0xc78, 0x6015001e, | ||
1101 | 0xc78, 0x6016001e, | ||
1102 | 0xc78, 0x6217001e, | ||
1103 | 0xc78, 0x6218001e, | ||
1104 | 0xc78, 0x6219001e, | ||
1105 | 0xc78, 0x621a001e, | ||
1106 | 0xc78, 0x621b001e, | ||
1107 | 0xc78, 0x621c001e, | ||
1108 | 0xc78, 0x621d001e, | ||
1109 | 0xc78, 0x621e001e, | ||
1110 | 0xc78, 0x621f001e, | ||
1111 | }; | ||
1112 | |||
1113 | u32 RTL8192CUAGCTAB_1TARRAY[RTL8192CUAGCTAB_1TARRAYLENGTH] = { | ||
1114 | 0xc78, 0x7b000001, | ||
1115 | 0xc78, 0x7b010001, | ||
1116 | 0xc78, 0x7b020001, | ||
1117 | 0xc78, 0x7b030001, | ||
1118 | 0xc78, 0x7b040001, | ||
1119 | 0xc78, 0x7b050001, | ||
1120 | 0xc78, 0x7a060001, | ||
1121 | 0xc78, 0x79070001, | ||
1122 | 0xc78, 0x78080001, | ||
1123 | 0xc78, 0x77090001, | ||
1124 | 0xc78, 0x760a0001, | ||
1125 | 0xc78, 0x750b0001, | ||
1126 | 0xc78, 0x740c0001, | ||
1127 | 0xc78, 0x730d0001, | ||
1128 | 0xc78, 0x720e0001, | ||
1129 | 0xc78, 0x710f0001, | ||
1130 | 0xc78, 0x70100001, | ||
1131 | 0xc78, 0x6f110001, | ||
1132 | 0xc78, 0x6e120001, | ||
1133 | 0xc78, 0x6d130001, | ||
1134 | 0xc78, 0x6c140001, | ||
1135 | 0xc78, 0x6b150001, | ||
1136 | 0xc78, 0x6a160001, | ||
1137 | 0xc78, 0x69170001, | ||
1138 | 0xc78, 0x68180001, | ||
1139 | 0xc78, 0x67190001, | ||
1140 | 0xc78, 0x661a0001, | ||
1141 | 0xc78, 0x651b0001, | ||
1142 | 0xc78, 0x641c0001, | ||
1143 | 0xc78, 0x631d0001, | ||
1144 | 0xc78, 0x621e0001, | ||
1145 | 0xc78, 0x611f0001, | ||
1146 | 0xc78, 0x60200001, | ||
1147 | 0xc78, 0x49210001, | ||
1148 | 0xc78, 0x48220001, | ||
1149 | 0xc78, 0x47230001, | ||
1150 | 0xc78, 0x46240001, | ||
1151 | 0xc78, 0x45250001, | ||
1152 | 0xc78, 0x44260001, | ||
1153 | 0xc78, 0x43270001, | ||
1154 | 0xc78, 0x42280001, | ||
1155 | 0xc78, 0x41290001, | ||
1156 | 0xc78, 0x402a0001, | ||
1157 | 0xc78, 0x262b0001, | ||
1158 | 0xc78, 0x252c0001, | ||
1159 | 0xc78, 0x242d0001, | ||
1160 | 0xc78, 0x232e0001, | ||
1161 | 0xc78, 0x222f0001, | ||
1162 | 0xc78, 0x21300001, | ||
1163 | 0xc78, 0x20310001, | ||
1164 | 0xc78, 0x06320001, | ||
1165 | 0xc78, 0x05330001, | ||
1166 | 0xc78, 0x04340001, | ||
1167 | 0xc78, 0x03350001, | ||
1168 | 0xc78, 0x02360001, | ||
1169 | 0xc78, 0x01370001, | ||
1170 | 0xc78, 0x00380001, | ||
1171 | 0xc78, 0x00390001, | ||
1172 | 0xc78, 0x003a0001, | ||
1173 | 0xc78, 0x003b0001, | ||
1174 | 0xc78, 0x003c0001, | ||
1175 | 0xc78, 0x003d0001, | ||
1176 | 0xc78, 0x003e0001, | ||
1177 | 0xc78, 0x003f0001, | ||
1178 | 0xc78, 0x7b400001, | ||
1179 | 0xc78, 0x7b410001, | ||
1180 | 0xc78, 0x7b420001, | ||
1181 | 0xc78, 0x7b430001, | ||
1182 | 0xc78, 0x7b440001, | ||
1183 | 0xc78, 0x7b450001, | ||
1184 | 0xc78, 0x7a460001, | ||
1185 | 0xc78, 0x79470001, | ||
1186 | 0xc78, 0x78480001, | ||
1187 | 0xc78, 0x77490001, | ||
1188 | 0xc78, 0x764a0001, | ||
1189 | 0xc78, 0x754b0001, | ||
1190 | 0xc78, 0x744c0001, | ||
1191 | 0xc78, 0x734d0001, | ||
1192 | 0xc78, 0x724e0001, | ||
1193 | 0xc78, 0x714f0001, | ||
1194 | 0xc78, 0x70500001, | ||
1195 | 0xc78, 0x6f510001, | ||
1196 | 0xc78, 0x6e520001, | ||
1197 | 0xc78, 0x6d530001, | ||
1198 | 0xc78, 0x6c540001, | ||
1199 | 0xc78, 0x6b550001, | ||
1200 | 0xc78, 0x6a560001, | ||
1201 | 0xc78, 0x69570001, | ||
1202 | 0xc78, 0x68580001, | ||
1203 | 0xc78, 0x67590001, | ||
1204 | 0xc78, 0x665a0001, | ||
1205 | 0xc78, 0x655b0001, | ||
1206 | 0xc78, 0x645c0001, | ||
1207 | 0xc78, 0x635d0001, | ||
1208 | 0xc78, 0x625e0001, | ||
1209 | 0xc78, 0x615f0001, | ||
1210 | 0xc78, 0x60600001, | ||
1211 | 0xc78, 0x49610001, | ||
1212 | 0xc78, 0x48620001, | ||
1213 | 0xc78, 0x47630001, | ||
1214 | 0xc78, 0x46640001, | ||
1215 | 0xc78, 0x45650001, | ||
1216 | 0xc78, 0x44660001, | ||
1217 | 0xc78, 0x43670001, | ||
1218 | 0xc78, 0x42680001, | ||
1219 | 0xc78, 0x41690001, | ||
1220 | 0xc78, 0x406a0001, | ||
1221 | 0xc78, 0x266b0001, | ||
1222 | 0xc78, 0x256c0001, | ||
1223 | 0xc78, 0x246d0001, | ||
1224 | 0xc78, 0x236e0001, | ||
1225 | 0xc78, 0x226f0001, | ||
1226 | 0xc78, 0x21700001, | ||
1227 | 0xc78, 0x20710001, | ||
1228 | 0xc78, 0x06720001, | ||
1229 | 0xc78, 0x05730001, | ||
1230 | 0xc78, 0x04740001, | ||
1231 | 0xc78, 0x03750001, | ||
1232 | 0xc78, 0x02760001, | ||
1233 | 0xc78, 0x01770001, | ||
1234 | 0xc78, 0x00780001, | ||
1235 | 0xc78, 0x00790001, | ||
1236 | 0xc78, 0x007a0001, | ||
1237 | 0xc78, 0x007b0001, | ||
1238 | 0xc78, 0x007c0001, | ||
1239 | 0xc78, 0x007d0001, | ||
1240 | 0xc78, 0x007e0001, | ||
1241 | 0xc78, 0x007f0001, | ||
1242 | 0xc78, 0x3800001e, | ||
1243 | 0xc78, 0x3801001e, | ||
1244 | 0xc78, 0x3802001e, | ||
1245 | 0xc78, 0x3803001e, | ||
1246 | 0xc78, 0x3804001e, | ||
1247 | 0xc78, 0x3805001e, | ||
1248 | 0xc78, 0x3806001e, | ||
1249 | 0xc78, 0x3807001e, | ||
1250 | 0xc78, 0x3808001e, | ||
1251 | 0xc78, 0x3c09001e, | ||
1252 | 0xc78, 0x3e0a001e, | ||
1253 | 0xc78, 0x400b001e, | ||
1254 | 0xc78, 0x440c001e, | ||
1255 | 0xc78, 0x480d001e, | ||
1256 | 0xc78, 0x4c0e001e, | ||
1257 | 0xc78, 0x500f001e, | ||
1258 | 0xc78, 0x5210001e, | ||
1259 | 0xc78, 0x5611001e, | ||
1260 | 0xc78, 0x5a12001e, | ||
1261 | 0xc78, 0x5e13001e, | ||
1262 | 0xc78, 0x6014001e, | ||
1263 | 0xc78, 0x6015001e, | ||
1264 | 0xc78, 0x6016001e, | ||
1265 | 0xc78, 0x6217001e, | ||
1266 | 0xc78, 0x6218001e, | ||
1267 | 0xc78, 0x6219001e, | ||
1268 | 0xc78, 0x621a001e, | ||
1269 | 0xc78, 0x621b001e, | ||
1270 | 0xc78, 0x621c001e, | ||
1271 | 0xc78, 0x621d001e, | ||
1272 | 0xc78, 0x621e001e, | ||
1273 | 0xc78, 0x621f001e, | ||
1274 | }; | ||
1275 | |||
1276 | u32 RTL8192CUPHY_REG_1T_HPArray[RTL8192CUPHY_REG_1T_HPArrayLength] = { | ||
1277 | 0x024, 0x0011800f, | ||
1278 | 0x028, 0x00ffdb83, | ||
1279 | 0x040, 0x000c0004, | ||
1280 | 0x800, 0x80040000, | ||
1281 | 0x804, 0x00000001, | ||
1282 | 0x808, 0x0000fc00, | ||
1283 | 0x80c, 0x0000000a, | ||
1284 | 0x810, 0x10005388, | ||
1285 | 0x814, 0x020c3d10, | ||
1286 | 0x818, 0x02200385, | ||
1287 | 0x81c, 0x00000000, | ||
1288 | 0x820, 0x01000100, | ||
1289 | 0x824, 0x00390204, | ||
1290 | 0x828, 0x00000000, | ||
1291 | 0x82c, 0x00000000, | ||
1292 | 0x830, 0x00000000, | ||
1293 | 0x834, 0x00000000, | ||
1294 | 0x838, 0x00000000, | ||
1295 | 0x83c, 0x00000000, | ||
1296 | 0x840, 0x00010000, | ||
1297 | 0x844, 0x00000000, | ||
1298 | 0x848, 0x00000000, | ||
1299 | 0x84c, 0x00000000, | ||
1300 | 0x850, 0x00000000, | ||
1301 | 0x854, 0x00000000, | ||
1302 | 0x858, 0x569a569a, | ||
1303 | 0x85c, 0x001b25a4, | ||
1304 | 0x860, 0x66e60230, | ||
1305 | 0x864, 0x061f0130, | ||
1306 | 0x868, 0x00000000, | ||
1307 | 0x86c, 0x20202000, | ||
1308 | 0x870, 0x03000300, | ||
1309 | 0x874, 0x22004000, | ||
1310 | 0x878, 0x00000808, | ||
1311 | 0x87c, 0x00ffc3f1, | ||
1312 | 0x880, 0xc0083070, | ||
1313 | 0x884, 0x000004d5, | ||
1314 | 0x888, 0x00000000, | ||
1315 | 0x88c, 0xccc000c0, | ||
1316 | 0x890, 0x00000800, | ||
1317 | 0x894, 0xfffffffe, | ||
1318 | 0x898, 0x40302010, | ||
1319 | 0x89c, 0x00706050, | ||
1320 | 0x900, 0x00000000, | ||
1321 | 0x904, 0x00000023, | ||
1322 | 0x908, 0x00000000, | ||
1323 | 0x90c, 0x81121111, | ||
1324 | 0xa00, 0x00d047c8, | ||
1325 | 0xa04, 0x80ff000c, | ||
1326 | 0xa08, 0x8c838300, | ||
1327 | 0xa0c, 0x2e68120f, | ||
1328 | 0xa10, 0x9500bb78, | ||
1329 | 0xa14, 0x11144028, | ||
1330 | 0xa18, 0x00881117, | ||
1331 | 0xa1c, 0x89140f00, | ||
1332 | 0xa20, 0x15160000, | ||
1333 | 0xa24, 0x070b0f12, | ||
1334 | 0xa28, 0x00000104, | ||
1335 | 0xa2c, 0x00d30000, | ||
1336 | 0xa70, 0x101fbf00, | ||
1337 | 0xa74, 0x00000007, | ||
1338 | 0xc00, 0x48071d40, | ||
1339 | 0xc04, 0x03a05611, | ||
1340 | 0xc08, 0x000000e4, | ||
1341 | 0xc0c, 0x6c6c6c6c, | ||
1342 | 0xc10, 0x08800000, | ||
1343 | 0xc14, 0x40000100, | ||
1344 | 0xc18, 0x08800000, | ||
1345 | 0xc1c, 0x40000100, | ||
1346 | 0xc20, 0x00000000, | ||
1347 | 0xc24, 0x00000000, | ||
1348 | 0xc28, 0x00000000, | ||
1349 | 0xc2c, 0x00000000, | ||
1350 | 0xc30, 0x69e9ac44, | ||
1351 | 0xc34, 0x469652cf, | ||
1352 | 0xc38, 0x49795994, | ||
1353 | 0xc3c, 0x0a97971c, | ||
1354 | 0xc40, 0x1f7c403f, | ||
1355 | 0xc44, 0x000100b7, | ||
1356 | 0xc48, 0xec020107, | ||
1357 | 0xc4c, 0x007f037f, | ||
1358 | 0xc50, 0x6954342e, | ||
1359 | 0xc54, 0x43bc0094, | ||
1360 | 0xc58, 0x6954342f, | ||
1361 | 0xc5c, 0x433c0094, | ||
1362 | 0xc60, 0x00000000, | ||
1363 | 0xc64, 0x5116848b, | ||
1364 | 0xc68, 0x47c00bff, | ||
1365 | 0xc6c, 0x00000036, | ||
1366 | 0xc70, 0x2c46000d, | ||
1367 | 0xc74, 0x018610db, | ||
1368 | 0xc78, 0x0000001f, | ||
1369 | 0xc7c, 0x00b91612, | ||
1370 | 0xc80, 0x24000090, | ||
1371 | 0xc84, 0x20f60000, | ||
1372 | 0xc88, 0x24000090, | ||
1373 | 0xc8c, 0x20200000, | ||
1374 | 0xc90, 0x00121820, | ||
1375 | 0xc94, 0x00000000, | ||
1376 | 0xc98, 0x00121820, | ||
1377 | 0xc9c, 0x00007f7f, | ||
1378 | 0xca0, 0x00000000, | ||
1379 | 0xca4, 0x00000080, | ||
1380 | 0xca8, 0x00000000, | ||
1381 | 0xcac, 0x00000000, | ||
1382 | 0xcb0, 0x00000000, | ||
1383 | 0xcb4, 0x00000000, | ||
1384 | 0xcb8, 0x00000000, | ||
1385 | 0xcbc, 0x28000000, | ||
1386 | 0xcc0, 0x00000000, | ||
1387 | 0xcc4, 0x00000000, | ||
1388 | 0xcc8, 0x00000000, | ||
1389 | 0xccc, 0x00000000, | ||
1390 | 0xcd0, 0x00000000, | ||
1391 | 0xcd4, 0x00000000, | ||
1392 | 0xcd8, 0x64b22427, | ||
1393 | 0xcdc, 0x00766932, | ||
1394 | 0xce0, 0x00222222, | ||
1395 | 0xce4, 0x00000000, | ||
1396 | 0xce8, 0x37644302, | ||
1397 | 0xcec, 0x2f97d40c, | ||
1398 | 0xd00, 0x00080740, | ||
1399 | 0xd04, 0x00020401, | ||
1400 | 0xd08, 0x0000907f, | ||
1401 | 0xd0c, 0x20010201, | ||
1402 | 0xd10, 0xa0633333, | ||
1403 | 0xd14, 0x3333bc43, | ||
1404 | 0xd18, 0x7a8f5b6b, | ||
1405 | 0xd2c, 0xcc979975, | ||
1406 | 0xd30, 0x00000000, | ||
1407 | 0xd34, 0x80608000, | ||
1408 | 0xd38, 0x00000000, | ||
1409 | 0xd3c, 0x00027293, | ||
1410 | 0xd40, 0x00000000, | ||
1411 | 0xd44, 0x00000000, | ||
1412 | 0xd48, 0x00000000, | ||
1413 | 0xd4c, 0x00000000, | ||
1414 | 0xd50, 0x6437140a, | ||
1415 | 0xd54, 0x00000000, | ||
1416 | 0xd58, 0x00000000, | ||
1417 | 0xd5c, 0x30032064, | ||
1418 | 0xd60, 0x4653de68, | ||
1419 | 0xd64, 0x04518a3c, | ||
1420 | 0xd68, 0x00002101, | ||
1421 | 0xd6c, 0x2a201c16, | ||
1422 | 0xd70, 0x1812362e, | ||
1423 | 0xd74, 0x322c2220, | ||
1424 | 0xd78, 0x000e3c24, | ||
1425 | 0xe00, 0x24242424, | ||
1426 | 0xe04, 0x24242424, | ||
1427 | 0xe08, 0x03902024, | ||
1428 | 0xe10, 0x24242424, | ||
1429 | 0xe14, 0x24242424, | ||
1430 | 0xe18, 0x24242424, | ||
1431 | 0xe1c, 0x24242424, | ||
1432 | 0xe28, 0x00000000, | ||
1433 | 0xe30, 0x1000dc1f, | ||
1434 | 0xe34, 0x10008c1f, | ||
1435 | 0xe38, 0x02140102, | ||
1436 | 0xe3c, 0x681604c2, | ||
1437 | 0xe40, 0x01007c00, | ||
1438 | 0xe44, 0x01004800, | ||
1439 | 0xe48, 0xfb000000, | ||
1440 | 0xe4c, 0x000028d1, | ||
1441 | 0xe50, 0x1000dc1f, | ||
1442 | 0xe54, 0x10008c1f, | ||
1443 | 0xe58, 0x02140102, | ||
1444 | 0xe5c, 0x28160d05, | ||
1445 | 0xe60, 0x00000008, | ||
1446 | 0xe68, 0x001b25a4, | ||
1447 | 0xe6c, 0x631b25a0, | ||
1448 | 0xe70, 0x631b25a0, | ||
1449 | 0xe74, 0x081b25a0, | ||
1450 | 0xe78, 0x081b25a0, | ||
1451 | 0xe7c, 0x081b25a0, | ||
1452 | 0xe80, 0x081b25a0, | ||
1453 | 0xe84, 0x631b25a0, | ||
1454 | 0xe88, 0x081b25a0, | ||
1455 | 0xe8c, 0x631b25a0, | ||
1456 | 0xed0, 0x631b25a0, | ||
1457 | 0xed4, 0x631b25a0, | ||
1458 | 0xed8, 0x631b25a0, | ||
1459 | 0xedc, 0x001b25a0, | ||
1460 | 0xee0, 0x001b25a0, | ||
1461 | 0xeec, 0x6b1b25a0, | ||
1462 | 0xee8, 0x31555448, | ||
1463 | 0xf14, 0x00000003, | ||
1464 | 0xf4c, 0x00000000, | ||
1465 | 0xf00, 0x00000300, | ||
1466 | }; | ||
1467 | |||
1468 | u32 RTL8192CUPHY_REG_Array_PG_HP[RTL8192CUPHY_REG_Array_PG_HPLength] = { | ||
1469 | 0xe00, 0xffffffff, 0x06080808, | ||
1470 | 0xe04, 0xffffffff, 0x00040406, | ||
1471 | 0xe08, 0x0000ff00, 0x00000000, | ||
1472 | 0x86c, 0xffffff00, 0x00000000, | ||
1473 | 0xe10, 0xffffffff, 0x04060608, | ||
1474 | 0xe14, 0xffffffff, 0x00020204, | ||
1475 | 0xe18, 0xffffffff, 0x04060608, | ||
1476 | 0xe1c, 0xffffffff, 0x00020204, | ||
1477 | 0x830, 0xffffffff, 0x06080808, | ||
1478 | 0x834, 0xffffffff, 0x00040406, | ||
1479 | 0x838, 0xffffff00, 0x00000000, | ||
1480 | 0x86c, 0x000000ff, 0x00000000, | ||
1481 | 0x83c, 0xffffffff, 0x04060608, | ||
1482 | 0x848, 0xffffffff, 0x00020204, | ||
1483 | 0x84c, 0xffffffff, 0x04060608, | ||
1484 | 0x868, 0xffffffff, 0x00020204, | ||
1485 | 0xe00, 0xffffffff, 0x00000000, | ||
1486 | 0xe04, 0xffffffff, 0x00000000, | ||
1487 | 0xe08, 0x0000ff00, 0x00000000, | ||
1488 | 0x86c, 0xffffff00, 0x00000000, | ||
1489 | 0xe10, 0xffffffff, 0x00000000, | ||
1490 | 0xe14, 0xffffffff, 0x00000000, | ||
1491 | 0xe18, 0xffffffff, 0x00000000, | ||
1492 | 0xe1c, 0xffffffff, 0x00000000, | ||
1493 | 0x830, 0xffffffff, 0x00000000, | ||
1494 | 0x834, 0xffffffff, 0x00000000, | ||
1495 | 0x838, 0xffffff00, 0x00000000, | ||
1496 | 0x86c, 0x000000ff, 0x00000000, | ||
1497 | 0x83c, 0xffffffff, 0x00000000, | ||
1498 | 0x848, 0xffffffff, 0x00000000, | ||
1499 | 0x84c, 0xffffffff, 0x00000000, | ||
1500 | 0x868, 0xffffffff, 0x00000000, | ||
1501 | 0xe00, 0xffffffff, 0x00000000, | ||
1502 | 0xe04, 0xffffffff, 0x00000000, | ||
1503 | 0xe08, 0x0000ff00, 0x00000000, | ||
1504 | 0x86c, 0xffffff00, 0x00000000, | ||
1505 | 0xe10, 0xffffffff, 0x00000000, | ||
1506 | 0xe14, 0xffffffff, 0x00000000, | ||
1507 | 0xe18, 0xffffffff, 0x00000000, | ||
1508 | 0xe1c, 0xffffffff, 0x00000000, | ||
1509 | 0x830, 0xffffffff, 0x00000000, | ||
1510 | 0x834, 0xffffffff, 0x00000000, | ||
1511 | 0x838, 0xffffff00, 0x00000000, | ||
1512 | 0x86c, 0x000000ff, 0x00000000, | ||
1513 | 0x83c, 0xffffffff, 0x00000000, | ||
1514 | 0x848, 0xffffffff, 0x00000000, | ||
1515 | 0x84c, 0xffffffff, 0x00000000, | ||
1516 | 0x868, 0xffffffff, 0x00000000, | ||
1517 | 0xe00, 0xffffffff, 0x00000000, | ||
1518 | 0xe04, 0xffffffff, 0x00000000, | ||
1519 | 0xe08, 0x0000ff00, 0x00000000, | ||
1520 | 0x86c, 0xffffff00, 0x00000000, | ||
1521 | 0xe10, 0xffffffff, 0x00000000, | ||
1522 | 0xe14, 0xffffffff, 0x00000000, | ||
1523 | 0xe18, 0xffffffff, 0x00000000, | ||
1524 | 0xe1c, 0xffffffff, 0x00000000, | ||
1525 | 0x830, 0xffffffff, 0x00000000, | ||
1526 | 0x834, 0xffffffff, 0x00000000, | ||
1527 | 0x838, 0xffffff00, 0x00000000, | ||
1528 | 0x86c, 0x000000ff, 0x00000000, | ||
1529 | 0x83c, 0xffffffff, 0x00000000, | ||
1530 | 0x848, 0xffffffff, 0x00000000, | ||
1531 | 0x84c, 0xffffffff, 0x00000000, | ||
1532 | 0x868, 0xffffffff, 0x00000000, | ||
1533 | 0xe00, 0xffffffff, 0x00000000, | ||
1534 | 0xe04, 0xffffffff, 0x00000000, | ||
1535 | 0xe08, 0x0000ff00, 0x00000000, | ||
1536 | 0x86c, 0xffffff00, 0x00000000, | ||
1537 | 0xe10, 0xffffffff, 0x00000000, | ||
1538 | 0xe14, 0xffffffff, 0x00000000, | ||
1539 | 0xe18, 0xffffffff, 0x00000000, | ||
1540 | 0xe1c, 0xffffffff, 0x00000000, | ||
1541 | 0x830, 0xffffffff, 0x00000000, | ||
1542 | 0x834, 0xffffffff, 0x00000000, | ||
1543 | 0x838, 0xffffff00, 0x00000000, | ||
1544 | 0x86c, 0x000000ff, 0x00000000, | ||
1545 | 0x83c, 0xffffffff, 0x00000000, | ||
1546 | 0x848, 0xffffffff, 0x00000000, | ||
1547 | 0x84c, 0xffffffff, 0x00000000, | ||
1548 | 0x868, 0xffffffff, 0x00000000, | ||
1549 | 0xe00, 0xffffffff, 0x00000000, | ||
1550 | 0xe04, 0xffffffff, 0x00000000, | ||
1551 | 0xe08, 0x0000ff00, 0x00000000, | ||
1552 | 0x86c, 0xffffff00, 0x00000000, | ||
1553 | 0xe10, 0xffffffff, 0x00000000, | ||
1554 | 0xe14, 0xffffffff, 0x00000000, | ||
1555 | 0xe18, 0xffffffff, 0x00000000, | ||
1556 | 0xe1c, 0xffffffff, 0x00000000, | ||
1557 | 0x830, 0xffffffff, 0x00000000, | ||
1558 | 0x834, 0xffffffff, 0x00000000, | ||
1559 | 0x838, 0xffffff00, 0x00000000, | ||
1560 | 0x86c, 0x000000ff, 0x00000000, | ||
1561 | 0x83c, 0xffffffff, 0x00000000, | ||
1562 | 0x848, 0xffffffff, 0x00000000, | ||
1563 | 0x84c, 0xffffffff, 0x00000000, | ||
1564 | 0x868, 0xffffffff, 0x00000000, | ||
1565 | 0xe00, 0xffffffff, 0x00000000, | ||
1566 | 0xe04, 0xffffffff, 0x00000000, | ||
1567 | 0xe08, 0x0000ff00, 0x00000000, | ||
1568 | 0x86c, 0xffffff00, 0x00000000, | ||
1569 | 0xe10, 0xffffffff, 0x00000000, | ||
1570 | 0xe14, 0xffffffff, 0x00000000, | ||
1571 | 0xe18, 0xffffffff, 0x00000000, | ||
1572 | 0xe1c, 0xffffffff, 0x00000000, | ||
1573 | 0x830, 0xffffffff, 0x00000000, | ||
1574 | 0x834, 0xffffffff, 0x00000000, | ||
1575 | 0x838, 0xffffff00, 0x00000000, | ||
1576 | 0x86c, 0x000000ff, 0x00000000, | ||
1577 | 0x83c, 0xffffffff, 0x00000000, | ||
1578 | 0x848, 0xffffffff, 0x00000000, | ||
1579 | 0x84c, 0xffffffff, 0x00000000, | ||
1580 | 0x868, 0xffffffff, 0x00000000, | ||
1581 | }; | ||
1582 | |||
1583 | u32 RTL8192CURadioA_1T_HPArray[RTL8192CURadioA_1T_HPArrayLength] = { | ||
1584 | 0x000, 0x00030159, | ||
1585 | 0x001, 0x00031284, | ||
1586 | 0x002, 0x00098000, | ||
1587 | 0x003, 0x00018c63, | ||
1588 | 0x004, 0x000210e7, | ||
1589 | 0x009, 0x0002044f, | ||
1590 | 0x00a, 0x0001adb0, | ||
1591 | 0x00b, 0x00054867, | ||
1592 | 0x00c, 0x0008992e, | ||
1593 | 0x00d, 0x0000e529, | ||
1594 | 0x00e, 0x00039ce7, | ||
1595 | 0x00f, 0x00000451, | ||
1596 | 0x019, 0x00000000, | ||
1597 | 0x01a, 0x00000255, | ||
1598 | 0x01b, 0x00060a00, | ||
1599 | 0x01c, 0x000fc378, | ||
1600 | 0x01d, 0x000a1250, | ||
1601 | 0x01e, 0x0004445f, | ||
1602 | 0x01f, 0x00080001, | ||
1603 | 0x020, 0x0000b614, | ||
1604 | 0x021, 0x0006c000, | ||
1605 | 0x022, 0x0000083c, | ||
1606 | 0x023, 0x00001558, | ||
1607 | 0x024, 0x00000060, | ||
1608 | 0x025, 0x00000483, | ||
1609 | 0x026, 0x0004f000, | ||
1610 | 0x027, 0x000ec7d9, | ||
1611 | 0x028, 0x000977c0, | ||
1612 | 0x029, 0x00004783, | ||
1613 | 0x02a, 0x00000001, | ||
1614 | 0x02b, 0x00021334, | ||
1615 | 0x02a, 0x00000000, | ||
1616 | 0x02b, 0x00000054, | ||
1617 | 0x02a, 0x00000001, | ||
1618 | 0x02b, 0x00000808, | ||
1619 | 0x02b, 0x00053333, | ||
1620 | 0x02c, 0x0000000c, | ||
1621 | 0x02a, 0x00000002, | ||
1622 | 0x02b, 0x00000808, | ||
1623 | 0x02b, 0x0005b333, | ||
1624 | 0x02c, 0x0000000d, | ||
1625 | 0x02a, 0x00000003, | ||
1626 | 0x02b, 0x00000808, | ||
1627 | 0x02b, 0x00063333, | ||
1628 | 0x02c, 0x0000000d, | ||
1629 | 0x02a, 0x00000004, | ||
1630 | 0x02b, 0x00000808, | ||
1631 | 0x02b, 0x0006b333, | ||
1632 | 0x02c, 0x0000000d, | ||
1633 | 0x02a, 0x00000005, | ||
1634 | 0x02b, 0x00000808, | ||
1635 | 0x02b, 0x00073333, | ||
1636 | 0x02c, 0x0000000d, | ||
1637 | 0x02a, 0x00000006, | ||
1638 | 0x02b, 0x00000709, | ||
1639 | 0x02b, 0x0005b333, | ||
1640 | 0x02c, 0x0000000d, | ||
1641 | 0x02a, 0x00000007, | ||
1642 | 0x02b, 0x00000709, | ||
1643 | 0x02b, 0x00063333, | ||
1644 | 0x02c, 0x0000000d, | ||
1645 | 0x02a, 0x00000008, | ||
1646 | 0x02b, 0x0000060a, | ||
1647 | 0x02b, 0x0004b333, | ||
1648 | 0x02c, 0x0000000d, | ||
1649 | 0x02a, 0x00000009, | ||
1650 | 0x02b, 0x0000060a, | ||
1651 | 0x02b, 0x00053333, | ||
1652 | 0x02c, 0x0000000d, | ||
1653 | 0x02a, 0x0000000a, | ||
1654 | 0x02b, 0x0000060a, | ||
1655 | 0x02b, 0x0005b333, | ||
1656 | 0x02c, 0x0000000d, | ||
1657 | 0x02a, 0x0000000b, | ||
1658 | 0x02b, 0x0000060a, | ||
1659 | 0x02b, 0x00063333, | ||
1660 | 0x02c, 0x0000000d, | ||
1661 | 0x02a, 0x0000000c, | ||
1662 | 0x02b, 0x0000060a, | ||
1663 | 0x02b, 0x0006b333, | ||
1664 | 0x02c, 0x0000000d, | ||
1665 | 0x02a, 0x0000000d, | ||
1666 | 0x02b, 0x0000060a, | ||
1667 | 0x02b, 0x00073333, | ||
1668 | 0x02c, 0x0000000d, | ||
1669 | 0x02a, 0x0000000e, | ||
1670 | 0x02b, 0x0000050b, | ||
1671 | 0x02b, 0x00066666, | ||
1672 | 0x02c, 0x0000001a, | ||
1673 | 0x02a, 0x000e0000, | ||
1674 | 0x010, 0x0004000f, | ||
1675 | 0x011, 0x000e31fc, | ||
1676 | 0x010, 0x0006000f, | ||
1677 | 0x011, 0x000ff9f8, | ||
1678 | 0x010, 0x0002000f, | ||
1679 | 0x011, 0x000203f9, | ||
1680 | 0x010, 0x0003000f, | ||
1681 | 0x011, 0x000ff500, | ||
1682 | 0x010, 0x00000000, | ||
1683 | 0x011, 0x00000000, | ||
1684 | 0x010, 0x0008000f, | ||
1685 | 0x011, 0x0003f100, | ||
1686 | 0x010, 0x0009000f, | ||
1687 | 0x011, 0x00023100, | ||
1688 | 0x012, 0x000d8000, | ||
1689 | 0x012, 0x00090000, | ||
1690 | 0x012, 0x00051000, | ||
1691 | 0x012, 0x00012000, | ||
1692 | 0x013, 0x00028fb4, | ||
1693 | 0x013, 0x00024fa8, | ||
1694 | 0x013, 0x000207a4, | ||
1695 | 0x013, 0x0001c798, | ||
1696 | 0x013, 0x000183a4, | ||
1697 | 0x013, 0x00014398, | ||
1698 | 0x013, 0x000101a4, | ||
1699 | 0x013, 0x0000c198, | ||
1700 | 0x013, 0x000080a4, | ||
1701 | 0x013, 0x00004098, | ||
1702 | 0x013, 0x00000000, | ||
1703 | 0x014, 0x0001944c, | ||
1704 | 0x014, 0x00059444, | ||
1705 | 0x014, 0x0009944c, | ||
1706 | 0x014, 0x000d9444, | ||
1707 | 0x015, 0x0000f405, | ||
1708 | 0x015, 0x0004f405, | ||
1709 | 0x015, 0x0008f405, | ||
1710 | 0x015, 0x000cf405, | ||
1711 | 0x016, 0x000e0330, | ||
1712 | 0x016, 0x000a0330, | ||
1713 | 0x016, 0x00060330, | ||
1714 | 0x016, 0x00020330, | ||
1715 | 0x000, 0x00010159, | ||
1716 | 0x018, 0x0000f401, | ||
1717 | 0x0fe, 0x00000000, | ||
1718 | 0x0fe, 0x00000000, | ||
1719 | 0x01f, 0x00080003, | ||
1720 | 0x0fe, 0x00000000, | ||
1721 | 0x0fe, 0x00000000, | ||
1722 | 0x01e, 0x00044457, | ||
1723 | 0x01f, 0x00080000, | ||
1724 | 0x000, 0x00030159, | ||
1725 | }; | ||
1726 | |||
1727 | u32 Rtl8192CUAGCTAB_1T_HPArray[RTL8192CUAGCTAB_1T_HPArrayLength] = { | ||
1728 | 0xc78, 0x7b000001, | ||
1729 | 0xc78, 0x7b010001, | ||
1730 | 0xc78, 0x7b020001, | ||
1731 | 0xc78, 0x7b030001, | ||
1732 | 0xc78, 0x7b040001, | ||
1733 | 0xc78, 0x7b050001, | ||
1734 | 0xc78, 0x7b060001, | ||
1735 | 0xc78, 0x7b070001, | ||
1736 | 0xc78, 0x7b080001, | ||
1737 | 0xc78, 0x7a090001, | ||
1738 | 0xc78, 0x790a0001, | ||
1739 | 0xc78, 0x780b0001, | ||
1740 | 0xc78, 0x770c0001, | ||
1741 | 0xc78, 0x760d0001, | ||
1742 | 0xc78, 0x750e0001, | ||
1743 | 0xc78, 0x740f0001, | ||
1744 | 0xc78, 0x73100001, | ||
1745 | 0xc78, 0x72110001, | ||
1746 | 0xc78, 0x71120001, | ||
1747 | 0xc78, 0x70130001, | ||
1748 | 0xc78, 0x6f140001, | ||
1749 | 0xc78, 0x6e150001, | ||
1750 | 0xc78, 0x6d160001, | ||
1751 | 0xc78, 0x6c170001, | ||
1752 | 0xc78, 0x6b180001, | ||
1753 | 0xc78, 0x6a190001, | ||
1754 | 0xc78, 0x691a0001, | ||
1755 | 0xc78, 0x681b0001, | ||
1756 | 0xc78, 0x671c0001, | ||
1757 | 0xc78, 0x661d0001, | ||
1758 | 0xc78, 0x651e0001, | ||
1759 | 0xc78, 0x641f0001, | ||
1760 | 0xc78, 0x63200001, | ||
1761 | 0xc78, 0x62210001, | ||
1762 | 0xc78, 0x61220001, | ||
1763 | 0xc78, 0x60230001, | ||
1764 | 0xc78, 0x46240001, | ||
1765 | 0xc78, 0x45250001, | ||
1766 | 0xc78, 0x44260001, | ||
1767 | 0xc78, 0x43270001, | ||
1768 | 0xc78, 0x42280001, | ||
1769 | 0xc78, 0x41290001, | ||
1770 | 0xc78, 0x402a0001, | ||
1771 | 0xc78, 0x262b0001, | ||
1772 | 0xc78, 0x252c0001, | ||
1773 | 0xc78, 0x242d0001, | ||
1774 | 0xc78, 0x232e0001, | ||
1775 | 0xc78, 0x222f0001, | ||
1776 | 0xc78, 0x21300001, | ||
1777 | 0xc78, 0x20310001, | ||
1778 | 0xc78, 0x06320001, | ||
1779 | 0xc78, 0x05330001, | ||
1780 | 0xc78, 0x04340001, | ||
1781 | 0xc78, 0x03350001, | ||
1782 | 0xc78, 0x02360001, | ||
1783 | 0xc78, 0x01370001, | ||
1784 | 0xc78, 0x00380001, | ||
1785 | 0xc78, 0x00390001, | ||
1786 | 0xc78, 0x003a0001, | ||
1787 | 0xc78, 0x003b0001, | ||
1788 | 0xc78, 0x003c0001, | ||
1789 | 0xc78, 0x003d0001, | ||
1790 | 0xc78, 0x003e0001, | ||
1791 | 0xc78, 0x003f0001, | ||
1792 | 0xc78, 0x7b400001, | ||
1793 | 0xc78, 0x7b410001, | ||
1794 | 0xc78, 0x7b420001, | ||
1795 | 0xc78, 0x7b430001, | ||
1796 | 0xc78, 0x7b440001, | ||
1797 | 0xc78, 0x7b450001, | ||
1798 | 0xc78, 0x7b460001, | ||
1799 | 0xc78, 0x7b470001, | ||
1800 | 0xc78, 0x7b480001, | ||
1801 | 0xc78, 0x7a490001, | ||
1802 | 0xc78, 0x794a0001, | ||
1803 | 0xc78, 0x784b0001, | ||
1804 | 0xc78, 0x774c0001, | ||
1805 | 0xc78, 0x764d0001, | ||
1806 | 0xc78, 0x754e0001, | ||
1807 | 0xc78, 0x744f0001, | ||
1808 | 0xc78, 0x73500001, | ||
1809 | 0xc78, 0x72510001, | ||
1810 | 0xc78, 0x71520001, | ||
1811 | 0xc78, 0x70530001, | ||
1812 | 0xc78, 0x6f540001, | ||
1813 | 0xc78, 0x6e550001, | ||
1814 | 0xc78, 0x6d560001, | ||
1815 | 0xc78, 0x6c570001, | ||
1816 | 0xc78, 0x6b580001, | ||
1817 | 0xc78, 0x6a590001, | ||
1818 | 0xc78, 0x695a0001, | ||
1819 | 0xc78, 0x685b0001, | ||
1820 | 0xc78, 0x675c0001, | ||
1821 | 0xc78, 0x665d0001, | ||
1822 | 0xc78, 0x655e0001, | ||
1823 | 0xc78, 0x645f0001, | ||
1824 | 0xc78, 0x63600001, | ||
1825 | 0xc78, 0x62610001, | ||
1826 | 0xc78, 0x61620001, | ||
1827 | 0xc78, 0x60630001, | ||
1828 | 0xc78, 0x46640001, | ||
1829 | 0xc78, 0x45650001, | ||
1830 | 0xc78, 0x44660001, | ||
1831 | 0xc78, 0x43670001, | ||
1832 | 0xc78, 0x42680001, | ||
1833 | 0xc78, 0x41690001, | ||
1834 | 0xc78, 0x406a0001, | ||
1835 | 0xc78, 0x266b0001, | ||
1836 | 0xc78, 0x256c0001, | ||
1837 | 0xc78, 0x246d0001, | ||
1838 | 0xc78, 0x236e0001, | ||
1839 | 0xc78, 0x226f0001, | ||
1840 | 0xc78, 0x21700001, | ||
1841 | 0xc78, 0x20710001, | ||
1842 | 0xc78, 0x06720001, | ||
1843 | 0xc78, 0x05730001, | ||
1844 | 0xc78, 0x04740001, | ||
1845 | 0xc78, 0x03750001, | ||
1846 | 0xc78, 0x02760001, | ||
1847 | 0xc78, 0x01770001, | ||
1848 | 0xc78, 0x00780001, | ||
1849 | 0xc78, 0x00790001, | ||
1850 | 0xc78, 0x007a0001, | ||
1851 | 0xc78, 0x007b0001, | ||
1852 | 0xc78, 0x007c0001, | ||
1853 | 0xc78, 0x007d0001, | ||
1854 | 0xc78, 0x007e0001, | ||
1855 | 0xc78, 0x007f0001, | ||
1856 | 0xc78, 0x3800001e, | ||
1857 | 0xc78, 0x3801001e, | ||
1858 | 0xc78, 0x3802001e, | ||
1859 | 0xc78, 0x3803001e, | ||
1860 | 0xc78, 0x3804001e, | ||
1861 | 0xc78, 0x3805001e, | ||
1862 | 0xc78, 0x3806001e, | ||
1863 | 0xc78, 0x3807001e, | ||
1864 | 0xc78, 0x3808001e, | ||
1865 | 0xc78, 0x3c09001e, | ||
1866 | 0xc78, 0x3e0a001e, | ||
1867 | 0xc78, 0x400b001e, | ||
1868 | 0xc78, 0x440c001e, | ||
1869 | 0xc78, 0x480d001e, | ||
1870 | 0xc78, 0x4c0e001e, | ||
1871 | 0xc78, 0x500f001e, | ||
1872 | 0xc78, 0x5210001e, | ||
1873 | 0xc78, 0x5611001e, | ||
1874 | 0xc78, 0x5a12001e, | ||
1875 | 0xc78, 0x5e13001e, | ||
1876 | 0xc78, 0x6014001e, | ||
1877 | 0xc78, 0x6015001e, | ||
1878 | 0xc78, 0x6016001e, | ||
1879 | 0xc78, 0x6217001e, | ||
1880 | 0xc78, 0x6218001e, | ||
1881 | 0xc78, 0x6219001e, | ||
1882 | 0xc78, 0x621a001e, | ||
1883 | 0xc78, 0x621b001e, | ||
1884 | 0xc78, 0x621c001e, | ||
1885 | 0xc78, 0x621d001e, | ||
1886 | 0xc78, 0x621e001e, | ||
1887 | 0xc78, 0x621f001e, | ||
1888 | }; | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/table.h b/drivers/net/wireless/rtlwifi/rtl8192cu/table.h new file mode 100644 index 000000000000..c3d5cd826cfa --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/table.h | |||
@@ -0,0 +1,71 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CU_TABLE__H_ | ||
31 | #define __RTL92CU_TABLE__H_ | ||
32 | |||
33 | #include <linux/types.h> | ||
34 | |||
35 | #define RTL8192CUPHY_REG_2TARRAY_LENGTH 374 | ||
36 | extern u32 RTL8192CUPHY_REG_2TARRAY[RTL8192CUPHY_REG_2TARRAY_LENGTH]; | ||
37 | #define RTL8192CUPHY_REG_1TARRAY_LENGTH 374 | ||
38 | extern u32 RTL8192CUPHY_REG_1TARRAY[RTL8192CUPHY_REG_1TARRAY_LENGTH]; | ||
39 | |||
40 | #define RTL8192CUPHY_REG_ARRAY_PGLENGTH 336 | ||
41 | extern u32 RTL8192CUPHY_REG_ARRAY_PG[RTL8192CUPHY_REG_ARRAY_PGLENGTH]; | ||
42 | |||
43 | #define RTL8192CURADIOA_2TARRAYLENGTH 282 | ||
44 | extern u32 RTL8192CURADIOA_2TARRAY[RTL8192CURADIOA_2TARRAYLENGTH]; | ||
45 | #define RTL8192CURADIOB_2TARRAYLENGTH 78 | ||
46 | extern u32 RTL8192CU_RADIOB_2TARRAY[RTL8192CURADIOB_2TARRAYLENGTH]; | ||
47 | #define RTL8192CURADIOA_1TARRAYLENGTH 282 | ||
48 | extern u32 RTL8192CU_RADIOA_1TARRAY[RTL8192CURADIOA_1TARRAYLENGTH]; | ||
49 | #define RTL8192CURADIOB_1TARRAYLENGTH 1 | ||
50 | extern u32 RTL8192CU_RADIOB_1TARRAY[RTL8192CURADIOB_1TARRAYLENGTH]; | ||
51 | |||
52 | #define RTL8192CUMAC_2T_ARRAYLENGTH 172 | ||
53 | extern u32 RTL8192CUMAC_2T_ARRAY[RTL8192CUMAC_2T_ARRAYLENGTH]; | ||
54 | |||
55 | #define RTL8192CUAGCTAB_2TARRAYLENGTH 320 | ||
56 | extern u32 RTL8192CUAGCTAB_2TARRAY[RTL8192CUAGCTAB_2TARRAYLENGTH]; | ||
57 | #define RTL8192CUAGCTAB_1TARRAYLENGTH 320 | ||
58 | extern u32 RTL8192CUAGCTAB_1TARRAY[RTL8192CUAGCTAB_1TARRAYLENGTH]; | ||
59 | |||
60 | #define RTL8192CUPHY_REG_1T_HPArrayLength 378 | ||
61 | extern u32 RTL8192CUPHY_REG_1T_HPArray[RTL8192CUPHY_REG_1T_HPArrayLength]; | ||
62 | |||
63 | #define RTL8192CUPHY_REG_Array_PG_HPLength 336 | ||
64 | extern u32 RTL8192CUPHY_REG_Array_PG_HP[RTL8192CUPHY_REG_Array_PG_HPLength]; | ||
65 | |||
66 | #define RTL8192CURadioA_1T_HPArrayLength 282 | ||
67 | extern u32 RTL8192CURadioA_1T_HPArray[RTL8192CURadioA_1T_HPArrayLength]; | ||
68 | #define RTL8192CUAGCTAB_1T_HPArrayLength 320 | ||
69 | extern u32 Rtl8192CUAGCTAB_1T_HPArray[RTL8192CUAGCTAB_1T_HPArrayLength]; | ||
70 | |||
71 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/trx.c b/drivers/net/wireless/rtlwifi/rtl8192cu/trx.c new file mode 100644 index 000000000000..3a92ba3c4a1e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/trx.c | |||
@@ -0,0 +1,687 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../usb.h" | ||
32 | #include "../ps.h" | ||
33 | #include "../base.h" | ||
34 | #include "reg.h" | ||
35 | #include "def.h" | ||
36 | #include "phy.h" | ||
37 | #include "rf.h" | ||
38 | #include "dm.h" | ||
39 | #include "mac.h" | ||
40 | #include "trx.h" | ||
41 | |||
42 | static int _ConfigVerTOutEP(struct ieee80211_hw *hw) | ||
43 | { | ||
44 | u8 ep_cfg, txqsele; | ||
45 | u8 ep_nums = 0; | ||
46 | |||
47 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
48 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
49 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
50 | |||
51 | rtlusb->out_queue_sel = 0; | ||
52 | ep_cfg = rtl_read_byte(rtlpriv, REG_TEST_SIE_OPTIONAL); | ||
53 | ep_cfg = (ep_cfg & USB_TEST_EP_MASK) >> USB_TEST_EP_SHIFT; | ||
54 | switch (ep_cfg) { | ||
55 | case 0: /* 2 bulk OUT, 1 bulk IN */ | ||
56 | case 3: | ||
57 | rtlusb->out_queue_sel = TX_SELE_HQ | TX_SELE_LQ; | ||
58 | ep_nums = 2; | ||
59 | break; | ||
60 | case 1: /* 1 bulk IN/OUT => map all endpoint to Low queue */ | ||
61 | case 2: /* 1 bulk IN, 1 bulk OUT => map all endpoint to High queue */ | ||
62 | txqsele = rtl_read_byte(rtlpriv, REG_TEST_USB_TXQS); | ||
63 | if (txqsele & 0x0F) /* /map all endpoint to High queue */ | ||
64 | rtlusb->out_queue_sel = TX_SELE_HQ; | ||
65 | else if (txqsele&0xF0) /* map all endpoint to Low queue */ | ||
66 | rtlusb->out_queue_sel = TX_SELE_LQ; | ||
67 | ep_nums = 1; | ||
68 | break; | ||
69 | default: | ||
70 | break; | ||
71 | } | ||
72 | return (rtlusb->out_ep_nums == ep_nums) ? 0 : -EINVAL; | ||
73 | } | ||
74 | |||
75 | static int _ConfigVerNOutEP(struct ieee80211_hw *hw) | ||
76 | { | ||
77 | u8 ep_cfg; | ||
78 | u8 ep_nums = 0; | ||
79 | |||
80 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
81 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
82 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
83 | |||
84 | rtlusb->out_queue_sel = 0; | ||
85 | /* Normal and High queue */ | ||
86 | ep_cfg = rtl_read_byte(rtlpriv, (REG_NORMAL_SIE_EP + 1)); | ||
87 | if (ep_cfg & USB_NORMAL_SIE_EP_MASK) { | ||
88 | rtlusb->out_queue_sel |= TX_SELE_HQ; | ||
89 | ep_nums++; | ||
90 | } | ||
91 | if ((ep_cfg >> USB_NORMAL_SIE_EP_SHIFT) & USB_NORMAL_SIE_EP_MASK) { | ||
92 | rtlusb->out_queue_sel |= TX_SELE_NQ; | ||
93 | ep_nums++; | ||
94 | } | ||
95 | /* Low queue */ | ||
96 | ep_cfg = rtl_read_byte(rtlpriv, (REG_NORMAL_SIE_EP + 2)); | ||
97 | if (ep_cfg & USB_NORMAL_SIE_EP_MASK) { | ||
98 | rtlusb->out_queue_sel |= TX_SELE_LQ; | ||
99 | ep_nums++; | ||
100 | } | ||
101 | return (rtlusb->out_ep_nums == ep_nums) ? 0 : -EINVAL; | ||
102 | } | ||
103 | |||
104 | static void _TwoOutEpMapping(struct ieee80211_hw *hw, bool bIsChipB, | ||
105 | bool bwificfg, struct rtl_ep_map *ep_map) | ||
106 | { | ||
107 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
108 | |||
109 | if (bwificfg) { /* for WMM */ | ||
110 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
111 | ("USB Chip-B & WMM Setting.....\n")); | ||
112 | ep_map->ep_mapping[RTL_TXQ_BE] = 2; | ||
113 | ep_map->ep_mapping[RTL_TXQ_BK] = 3; | ||
114 | ep_map->ep_mapping[RTL_TXQ_VI] = 3; | ||
115 | ep_map->ep_mapping[RTL_TXQ_VO] = 2; | ||
116 | ep_map->ep_mapping[RTL_TXQ_MGT] = 2; | ||
117 | ep_map->ep_mapping[RTL_TXQ_BCN] = 2; | ||
118 | ep_map->ep_mapping[RTL_TXQ_HI] = 2; | ||
119 | } else { /* typical setting */ | ||
120 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
121 | ("USB typical Setting.....\n")); | ||
122 | ep_map->ep_mapping[RTL_TXQ_BE] = 3; | ||
123 | ep_map->ep_mapping[RTL_TXQ_BK] = 3; | ||
124 | ep_map->ep_mapping[RTL_TXQ_VI] = 2; | ||
125 | ep_map->ep_mapping[RTL_TXQ_VO] = 2; | ||
126 | ep_map->ep_mapping[RTL_TXQ_MGT] = 2; | ||
127 | ep_map->ep_mapping[RTL_TXQ_BCN] = 2; | ||
128 | ep_map->ep_mapping[RTL_TXQ_HI] = 2; | ||
129 | } | ||
130 | } | ||
131 | |||
132 | static void _ThreeOutEpMapping(struct ieee80211_hw *hw, bool bwificfg, | ||
133 | struct rtl_ep_map *ep_map) | ||
134 | { | ||
135 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
136 | if (bwificfg) { /* for WMM */ | ||
137 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
138 | ("USB 3EP Setting for WMM.....\n")); | ||
139 | ep_map->ep_mapping[RTL_TXQ_BE] = 5; | ||
140 | ep_map->ep_mapping[RTL_TXQ_BK] = 3; | ||
141 | ep_map->ep_mapping[RTL_TXQ_VI] = 3; | ||
142 | ep_map->ep_mapping[RTL_TXQ_VO] = 2; | ||
143 | ep_map->ep_mapping[RTL_TXQ_MGT] = 2; | ||
144 | ep_map->ep_mapping[RTL_TXQ_BCN] = 2; | ||
145 | ep_map->ep_mapping[RTL_TXQ_HI] = 2; | ||
146 | } else { /* typical setting */ | ||
147 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
148 | ("USB 3EP Setting for typical.....\n")); | ||
149 | ep_map->ep_mapping[RTL_TXQ_BE] = 5; | ||
150 | ep_map->ep_mapping[RTL_TXQ_BK] = 5; | ||
151 | ep_map->ep_mapping[RTL_TXQ_VI] = 3; | ||
152 | ep_map->ep_mapping[RTL_TXQ_VO] = 2; | ||
153 | ep_map->ep_mapping[RTL_TXQ_MGT] = 2; | ||
154 | ep_map->ep_mapping[RTL_TXQ_BCN] = 2; | ||
155 | ep_map->ep_mapping[RTL_TXQ_HI] = 2; | ||
156 | } | ||
157 | } | ||
158 | |||
159 | static void _OneOutEpMapping(struct ieee80211_hw *hw, struct rtl_ep_map *ep_map) | ||
160 | { | ||
161 | ep_map->ep_mapping[RTL_TXQ_BE] = 2; | ||
162 | ep_map->ep_mapping[RTL_TXQ_BK] = 2; | ||
163 | ep_map->ep_mapping[RTL_TXQ_VI] = 2; | ||
164 | ep_map->ep_mapping[RTL_TXQ_VO] = 2; | ||
165 | ep_map->ep_mapping[RTL_TXQ_MGT] = 2; | ||
166 | ep_map->ep_mapping[RTL_TXQ_BCN] = 2; | ||
167 | ep_map->ep_mapping[RTL_TXQ_HI] = 2; | ||
168 | } | ||
169 | static int _out_ep_mapping(struct ieee80211_hw *hw) | ||
170 | { | ||
171 | int err = 0; | ||
172 | bool bIsChipN, bwificfg = false; | ||
173 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
174 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
175 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
176 | struct rtl_ep_map *ep_map = &(rtlusb->ep_map); | ||
177 | |||
178 | bIsChipN = IS_NORMAL_CHIP(rtlhal->version); | ||
179 | switch (rtlusb->out_ep_nums) { | ||
180 | case 2: | ||
181 | _TwoOutEpMapping(hw, bIsChipN, bwificfg, ep_map); | ||
182 | break; | ||
183 | case 3: | ||
184 | /* Test chip doesn't support three out EPs. */ | ||
185 | if (!bIsChipN) { | ||
186 | err = -EINVAL; | ||
187 | goto err_out; | ||
188 | } | ||
189 | _ThreeOutEpMapping(hw, bIsChipN, ep_map); | ||
190 | break; | ||
191 | case 1: | ||
192 | _OneOutEpMapping(hw, ep_map); | ||
193 | break; | ||
194 | default: | ||
195 | err = -EINVAL; | ||
196 | break; | ||
197 | } | ||
198 | err_out: | ||
199 | return err; | ||
200 | |||
201 | } | ||
202 | /* endpoint mapping */ | ||
203 | int rtl8192cu_endpoint_mapping(struct ieee80211_hw *hw) | ||
204 | { | ||
205 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
206 | int error = 0; | ||
207 | if (likely(IS_NORMAL_CHIP(rtlhal->version))) | ||
208 | error = _ConfigVerNOutEP(hw); | ||
209 | else | ||
210 | error = _ConfigVerTOutEP(hw); | ||
211 | if (error) | ||
212 | goto err_out; | ||
213 | error = _out_ep_mapping(hw); | ||
214 | if (error) | ||
215 | goto err_out; | ||
216 | err_out: | ||
217 | return error; | ||
218 | } | ||
219 | |||
220 | u16 rtl8192cu_mq_to_hwq(__le16 fc, u16 mac80211_queue_index) | ||
221 | { | ||
222 | u16 hw_queue_index; | ||
223 | |||
224 | if (unlikely(ieee80211_is_beacon(fc))) { | ||
225 | hw_queue_index = RTL_TXQ_BCN; | ||
226 | goto out; | ||
227 | } | ||
228 | if (ieee80211_is_mgmt(fc)) { | ||
229 | hw_queue_index = RTL_TXQ_MGT; | ||
230 | goto out; | ||
231 | } | ||
232 | switch (mac80211_queue_index) { | ||
233 | case 0: | ||
234 | hw_queue_index = RTL_TXQ_VO; | ||
235 | break; | ||
236 | case 1: | ||
237 | hw_queue_index = RTL_TXQ_VI; | ||
238 | break; | ||
239 | case 2: | ||
240 | hw_queue_index = RTL_TXQ_BE; | ||
241 | break; | ||
242 | case 3: | ||
243 | hw_queue_index = RTL_TXQ_BK; | ||
244 | break; | ||
245 | default: | ||
246 | hw_queue_index = RTL_TXQ_BE; | ||
247 | RT_ASSERT(false, ("QSLT_BE queue, skb_queue:%d\n", | ||
248 | mac80211_queue_index)); | ||
249 | break; | ||
250 | } | ||
251 | out: | ||
252 | return hw_queue_index; | ||
253 | } | ||
254 | |||
255 | static enum rtl_desc_qsel _rtl8192cu_mq_to_descq(struct ieee80211_hw *hw, | ||
256 | __le16 fc, u16 mac80211_queue_index) | ||
257 | { | ||
258 | enum rtl_desc_qsel qsel; | ||
259 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
260 | |||
261 | if (unlikely(ieee80211_is_beacon(fc))) { | ||
262 | qsel = QSLT_BEACON; | ||
263 | goto out; | ||
264 | } | ||
265 | if (ieee80211_is_mgmt(fc)) { | ||
266 | qsel = QSLT_MGNT; | ||
267 | goto out; | ||
268 | } | ||
269 | switch (mac80211_queue_index) { | ||
270 | case 0: /* VO */ | ||
271 | qsel = QSLT_VO; | ||
272 | RT_TRACE(rtlpriv, COMP_USB, DBG_DMESG, | ||
273 | ("VO queue, set qsel = 0x%x\n", QSLT_VO)); | ||
274 | break; | ||
275 | case 1: /* VI */ | ||
276 | qsel = QSLT_VI; | ||
277 | RT_TRACE(rtlpriv, COMP_USB, DBG_DMESG, | ||
278 | ("VI queue, set qsel = 0x%x\n", QSLT_VI)); | ||
279 | break; | ||
280 | case 3: /* BK */ | ||
281 | qsel = QSLT_BK; | ||
282 | RT_TRACE(rtlpriv, COMP_USB, DBG_DMESG, | ||
283 | ("BK queue, set qsel = 0x%x\n", QSLT_BK)); | ||
284 | break; | ||
285 | case 2: /* BE */ | ||
286 | default: | ||
287 | qsel = QSLT_BE; | ||
288 | RT_TRACE(rtlpriv, COMP_USB, DBG_DMESG, | ||
289 | ("BE queue, set qsel = 0x%x\n", QSLT_BE)); | ||
290 | break; | ||
291 | } | ||
292 | out: | ||
293 | return qsel; | ||
294 | } | ||
295 | |||
296 | /* =============================================================== */ | ||
297 | |||
298 | /*---------------------------------------------------------------------- | ||
299 | * | ||
300 | * Rx handler | ||
301 | * | ||
302 | *---------------------------------------------------------------------- */ | ||
303 | bool rtl92cu_rx_query_desc(struct ieee80211_hw *hw, | ||
304 | struct rtl_stats *stats, | ||
305 | struct ieee80211_rx_status *rx_status, | ||
306 | u8 *p_desc, struct sk_buff *skb) | ||
307 | { | ||
308 | struct rx_fwinfo_92c *p_drvinfo; | ||
309 | struct rx_desc_92c *pdesc = (struct rx_desc_92c *)p_desc; | ||
310 | u32 phystatus = GET_RX_DESC_PHY_STATUS(pdesc); | ||
311 | |||
312 | stats->length = (u16) GET_RX_DESC_PKT_LEN(pdesc); | ||
313 | stats->rx_drvinfo_size = (u8)GET_RX_DESC_DRVINFO_SIZE(pdesc) * | ||
314 | RX_DRV_INFO_SIZE_UNIT; | ||
315 | stats->rx_bufshift = (u8) (GET_RX_DESC_SHIFT(pdesc) & 0x03); | ||
316 | stats->icv = (u16) GET_RX_DESC_ICV(pdesc); | ||
317 | stats->crc = (u16) GET_RX_DESC_CRC32(pdesc); | ||
318 | stats->hwerror = (stats->crc | stats->icv); | ||
319 | stats->decrypted = !GET_RX_DESC_SWDEC(pdesc); | ||
320 | stats->rate = (u8) GET_RX_DESC_RX_MCS(pdesc); | ||
321 | stats->shortpreamble = (u16) GET_RX_DESC_SPLCP(pdesc); | ||
322 | stats->isampdu = (bool) (GET_RX_DESC_PAGGR(pdesc) == 1); | ||
323 | stats->isampdu = (bool) ((GET_RX_DESC_PAGGR(pdesc) == 1) | ||
324 | && (GET_RX_DESC_FAGGR(pdesc) == 1)); | ||
325 | stats->timestamp_low = GET_RX_DESC_TSFL(pdesc); | ||
326 | stats->rx_is40Mhzpacket = (bool) GET_RX_DESC_BW(pdesc); | ||
327 | rx_status->freq = hw->conf.channel->center_freq; | ||
328 | rx_status->band = hw->conf.channel->band; | ||
329 | if (GET_RX_DESC_CRC32(pdesc)) | ||
330 | rx_status->flag |= RX_FLAG_FAILED_FCS_CRC; | ||
331 | if (!GET_RX_DESC_SWDEC(pdesc)) | ||
332 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
333 | if (GET_RX_DESC_BW(pdesc)) | ||
334 | rx_status->flag |= RX_FLAG_40MHZ; | ||
335 | if (GET_RX_DESC_RX_HT(pdesc)) | ||
336 | rx_status->flag |= RX_FLAG_HT; | ||
337 | rx_status->flag |= RX_FLAG_MACTIME_MPDU; | ||
338 | if (stats->decrypted) | ||
339 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
340 | rx_status->rate_idx = _rtl92c_rate_mapping(hw, | ||
341 | (bool)GET_RX_DESC_RX_HT(pdesc), | ||
342 | (u8)GET_RX_DESC_RX_MCS(pdesc), | ||
343 | (bool)GET_RX_DESC_PAGGR(pdesc)); | ||
344 | rx_status->mactime = GET_RX_DESC_TSFL(pdesc); | ||
345 | if (phystatus == true) { | ||
346 | p_drvinfo = (struct rx_fwinfo_92c *)(pdesc + RTL_RX_DESC_SIZE); | ||
347 | rtl92c_translate_rx_signal_stuff(hw, skb, stats, pdesc, | ||
348 | p_drvinfo); | ||
349 | } | ||
350 | /*rx_status->qual = stats->signal; */ | ||
351 | rx_status->signal = stats->rssi + 10; | ||
352 | /*rx_status->noise = -stats->noise; */ | ||
353 | return true; | ||
354 | } | ||
355 | |||
356 | #define RTL_RX_DRV_INFO_UNIT 8 | ||
357 | |||
358 | static void _rtl_rx_process(struct ieee80211_hw *hw, struct sk_buff *skb) | ||
359 | { | ||
360 | struct ieee80211_rx_status *rx_status = | ||
361 | (struct ieee80211_rx_status *)IEEE80211_SKB_RXCB(skb); | ||
362 | u32 skb_len, pkt_len, drvinfo_len; | ||
363 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
364 | u8 *rxdesc; | ||
365 | struct rtl_stats stats = { | ||
366 | .signal = 0, | ||
367 | .noise = -98, | ||
368 | .rate = 0, | ||
369 | }; | ||
370 | struct rx_fwinfo_92c *p_drvinfo; | ||
371 | bool bv; | ||
372 | __le16 fc; | ||
373 | struct ieee80211_hdr *hdr; | ||
374 | |||
375 | memset(rx_status, 0, sizeof(*rx_status)); | ||
376 | rxdesc = skb->data; | ||
377 | skb_len = skb->len; | ||
378 | drvinfo_len = (GET_RX_DESC_DRVINFO_SIZE(rxdesc) * RTL_RX_DRV_INFO_UNIT); | ||
379 | pkt_len = GET_RX_DESC_PKT_LEN(rxdesc); | ||
380 | /* TODO: Error recovery. drop this skb or something. */ | ||
381 | WARN_ON(skb_len < (pkt_len + RTL_RX_DESC_SIZE + drvinfo_len)); | ||
382 | stats.length = (u16) GET_RX_DESC_PKT_LEN(rxdesc); | ||
383 | stats.rx_drvinfo_size = (u8)GET_RX_DESC_DRVINFO_SIZE(rxdesc) * | ||
384 | RX_DRV_INFO_SIZE_UNIT; | ||
385 | stats.rx_bufshift = (u8) (GET_RX_DESC_SHIFT(rxdesc) & 0x03); | ||
386 | stats.icv = (u16) GET_RX_DESC_ICV(rxdesc); | ||
387 | stats.crc = (u16) GET_RX_DESC_CRC32(rxdesc); | ||
388 | stats.hwerror = (stats.crc | stats.icv); | ||
389 | stats.decrypted = !GET_RX_DESC_SWDEC(rxdesc); | ||
390 | stats.rate = (u8) GET_RX_DESC_RX_MCS(rxdesc); | ||
391 | stats.shortpreamble = (u16) GET_RX_DESC_SPLCP(rxdesc); | ||
392 | stats.isampdu = (bool) ((GET_RX_DESC_PAGGR(rxdesc) == 1) | ||
393 | && (GET_RX_DESC_FAGGR(rxdesc) == 1)); | ||
394 | stats.timestamp_low = GET_RX_DESC_TSFL(rxdesc); | ||
395 | stats.rx_is40Mhzpacket = (bool) GET_RX_DESC_BW(rxdesc); | ||
396 | /* TODO: is center_freq changed when doing scan? */ | ||
397 | /* TODO: Shall we add protection or just skip those two step? */ | ||
398 | rx_status->freq = hw->conf.channel->center_freq; | ||
399 | rx_status->band = hw->conf.channel->band; | ||
400 | if (GET_RX_DESC_CRC32(rxdesc)) | ||
401 | rx_status->flag |= RX_FLAG_FAILED_FCS_CRC; | ||
402 | if (!GET_RX_DESC_SWDEC(rxdesc)) | ||
403 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
404 | if (GET_RX_DESC_BW(rxdesc)) | ||
405 | rx_status->flag |= RX_FLAG_40MHZ; | ||
406 | if (GET_RX_DESC_RX_HT(rxdesc)) | ||
407 | rx_status->flag |= RX_FLAG_HT; | ||
408 | /* Data rate */ | ||
409 | rx_status->rate_idx = _rtl92c_rate_mapping(hw, | ||
410 | (bool)GET_RX_DESC_RX_HT(rxdesc), | ||
411 | (u8)GET_RX_DESC_RX_MCS(rxdesc), | ||
412 | (bool)GET_RX_DESC_PAGGR(rxdesc) | ||
413 | ); | ||
414 | /* There is a phy status after this rx descriptor. */ | ||
415 | if (GET_RX_DESC_PHY_STATUS(rxdesc)) { | ||
416 | p_drvinfo = (struct rx_fwinfo_92c *)(rxdesc + RTL_RX_DESC_SIZE); | ||
417 | rtl92c_translate_rx_signal_stuff(hw, skb, &stats, | ||
418 | (struct rx_desc_92c *)rxdesc, p_drvinfo); | ||
419 | } | ||
420 | skb_pull(skb, (drvinfo_len + RTL_RX_DESC_SIZE)); | ||
421 | hdr = (struct ieee80211_hdr *)(skb->data); | ||
422 | fc = hdr->frame_control; | ||
423 | bv = ieee80211_is_probe_resp(fc); | ||
424 | if (bv) | ||
425 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
426 | ("Got probe response frame.\n")); | ||
427 | if (ieee80211_is_beacon(fc)) | ||
428 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
429 | ("Got beacon frame.\n")); | ||
430 | if (ieee80211_is_data(fc)) | ||
431 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Got data frame.\n")); | ||
432 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
433 | ("Fram: fc = 0x%X addr1 = 0x%02X:0x%02X:0x%02X:0x%02X:0x%02X:" | ||
434 | "0x%02X\n", fc, (u32)hdr->addr1[0], (u32)hdr->addr1[1], | ||
435 | (u32)hdr->addr1[2], (u32)hdr->addr1[3], (u32)hdr->addr1[4], | ||
436 | (u32)hdr->addr1[5])); | ||
437 | memcpy(IEEE80211_SKB_RXCB(skb), rx_status, sizeof(*rx_status)); | ||
438 | ieee80211_rx_irqsafe(hw, skb); | ||
439 | } | ||
440 | |||
441 | void rtl8192cu_rx_hdl(struct ieee80211_hw *hw, struct sk_buff * skb) | ||
442 | { | ||
443 | _rtl_rx_process(hw, skb); | ||
444 | } | ||
445 | |||
446 | void rtl8192c_rx_segregate_hdl( | ||
447 | struct ieee80211_hw *hw, | ||
448 | struct sk_buff *skb, | ||
449 | struct sk_buff_head *skb_list) | ||
450 | { | ||
451 | } | ||
452 | |||
453 | /*---------------------------------------------------------------------- | ||
454 | * | ||
455 | * Tx handler | ||
456 | * | ||
457 | *---------------------------------------------------------------------- */ | ||
458 | void rtl8192c_tx_cleanup(struct ieee80211_hw *hw, struct sk_buff *skb) | ||
459 | { | ||
460 | } | ||
461 | |||
462 | int rtl8192c_tx_post_hdl(struct ieee80211_hw *hw, struct urb *urb, | ||
463 | struct sk_buff *skb) | ||
464 | { | ||
465 | return 0; | ||
466 | } | ||
467 | |||
468 | struct sk_buff *rtl8192c_tx_aggregate_hdl(struct ieee80211_hw *hw, | ||
469 | struct sk_buff_head *list) | ||
470 | { | ||
471 | return skb_dequeue(list); | ||
472 | } | ||
473 | |||
474 | /*======================================== trx ===============================*/ | ||
475 | |||
476 | static void _rtl_fill_usb_tx_desc(u8 *txdesc) | ||
477 | { | ||
478 | SET_TX_DESC_OWN(txdesc, 1); | ||
479 | SET_TX_DESC_LAST_SEG(txdesc, 1); | ||
480 | SET_TX_DESC_FIRST_SEG(txdesc, 1); | ||
481 | } | ||
482 | /** | ||
483 | * For HW recovery information | ||
484 | */ | ||
485 | static void _rtl_tx_desc_checksum(u8 *txdesc) | ||
486 | { | ||
487 | u16 *ptr = (u16 *)txdesc; | ||
488 | u16 checksum = 0; | ||
489 | u32 index; | ||
490 | |||
491 | /* Clear first */ | ||
492 | SET_TX_DESC_TX_DESC_CHECKSUM(txdesc, 0); | ||
493 | for (index = 0; index < 16; index++) | ||
494 | checksum = checksum ^ (*(ptr + index)); | ||
495 | SET_TX_DESC_TX_DESC_CHECKSUM(txdesc, checksum); | ||
496 | } | ||
497 | |||
498 | void rtl92cu_tx_fill_desc(struct ieee80211_hw *hw, | ||
499 | struct ieee80211_hdr *hdr, u8 *pdesc_tx, | ||
500 | struct ieee80211_tx_info *info, struct sk_buff *skb, | ||
501 | u8 queue_index, | ||
502 | struct rtl_tcb_desc *tcb_desc) | ||
503 | { | ||
504 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
505 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
506 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
507 | bool defaultadapter = true; | ||
508 | struct ieee80211_sta *sta = info->control.sta = info->control.sta; | ||
509 | u8 *qc = ieee80211_get_qos_ctl(hdr); | ||
510 | u8 tid = qc[0] & IEEE80211_QOS_CTL_TID_MASK; | ||
511 | u16 seq_number; | ||
512 | __le16 fc = hdr->frame_control; | ||
513 | u8 rate_flag = info->control.rates[0].flags; | ||
514 | u16 pktlen = skb->len; | ||
515 | enum rtl_desc_qsel fw_qsel = _rtl8192cu_mq_to_descq(hw, fc, | ||
516 | skb_get_queue_mapping(skb)); | ||
517 | u8 *txdesc; | ||
518 | |||
519 | seq_number = (le16_to_cpu(hdr->seq_ctrl) & IEEE80211_SCTL_SEQ) >> 4; | ||
520 | rtl_get_tcb_desc(hw, info, sta, skb, tcb_desc); | ||
521 | txdesc = (u8 *)skb_push(skb, RTL_TX_HEADER_SIZE); | ||
522 | memset(txdesc, 0, RTL_TX_HEADER_SIZE); | ||
523 | SET_TX_DESC_PKT_SIZE(txdesc, pktlen); | ||
524 | SET_TX_DESC_LINIP(txdesc, 0); | ||
525 | SET_TX_DESC_PKT_OFFSET(txdesc, RTL_DUMMY_OFFSET); | ||
526 | SET_TX_DESC_OFFSET(txdesc, RTL_TX_HEADER_SIZE); | ||
527 | SET_TX_DESC_TX_RATE(txdesc, tcb_desc->hw_rate); | ||
528 | if (tcb_desc->use_shortgi || tcb_desc->use_shortpreamble) | ||
529 | SET_TX_DESC_DATA_SHORTGI(txdesc, 1); | ||
530 | if (mac->tids[tid].agg.agg_state == RTL_AGG_ON && | ||
531 | info->flags & IEEE80211_TX_CTL_AMPDU) { | ||
532 | SET_TX_DESC_AGG_ENABLE(txdesc, 1); | ||
533 | SET_TX_DESC_MAX_AGG_NUM(txdesc, 0x14); | ||
534 | } else { | ||
535 | SET_TX_DESC_AGG_BREAK(txdesc, 1); | ||
536 | } | ||
537 | SET_TX_DESC_SEQ(txdesc, seq_number); | ||
538 | SET_TX_DESC_RTS_ENABLE(txdesc, ((tcb_desc->rts_enable && | ||
539 | !tcb_desc->cts_enable) ? 1 : 0)); | ||
540 | SET_TX_DESC_HW_RTS_ENABLE(txdesc, ((tcb_desc->rts_enable || | ||
541 | tcb_desc->cts_enable) ? 1 : 0)); | ||
542 | SET_TX_DESC_CTS2SELF(txdesc, ((tcb_desc->cts_enable) ? 1 : 0)); | ||
543 | SET_TX_DESC_RTS_STBC(txdesc, ((tcb_desc->rts_stbc) ? 1 : 0)); | ||
544 | SET_TX_DESC_RTS_RATE(txdesc, tcb_desc->rts_rate); | ||
545 | SET_TX_DESC_RTS_BW(txdesc, 0); | ||
546 | SET_TX_DESC_RTS_SC(txdesc, tcb_desc->rts_sc); | ||
547 | SET_TX_DESC_RTS_SHORT(txdesc, | ||
548 | ((tcb_desc->rts_rate <= DESC92C_RATE54M) ? | ||
549 | (tcb_desc->rts_use_shortpreamble ? 1 : 0) | ||
550 | : (tcb_desc->rts_use_shortgi ? 1 : 0))); | ||
551 | if (mac->bw_40) { | ||
552 | if (tcb_desc->packet_bw) { | ||
553 | SET_TX_DESC_DATA_BW(txdesc, 1); | ||
554 | SET_TX_DESC_DATA_SC(txdesc, 3); | ||
555 | } else { | ||
556 | SET_TX_DESC_DATA_BW(txdesc, 0); | ||
557 | if (rate_flag & IEEE80211_TX_RC_DUP_DATA) | ||
558 | SET_TX_DESC_DATA_SC(txdesc, | ||
559 | mac->cur_40_prime_sc); | ||
560 | } | ||
561 | } else { | ||
562 | SET_TX_DESC_DATA_BW(txdesc, 0); | ||
563 | SET_TX_DESC_DATA_SC(txdesc, 0); | ||
564 | } | ||
565 | rcu_read_lock(); | ||
566 | sta = ieee80211_find_sta(mac->vif, mac->bssid); | ||
567 | if (sta) { | ||
568 | u8 ampdu_density = sta->ht_cap.ampdu_density; | ||
569 | SET_TX_DESC_AMPDU_DENSITY(txdesc, ampdu_density); | ||
570 | } | ||
571 | rcu_read_unlock(); | ||
572 | if (info->control.hw_key) { | ||
573 | struct ieee80211_key_conf *keyconf = info->control.hw_key; | ||
574 | switch (keyconf->cipher) { | ||
575 | case WLAN_CIPHER_SUITE_WEP40: | ||
576 | case WLAN_CIPHER_SUITE_WEP104: | ||
577 | case WLAN_CIPHER_SUITE_TKIP: | ||
578 | SET_TX_DESC_SEC_TYPE(txdesc, 0x1); | ||
579 | break; | ||
580 | case WLAN_CIPHER_SUITE_CCMP: | ||
581 | SET_TX_DESC_SEC_TYPE(txdesc, 0x3); | ||
582 | break; | ||
583 | default: | ||
584 | SET_TX_DESC_SEC_TYPE(txdesc, 0x0); | ||
585 | break; | ||
586 | } | ||
587 | } | ||
588 | SET_TX_DESC_PKT_ID(txdesc, 0); | ||
589 | SET_TX_DESC_QUEUE_SEL(txdesc, fw_qsel); | ||
590 | SET_TX_DESC_DATA_RATE_FB_LIMIT(txdesc, 0x1F); | ||
591 | SET_TX_DESC_RTS_RATE_FB_LIMIT(txdesc, 0xF); | ||
592 | SET_TX_DESC_DISABLE_FB(txdesc, 0); | ||
593 | SET_TX_DESC_USE_RATE(txdesc, tcb_desc->use_driver_rate ? 1 : 0); | ||
594 | if (ieee80211_is_data_qos(fc)) { | ||
595 | if (mac->rdg_en) { | ||
596 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, | ||
597 | ("Enable RDG function.\n")); | ||
598 | SET_TX_DESC_RDG_ENABLE(txdesc, 1); | ||
599 | SET_TX_DESC_HTC(txdesc, 1); | ||
600 | } | ||
601 | } | ||
602 | if (rtlpriv->dm.useramask) { | ||
603 | SET_TX_DESC_RATE_ID(txdesc, tcb_desc->ratr_index); | ||
604 | SET_TX_DESC_MACID(txdesc, tcb_desc->mac_id); | ||
605 | } else { | ||
606 | SET_TX_DESC_RATE_ID(txdesc, 0xC + tcb_desc->ratr_index); | ||
607 | SET_TX_DESC_MACID(txdesc, tcb_desc->ratr_index); | ||
608 | } | ||
609 | if ((!ieee80211_is_data_qos(fc)) && ppsc->leisure_ps && | ||
610 | ppsc->fwctrl_lps) { | ||
611 | SET_TX_DESC_HWSEQ_EN(txdesc, 1); | ||
612 | SET_TX_DESC_PKT_ID(txdesc, 8); | ||
613 | if (!defaultadapter) | ||
614 | SET_TX_DESC_QOS(txdesc, 1); | ||
615 | } | ||
616 | if (ieee80211_has_morefrags(fc)) | ||
617 | SET_TX_DESC_MORE_FRAG(txdesc, 1); | ||
618 | if (is_multicast_ether_addr(ieee80211_get_DA(hdr)) || | ||
619 | is_broadcast_ether_addr(ieee80211_get_DA(hdr))) | ||
620 | SET_TX_DESC_BMC(txdesc, 1); | ||
621 | _rtl_fill_usb_tx_desc(txdesc); | ||
622 | _rtl_tx_desc_checksum(txdesc); | ||
623 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, (" %s ==>\n", __func__)); | ||
624 | } | ||
625 | |||
626 | void rtl92cu_fill_fake_txdesc(struct ieee80211_hw *hw, u8 * pDesc, | ||
627 | u32 buffer_len, bool bIsPsPoll) | ||
628 | { | ||
629 | /* Clear all status */ | ||
630 | memset(pDesc, 0, RTL_TX_HEADER_SIZE); | ||
631 | SET_TX_DESC_FIRST_SEG(pDesc, 1); /* bFirstSeg; */ | ||
632 | SET_TX_DESC_LAST_SEG(pDesc, 1); /* bLastSeg; */ | ||
633 | SET_TX_DESC_OFFSET(pDesc, RTL_TX_HEADER_SIZE); /* Offset = 32 */ | ||
634 | SET_TX_DESC_PKT_SIZE(pDesc, buffer_len); /* Buffer size + command hdr */ | ||
635 | SET_TX_DESC_QUEUE_SEL(pDesc, QSLT_MGNT); /* Fixed queue of Mgnt queue */ | ||
636 | /* Set NAVUSEHDR to prevent Ps-poll AId filed to be changed to error | ||
637 | * vlaue by Hw. */ | ||
638 | if (bIsPsPoll) { | ||
639 | SET_TX_DESC_NAV_USE_HDR(pDesc, 1); | ||
640 | } else { | ||
641 | SET_TX_DESC_HWSEQ_EN(pDesc, 1); /* Hw set sequence number */ | ||
642 | SET_TX_DESC_PKT_ID(pDesc, 0x100); /* set bit3 to 1. */ | ||
643 | } | ||
644 | SET_TX_DESC_USE_RATE(pDesc, 1); /* use data rate which is set by Sw */ | ||
645 | SET_TX_DESC_OWN(pDesc, 1); | ||
646 | SET_TX_DESC_TX_RATE(pDesc, DESC92C_RATE1M); | ||
647 | _rtl_tx_desc_checksum(pDesc); | ||
648 | } | ||
649 | |||
650 | void rtl92cu_tx_fill_cmddesc(struct ieee80211_hw *hw, | ||
651 | u8 *pdesc, bool firstseg, | ||
652 | bool lastseg, struct sk_buff *skb) | ||
653 | { | ||
654 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
655 | u8 fw_queue = QSLT_BEACON; | ||
656 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)(skb->data); | ||
657 | __le16 fc = hdr->frame_control; | ||
658 | |||
659 | memset((void *)pdesc, 0, RTL_TX_HEADER_SIZE); | ||
660 | if (firstseg) | ||
661 | SET_TX_DESC_OFFSET(pdesc, RTL_TX_HEADER_SIZE); | ||
662 | SET_TX_DESC_TX_RATE(pdesc, DESC92C_RATE1M); | ||
663 | SET_TX_DESC_SEQ(pdesc, 0); | ||
664 | SET_TX_DESC_LINIP(pdesc, 0); | ||
665 | SET_TX_DESC_QUEUE_SEL(pdesc, fw_queue); | ||
666 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
667 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
668 | SET_TX_DESC_RATE_ID(pdesc, 7); | ||
669 | SET_TX_DESC_MACID(pdesc, 0); | ||
670 | SET_TX_DESC_OWN(pdesc, 1); | ||
671 | SET_TX_DESC_PKT_SIZE((u8 *) pdesc, (u16) (skb->len)); | ||
672 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
673 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
674 | SET_TX_DESC_OFFSET(pdesc, 0x20); | ||
675 | SET_TX_DESC_USE_RATE(pdesc, 1); | ||
676 | if (!ieee80211_is_data_qos(fc)) { | ||
677 | SET_TX_DESC_HWSEQ_EN(pdesc, 1); | ||
678 | SET_TX_DESC_PKT_ID(pdesc, 8); | ||
679 | } | ||
680 | RT_PRINT_DATA(rtlpriv, COMP_CMD, DBG_LOUD, "H2C Tx Cmd Content\n", | ||
681 | pdesc, RTL_TX_DESC_SIZE); | ||
682 | } | ||
683 | |||
684 | bool rtl92cu_cmd_send_packet(struct ieee80211_hw *hw, struct sk_buff *skb) | ||
685 | { | ||
686 | return true; | ||
687 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192cu/trx.h b/drivers/net/wireless/rtlwifi/rtl8192cu/trx.h new file mode 100644 index 000000000000..53de5f66e242 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192cu/trx.h | |||
@@ -0,0 +1,433 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL92CU_TRX_H__ | ||
31 | #define __RTL92CU_TRX_H__ | ||
32 | |||
33 | #define RTL92C_USB_BULK_IN_NUM 1 | ||
34 | #define RTL92C_NUM_RX_URBS 8 | ||
35 | #define RTL92C_NUM_TX_URBS 32 | ||
36 | |||
37 | #define RTL92C_SIZE_MAX_RX_BUFFER 15360 /* 8192 */ | ||
38 | #define RX_DRV_INFO_SIZE_UNIT 8 | ||
39 | |||
40 | #define RTL_AGG_ON 1 | ||
41 | |||
42 | enum usb_rx_agg_mode { | ||
43 | USB_RX_AGG_DISABLE, | ||
44 | USB_RX_AGG_DMA, | ||
45 | USB_RX_AGG_USB, | ||
46 | USB_RX_AGG_DMA_USB | ||
47 | }; | ||
48 | |||
49 | #define TX_SELE_HQ BIT(0) /* High Queue */ | ||
50 | #define TX_SELE_LQ BIT(1) /* Low Queue */ | ||
51 | #define TX_SELE_NQ BIT(2) /* Normal Queue */ | ||
52 | |||
53 | #define RTL_USB_TX_AGG_NUM_DESC 5 | ||
54 | |||
55 | #define RTL_USB_RX_AGG_PAGE_NUM 4 | ||
56 | #define RTL_USB_RX_AGG_PAGE_TIMEOUT 3 | ||
57 | |||
58 | #define RTL_USB_RX_AGG_BLOCK_NUM 5 | ||
59 | #define RTL_USB_RX_AGG_BLOCK_TIMEOUT 3 | ||
60 | |||
61 | /*======================== rx status =========================================*/ | ||
62 | |||
63 | struct rx_drv_info_92c { | ||
64 | /* | ||
65 | * Driver info contain PHY status and other variabel size info | ||
66 | * PHY Status content as below | ||
67 | */ | ||
68 | |||
69 | /* DWORD 0 */ | ||
70 | u8 gain_trsw[4]; | ||
71 | |||
72 | /* DWORD 1 */ | ||
73 | u8 pwdb_all; | ||
74 | u8 cfosho[4]; | ||
75 | |||
76 | /* DWORD 2 */ | ||
77 | u8 cfotail[4]; | ||
78 | |||
79 | /* DWORD 3 */ | ||
80 | s8 rxevm[2]; | ||
81 | s8 rxsnr[4]; | ||
82 | |||
83 | /* DWORD 4 */ | ||
84 | u8 pdsnr[2]; | ||
85 | |||
86 | /* DWORD 5 */ | ||
87 | u8 csi_current[2]; | ||
88 | u8 csi_target[2]; | ||
89 | |||
90 | /* DWORD 6 */ | ||
91 | u8 sigevm; | ||
92 | u8 max_ex_pwr; | ||
93 | u8 ex_intf_flag:1; | ||
94 | u8 sgi_en:1; | ||
95 | u8 rxsc:2; | ||
96 | u8 reserve:4; | ||
97 | } __packed; | ||
98 | |||
99 | /* Define a macro that takes a le32 word, converts it to host ordering, | ||
100 | * right shifts by a specified count, creates a mask of the specified | ||
101 | * bit count, and extracts that number of bits. | ||
102 | */ | ||
103 | |||
104 | #define SHIFT_AND_MASK_LE(__pdesc, __shift, __bits) \ | ||
105 | ((le32_to_cpu(*(((__le32 *)(__pdesc)))) >> (__shift)) & \ | ||
106 | BIT_LEN_MASK_32(__bits)) | ||
107 | |||
108 | /* Define a macro that clears a bit field in an le32 word and | ||
109 | * sets the specified value into that bit field. The resulting | ||
110 | * value remains in le32 ordering; however, it is properly converted | ||
111 | * to host ordering for the clear and set operations before conversion | ||
112 | * back to le32. | ||
113 | */ | ||
114 | |||
115 | #define SET_BITS_OFFSET_LE(__pdesc, __shift, __len, __val) \ | ||
116 | (*(__le32 *)(__pdesc) = \ | ||
117 | (cpu_to_le32((le32_to_cpu(*((__le32 *)(__pdesc))) & \ | ||
118 | (~(BIT_OFFSET_LEN_MASK_32((__shift), __len)))) | \ | ||
119 | (((u32)(__val) & BIT_LEN_MASK_32(__len)) << (__shift))))); | ||
120 | |||
121 | /* macros to read various fields in RX descriptor */ | ||
122 | |||
123 | /* DWORD 0 */ | ||
124 | #define GET_RX_DESC_PKT_LEN(__rxdesc) \ | ||
125 | SHIFT_AND_MASK_LE((__rxdesc), 0, 14) | ||
126 | #define GET_RX_DESC_CRC32(__rxdesc) \ | ||
127 | SHIFT_AND_MASK_LE(__rxdesc, 14, 1) | ||
128 | #define GET_RX_DESC_ICV(__rxdesc) \ | ||
129 | SHIFT_AND_MASK_LE(__rxdesc, 15, 1) | ||
130 | #define GET_RX_DESC_DRVINFO_SIZE(__rxdesc) \ | ||
131 | SHIFT_AND_MASK_LE(__rxdesc, 16, 4) | ||
132 | #define GET_RX_DESC_SECURITY(__rxdesc) \ | ||
133 | SHIFT_AND_MASK_LE(__rxdesc, 20, 3) | ||
134 | #define GET_RX_DESC_QOS(__rxdesc) \ | ||
135 | SHIFT_AND_MASK_LE(__rxdesc, 23, 1) | ||
136 | #define GET_RX_DESC_SHIFT(__rxdesc) \ | ||
137 | SHIFT_AND_MASK_LE(__rxdesc, 24, 2) | ||
138 | #define GET_RX_DESC_PHY_STATUS(__rxdesc) \ | ||
139 | SHIFT_AND_MASK_LE(__rxdesc, 26, 1) | ||
140 | #define GET_RX_DESC_SWDEC(__rxdesc) \ | ||
141 | SHIFT_AND_MASK_LE(__rxdesc, 27, 1) | ||
142 | #define GET_RX_DESC_LAST_SEG(__rxdesc) \ | ||
143 | SHIFT_AND_MASK_LE(__rxdesc, 28, 1) | ||
144 | #define GET_RX_DESC_FIRST_SEG(__rxdesc) \ | ||
145 | SHIFT_AND_MASK_LE(__rxdesc, 29, 1) | ||
146 | #define GET_RX_DESC_EOR(__rxdesc) \ | ||
147 | SHIFT_AND_MASK_LE(__rxdesc, 30, 1) | ||
148 | #define GET_RX_DESC_OWN(__rxdesc) \ | ||
149 | SHIFT_AND_MASK_LE(__rxdesc, 31, 1) | ||
150 | |||
151 | /* DWORD 1 */ | ||
152 | #define GET_RX_DESC_MACID(__rxdesc) \ | ||
153 | SHIFT_AND_MASK_LE(__rxdesc+4, 0, 5) | ||
154 | #define GET_RX_DESC_TID(__rxdesc) \ | ||
155 | SHIFT_AND_MASK_LE(__rxdesc+4, 5, 4) | ||
156 | #define GET_RX_DESC_PAGGR(__rxdesc) \ | ||
157 | SHIFT_AND_MASK_LE(__rxdesc+4, 14, 1) | ||
158 | #define GET_RX_DESC_FAGGR(__rxdesc) \ | ||
159 | SHIFT_AND_MASK_LE(__rxdesc+4, 15, 1) | ||
160 | #define GET_RX_DESC_A1_FIT(__rxdesc) \ | ||
161 | SHIFT_AND_MASK_LE(__rxdesc+4, 16, 4) | ||
162 | #define GET_RX_DESC_A2_FIT(__rxdesc) \ | ||
163 | SHIFT_AND_MASK_LE(__rxdesc+4, 20, 4) | ||
164 | #define GET_RX_DESC_PAM(__rxdesc) \ | ||
165 | SHIFT_AND_MASK_LE(__rxdesc+4, 24, 1) | ||
166 | #define GET_RX_DESC_PWR(__rxdesc) \ | ||
167 | SHIFT_AND_MASK_LE(__rxdesc+4, 25, 1) | ||
168 | #define GET_RX_DESC_MORE_DATA(__rxdesc) \ | ||
169 | SHIFT_AND_MASK_LE(__rxdesc+4, 26, 1) | ||
170 | #define GET_RX_DESC_MORE_FRAG(__rxdesc) \ | ||
171 | SHIFT_AND_MASK_LE(__rxdesc+4, 27, 1) | ||
172 | #define GET_RX_DESC_TYPE(__rxdesc) \ | ||
173 | SHIFT_AND_MASK_LE(__rxdesc+4, 28, 2) | ||
174 | #define GET_RX_DESC_MC(__rxdesc) \ | ||
175 | SHIFT_AND_MASK_LE(__rxdesc+4, 30, 1) | ||
176 | #define GET_RX_DESC_BC(__rxdesc) \ | ||
177 | SHIFT_AND_MASK_LE(__rxdesc+4, 31, 1) | ||
178 | |||
179 | /* DWORD 2 */ | ||
180 | #define GET_RX_DESC_SEQ(__rxdesc) \ | ||
181 | SHIFT_AND_MASK_LE(__rxdesc+8, 0, 12) | ||
182 | #define GET_RX_DESC_FRAG(__rxdesc) \ | ||
183 | SHIFT_AND_MASK_LE(__rxdesc+8, 12, 4) | ||
184 | #define GET_RX_DESC_USB_AGG_PKTNUM(__rxdesc) \ | ||
185 | SHIFT_AND_MASK_LE(__rxdesc+8, 16, 8) | ||
186 | #define GET_RX_DESC_NEXT_IND(__rxdesc) \ | ||
187 | SHIFT_AND_MASK_LE(__rxdesc+8, 30, 1) | ||
188 | |||
189 | /* DWORD 3 */ | ||
190 | #define GET_RX_DESC_RX_MCS(__rxdesc) \ | ||
191 | SHIFT_AND_MASK_LE(__rxdesc+12, 0, 6) | ||
192 | #define GET_RX_DESC_RX_HT(__rxdesc) \ | ||
193 | SHIFT_AND_MASK_LE(__rxdesc+12, 6, 1) | ||
194 | #define GET_RX_DESC_AMSDU(__rxdesc) \ | ||
195 | SHIFT_AND_MASK_LE(__rxdesc+12, 7, 1) | ||
196 | #define GET_RX_DESC_SPLCP(__rxdesc) \ | ||
197 | SHIFT_AND_MASK_LE(__rxdesc+12, 8, 1) | ||
198 | #define GET_RX_DESC_BW(__rxdesc) \ | ||
199 | SHIFT_AND_MASK_LE(__rxdesc+12, 9, 1) | ||
200 | #define GET_RX_DESC_HTC(__rxdesc) \ | ||
201 | SHIFT_AND_MASK_LE(__rxdesc+12, 10, 1) | ||
202 | #define GET_RX_DESC_TCP_CHK_RPT(__rxdesc) \ | ||
203 | SHIFT_AND_MASK_LE(__rxdesc+12, 11, 1) | ||
204 | #define GET_RX_DESC_IP_CHK_RPT(__rxdesc) \ | ||
205 | SHIFT_AND_MASK_LE(__rxdesc+12, 12, 1) | ||
206 | #define GET_RX_DESC_TCP_CHK_VALID(__rxdesc) \ | ||
207 | SHIFT_AND_MASK_LE(__rxdesc+12, 13, 1) | ||
208 | #define GET_RX_DESC_HWPC_ERR(__rxdesc) \ | ||
209 | SHIFT_AND_MASK_LE(__rxdesc+12, 14, 1) | ||
210 | #define GET_RX_DESC_HWPC_IND(__rxdesc) \ | ||
211 | SHIFT_AND_MASK_LE(__rxdesc+12, 15, 1) | ||
212 | #define GET_RX_DESC_IV0(__rxdesc) \ | ||
213 | SHIFT_AND_MASK_LE(__rxdesc+12, 16, 16) | ||
214 | |||
215 | /* DWORD 4 */ | ||
216 | #define GET_RX_DESC_IV1(__rxdesc) \ | ||
217 | SHIFT_AND_MASK_LE(__rxdesc+16, 0, 32) | ||
218 | |||
219 | /* DWORD 5 */ | ||
220 | #define GET_RX_DESC_TSFL(__rxdesc) \ | ||
221 | SHIFT_AND_MASK_LE(__rxdesc+20, 0, 32) | ||
222 | |||
223 | /*======================= tx desc ============================================*/ | ||
224 | |||
225 | /* macros to set various fields in TX descriptor */ | ||
226 | |||
227 | /* Dword 0 */ | ||
228 | #define SET_TX_DESC_PKT_SIZE(__txdesc, __value) \ | ||
229 | SET_BITS_OFFSET_LE(__txdesc, 0, 16, __value) | ||
230 | #define SET_TX_DESC_OFFSET(__txdesc, __value) \ | ||
231 | SET_BITS_OFFSET_LE(__txdesc, 16, 8, __value) | ||
232 | #define SET_TX_DESC_BMC(__txdesc, __value) \ | ||
233 | SET_BITS_OFFSET_LE(__txdesc, 24, 1, __value) | ||
234 | #define SET_TX_DESC_HTC(__txdesc, __value) \ | ||
235 | SET_BITS_OFFSET_LE(__txdesc, 25, 1, __value) | ||
236 | #define SET_TX_DESC_LAST_SEG(__txdesc, __value) \ | ||
237 | SET_BITS_OFFSET_LE(__txdesc, 26, 1, __value) | ||
238 | #define SET_TX_DESC_FIRST_SEG(__txdesc, __value) \ | ||
239 | SET_BITS_OFFSET_LE(__txdesc, 27, 1, __value) | ||
240 | #define SET_TX_DESC_LINIP(__txdesc, __value) \ | ||
241 | SET_BITS_OFFSET_LE(__txdesc, 28, 1, __value) | ||
242 | #define SET_TX_DESC_NO_ACM(__txdesc, __value) \ | ||
243 | SET_BITS_OFFSET_LE(__txdesc, 29, 1, __value) | ||
244 | #define SET_TX_DESC_GF(__txdesc, __value) \ | ||
245 | SET_BITS_OFFSET_LE(__txdesc, 30, 1, __value) | ||
246 | #define SET_TX_DESC_OWN(__txdesc, __value) \ | ||
247 | SET_BITS_OFFSET_LE(__txdesc, 31, 1, __value) | ||
248 | |||
249 | |||
250 | /* Dword 1 */ | ||
251 | #define SET_TX_DESC_MACID(__txdesc, __value) \ | ||
252 | SET_BITS_OFFSET_LE(__txdesc+4, 0, 5, __value) | ||
253 | #define SET_TX_DESC_AGG_ENABLE(__txdesc, __value) \ | ||
254 | SET_BITS_OFFSET_LE(__txdesc+4, 5, 1, __value) | ||
255 | #define SET_TX_DESC_AGG_BREAK(__txdesc, __value) \ | ||
256 | SET_BITS_OFFSET_LE(__txdesc+4, 6, 1, __value) | ||
257 | #define SET_TX_DESC_RDG_ENABLE(__txdesc, __value) \ | ||
258 | SET_BITS_OFFSET_LE(__txdesc+4, 7, 1, __value) | ||
259 | #define SET_TX_DESC_QUEUE_SEL(__txdesc, __value) \ | ||
260 | SET_BITS_OFFSET_LE(__txdesc+4, 8, 5, __value) | ||
261 | #define SET_TX_DESC_RDG_NAV_EXT(__txdesc, __value) \ | ||
262 | SET_BITS_OFFSET_LE(__txdesc+4, 13, 1, __value) | ||
263 | #define SET_TX_DESC_LSIG_TXOP_EN(__txdesc, __value) \ | ||
264 | SET_BITS_OFFSET_LE(__txdesc+4, 14, 1, __value) | ||
265 | #define SET_TX_DESC_PIFS(__txdesc, __value) \ | ||
266 | SET_BITS_OFFSET_LE(__txdesc+4, 15, 1, __value) | ||
267 | #define SET_TX_DESC_RATE_ID(__txdesc, __value) \ | ||
268 | SET_BITS_OFFSET_LE(__txdesc+4, 16, 4, __value) | ||
269 | #define SET_TX_DESC_RA_BRSR_ID(__txdesc, __value) \ | ||
270 | SET_BITS_OFFSET_LE(__txdesc+4, 16, 4, __value) | ||
271 | #define SET_TX_DESC_NAV_USE_HDR(__txdesc, __value) \ | ||
272 | SET_BITS_OFFSET_LE(__txdesc+4, 20, 1, __value) | ||
273 | #define SET_TX_DESC_EN_DESC_ID(__txdesc, __value) \ | ||
274 | SET_BITS_OFFSET_LE(__txdesc+4, 21, 1, __value) | ||
275 | #define SET_TX_DESC_SEC_TYPE(__txdesc, __value) \ | ||
276 | SET_BITS_OFFSET_LE(__txdesc+4, 22, 2, __value) | ||
277 | #define SET_TX_DESC_PKT_OFFSET(__txdesc, __value) \ | ||
278 | SET_BITS_OFFSET_LE(__txdesc+4, 26, 5, __value) | ||
279 | |||
280 | /* Dword 2 */ | ||
281 | #define SET_TX_DESC_RTS_RC(__txdesc, __value) \ | ||
282 | SET_BITS_OFFSET_LE(__txdesc+8, 0, 6, __value) | ||
283 | #define SET_TX_DESC_DATA_RC(__txdesc, __value) \ | ||
284 | SET_BITS_OFFSET_LE(__txdesc+8, 6, 6, __value) | ||
285 | #define SET_TX_DESC_BAR_RTY_TH(__txdesc, __value) \ | ||
286 | SET_BITS_OFFSET_LE(__txdesc+8, 14, 2, __value) | ||
287 | #define SET_TX_DESC_MORE_FRAG(__txdesc, __value) \ | ||
288 | SET_BITS_OFFSET_LE(__txdesc+8, 17, 1, __value) | ||
289 | #define SET_TX_DESC_RAW(__txdesc, __value) \ | ||
290 | SET_BITS_OFFSET_LE(__txdesc+8, 18, 1, __value) | ||
291 | #define SET_TX_DESC_CCX(__txdesc, __value) \ | ||
292 | SET_BITS_OFFSET_LE(__txdesc+8, 19, 1, __value) | ||
293 | #define SET_TX_DESC_AMPDU_DENSITY(__txdesc, __value) \ | ||
294 | SET_BITS_OFFSET_LE(__txdesc+8, 20, 3, __value) | ||
295 | #define SET_TX_DESC_ANTSEL_A(__txdesc, __value) \ | ||
296 | SET_BITS_OFFSET_LE(__txdesc+8, 24, 1, __value) | ||
297 | #define SET_TX_DESC_ANTSEL_B(__txdesc, __value) \ | ||
298 | SET_BITS_OFFSET_LE(__txdesc+8, 25, 1, __value) | ||
299 | #define SET_TX_DESC_TX_ANT_CCK(__txdesc, __value) \ | ||
300 | SET_BITS_OFFSET_LE(__txdesc+8, 26, 2, __value) | ||
301 | #define SET_TX_DESC_TX_ANTL(__txdesc, __value) \ | ||
302 | SET_BITS_OFFSET_LE(__txdesc+8, 28, 2, __value) | ||
303 | #define SET_TX_DESC_TX_ANT_HT(__txdesc, __value) \ | ||
304 | SET_BITS_OFFSET_LE(__txdesc+8, 30, 2, __value) | ||
305 | |||
306 | /* Dword 3 */ | ||
307 | #define SET_TX_DESC_NEXT_HEAP_PAGE(__txdesc, __value) \ | ||
308 | SET_BITS_OFFSET_LE(__txdesc+12, 0, 8, __value) | ||
309 | #define SET_TX_DESC_TAIL_PAGE(__txdesc, __value) \ | ||
310 | SET_BITS_OFFSET_LE(__txdesc+12, 8, 8, __value) | ||
311 | #define SET_TX_DESC_SEQ(__txdesc, __value) \ | ||
312 | SET_BITS_OFFSET_LE(__txdesc+12, 16, 12, __value) | ||
313 | #define SET_TX_DESC_PKT_ID(__txdesc, __value) \ | ||
314 | SET_BITS_OFFSET_LE(__txdesc+12, 28, 4, __value) | ||
315 | |||
316 | /* Dword 4 */ | ||
317 | #define SET_TX_DESC_RTS_RATE(__txdesc, __value) \ | ||
318 | SET_BITS_OFFSET_LE(__txdesc+16, 0, 5, __value) | ||
319 | #define SET_TX_DESC_AP_DCFE(__txdesc, __value) \ | ||
320 | SET_BITS_OFFSET_LE(__txdesc+16, 5, 1, __value) | ||
321 | #define SET_TX_DESC_QOS(__txdesc, __value) \ | ||
322 | SET_BITS_OFFSET_LE(__txdesc+16, 6, 1, __value) | ||
323 | #define SET_TX_DESC_HWSEQ_EN(__txdesc, __value) \ | ||
324 | SET_BITS_OFFSET_LE(__txdesc+16, 7, 1, __value) | ||
325 | #define SET_TX_DESC_USE_RATE(__txdesc, __value) \ | ||
326 | SET_BITS_OFFSET_LE(__txdesc+16, 8, 1, __value) | ||
327 | #define SET_TX_DESC_DISABLE_RTS_FB(__txdesc, __value) \ | ||
328 | SET_BITS_OFFSET_LE(__txdesc+16, 9, 1, __value) | ||
329 | #define SET_TX_DESC_DISABLE_FB(__txdesc, __value) \ | ||
330 | SET_BITS_OFFSET_LE(__txdesc+16, 10, 1, __value) | ||
331 | #define SET_TX_DESC_CTS2SELF(__txdesc, __value) \ | ||
332 | SET_BITS_OFFSET_LE(__txdesc+16, 11, 1, __value) | ||
333 | #define SET_TX_DESC_RTS_ENABLE(__txdesc, __value) \ | ||
334 | SET_BITS_OFFSET_LE(__txdesc+16, 12, 1, __value) | ||
335 | #define SET_TX_DESC_HW_RTS_ENABLE(__txdesc, __value) \ | ||
336 | SET_BITS_OFFSET_LE(__txdesc+16, 13, 1, __value) | ||
337 | #define SET_TX_DESC_WAIT_DCTS(__txdesc, __value) \ | ||
338 | SET_BITS_OFFSET_LE(__txdesc+16, 18, 1, __value) | ||
339 | #define SET_TX_DESC_CTS2AP_EN(__txdesc, __value) \ | ||
340 | SET_BITS_OFFSET_LE(__txdesc+16, 19, 1, __value) | ||
341 | #define SET_TX_DESC_DATA_SC(__txdesc, __value) \ | ||
342 | SET_BITS_OFFSET_LE(__txdesc+16, 20, 2, __value) | ||
343 | #define SET_TX_DESC_DATA_STBC(__txdesc, __value) \ | ||
344 | SET_BITS_OFFSET_LE(__txdesc+16, 22, 2, __value) | ||
345 | #define SET_TX_DESC_DATA_SHORT(__txdesc, __value) \ | ||
346 | SET_BITS_OFFSET_LE(__txdesc+16, 24, 1, __value) | ||
347 | #define SET_TX_DESC_DATA_BW(__txdesc, __value) \ | ||
348 | SET_BITS_OFFSET_LE(__txdesc+16, 25, 1, __value) | ||
349 | #define SET_TX_DESC_RTS_SHORT(__txdesc, __value) \ | ||
350 | SET_BITS_OFFSET_LE(__txdesc+16, 26, 1, __value) | ||
351 | #define SET_TX_DESC_RTS_BW(__txdesc, __value) \ | ||
352 | SET_BITS_OFFSET_LE(__txdesc+16, 27, 1, __value) | ||
353 | #define SET_TX_DESC_RTS_SC(__txdesc, __value) \ | ||
354 | SET_BITS_OFFSET_LE(__txdesc+16, 28, 2, __value) | ||
355 | #define SET_TX_DESC_RTS_STBC(__txdesc, __value) \ | ||
356 | SET_BITS_OFFSET_LE(__txdesc+16, 30, 2, __value) | ||
357 | |||
358 | /* Dword 5 */ | ||
359 | #define SET_TX_DESC_TX_RATE(__pdesc, __val) \ | ||
360 | SET_BITS_OFFSET_LE(__pdesc+20, 0, 6, __val) | ||
361 | #define SET_TX_DESC_DATA_SHORTGI(__pdesc, __val) \ | ||
362 | SET_BITS_OFFSET_LE(__pdesc+20, 6, 1, __val) | ||
363 | #define SET_TX_DESC_CCX_TAG(__pdesc, __val) \ | ||
364 | SET_BITS_OFFSET_LE(__pdesc+20, 7, 1, __val) | ||
365 | #define SET_TX_DESC_DATA_RATE_FB_LIMIT(__txdesc, __value) \ | ||
366 | SET_BITS_OFFSET_LE(__txdesc+20, 8, 5, __value) | ||
367 | #define SET_TX_DESC_RTS_RATE_FB_LIMIT(__txdesc, __value) \ | ||
368 | SET_BITS_OFFSET_LE(__txdesc+20, 13, 4, __value) | ||
369 | #define SET_TX_DESC_RETRY_LIMIT_ENABLE(__txdesc, __value) \ | ||
370 | SET_BITS_OFFSET_LE(__txdesc+20, 17, 1, __value) | ||
371 | #define SET_TX_DESC_DATA_RETRY_LIMIT(__txdesc, __value) \ | ||
372 | SET_BITS_OFFSET_LE(__txdesc+20, 18, 6, __value) | ||
373 | #define SET_TX_DESC_USB_TXAGG_NUM(__txdesc, __value) \ | ||
374 | SET_BITS_OFFSET_LE(__txdesc+20, 24, 8, __value) | ||
375 | |||
376 | /* Dword 6 */ | ||
377 | #define SET_TX_DESC_TXAGC_A(__txdesc, __value) \ | ||
378 | SET_BITS_OFFSET_LE(__txdesc+24, 0, 5, __value) | ||
379 | #define SET_TX_DESC_TXAGC_B(__txdesc, __value) \ | ||
380 | SET_BITS_OFFSET_LE(__txdesc+24, 5, 5, __value) | ||
381 | #define SET_TX_DESC_USB_MAX_LEN(__txdesc, __value) \ | ||
382 | SET_BITS_OFFSET_LE(__txdesc+24, 10, 1, __value) | ||
383 | #define SET_TX_DESC_MAX_AGG_NUM(__txdesc, __value) \ | ||
384 | SET_BITS_OFFSET_LE(__txdesc+24, 11, 5, __value) | ||
385 | #define SET_TX_DESC_MCSG1_MAX_LEN(__txdesc, __value) \ | ||
386 | SET_BITS_OFFSET_LE(__txdesc+24, 16, 4, __value) | ||
387 | #define SET_TX_DESC_MCSG2_MAX_LEN(__txdesc, __value) \ | ||
388 | SET_BITS_OFFSET_LE(__txdesc+24, 20, 4, __value) | ||
389 | #define SET_TX_DESC_MCSG3_MAX_LEN(__txdesc, __value) \ | ||
390 | SET_BITS_OFFSET_LE(__txdesc+24, 24, 4, __value) | ||
391 | #define SET_TX_DESC_MCSG7_MAX_LEN(__txdesc, __value) \ | ||
392 | SET_BITS_OFFSET_LE(__txdesc+24, 28, 4, __value) | ||
393 | |||
394 | /* Dword 7 */ | ||
395 | #define SET_TX_DESC_TX_DESC_CHECKSUM(__txdesc, __value) \ | ||
396 | SET_BITS_OFFSET_LE(__txdesc+28, 0, 16, __value) | ||
397 | #define SET_TX_DESC_MCSG4_MAX_LEN(__txdesc, __value) \ | ||
398 | SET_BITS_OFFSET_LE(__txdesc+28, 16, 4, __value) | ||
399 | #define SET_TX_DESC_MCSG5_MAX_LEN(__txdesc, __value) \ | ||
400 | SET_BITS_OFFSET_LE(__txdesc+28, 20, 4, __value) | ||
401 | #define SET_TX_DESC_MCSG6_MAX_LEN(__txdesc, __value) \ | ||
402 | SET_BITS_OFFSET_LE(__txdesc+28, 24, 4, __value) | ||
403 | #define SET_TX_DESC_MCSG15_MAX_LEN(__txdesc, __value) \ | ||
404 | SET_BITS_OFFSET_LE(__txdesc+28, 28, 4, __value) | ||
405 | |||
406 | |||
407 | int rtl8192cu_endpoint_mapping(struct ieee80211_hw *hw); | ||
408 | u16 rtl8192cu_mq_to_hwq(__le16 fc, u16 mac80211_queue_index); | ||
409 | bool rtl92cu_rx_query_desc(struct ieee80211_hw *hw, | ||
410 | struct rtl_stats *stats, | ||
411 | struct ieee80211_rx_status *rx_status, | ||
412 | u8 *p_desc, struct sk_buff *skb); | ||
413 | void rtl8192cu_rx_hdl(struct ieee80211_hw *hw, struct sk_buff * skb); | ||
414 | void rtl8192c_rx_segregate_hdl(struct ieee80211_hw *, struct sk_buff *, | ||
415 | struct sk_buff_head *); | ||
416 | void rtl8192c_tx_cleanup(struct ieee80211_hw *hw, struct sk_buff *skb); | ||
417 | int rtl8192c_tx_post_hdl(struct ieee80211_hw *hw, struct urb *urb, | ||
418 | struct sk_buff *skb); | ||
419 | struct sk_buff *rtl8192c_tx_aggregate_hdl(struct ieee80211_hw *, | ||
420 | struct sk_buff_head *); | ||
421 | void rtl92cu_tx_fill_desc(struct ieee80211_hw *hw, | ||
422 | struct ieee80211_hdr *hdr, u8 *pdesc_tx, | ||
423 | struct ieee80211_tx_info *info, struct sk_buff *skb, | ||
424 | u8 queue_index, | ||
425 | struct rtl_tcb_desc *tcb_desc); | ||
426 | void rtl92cu_fill_fake_txdesc(struct ieee80211_hw *hw, u8 * pDesc, | ||
427 | u32 buffer_len, bool bIsPsPoll); | ||
428 | void rtl92cu_tx_fill_cmddesc(struct ieee80211_hw *hw, | ||
429 | u8 *pdesc, bool b_firstseg, | ||
430 | bool b_lastseg, struct sk_buff *skb); | ||
431 | bool rtl92cu_cmd_send_packet(struct ieee80211_hw *hw, struct sk_buff *skb); | ||
432 | |||
433 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/Makefile b/drivers/net/wireless/rtlwifi/rtl8192se/Makefile new file mode 100644 index 000000000000..b7eb13819cbc --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/Makefile | |||
@@ -0,0 +1,15 @@ | |||
1 | rtl8192se-objs := \ | ||
2 | dm.o \ | ||
3 | fw.o \ | ||
4 | hw.o \ | ||
5 | led.o \ | ||
6 | phy.o \ | ||
7 | rf.o \ | ||
8 | sw.o \ | ||
9 | table.o \ | ||
10 | trx.o | ||
11 | |||
12 | obj-$(CONFIG_RTL8192SE) += rtl8192se.o | ||
13 | |||
14 | ccflags-y += -D__CHECK_ENDIAN__ | ||
15 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/def.h b/drivers/net/wireless/rtlwifi/rtl8192se/def.h new file mode 100644 index 000000000000..69828f2b3fab --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/def.h | |||
@@ -0,0 +1,598 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_92S_DEF_H__ | ||
30 | #define __REALTEK_92S_DEF_H__ | ||
31 | |||
32 | #define RX_MPDU_QUEUE 0 | ||
33 | #define RX_CMD_QUEUE 1 | ||
34 | #define RX_MAX_QUEUE 2 | ||
35 | |||
36 | #define DESC92S_RATE1M 0x00 | ||
37 | #define DESC92S_RATE2M 0x01 | ||
38 | #define DESC92S_RATE5_5M 0x02 | ||
39 | #define DESC92S_RATE11M 0x03 | ||
40 | #define DESC92S_RATE6M 0x04 | ||
41 | #define DESC92S_RATE9M 0x05 | ||
42 | #define DESC92S_RATE12M 0x06 | ||
43 | #define DESC92S_RATE18M 0x07 | ||
44 | #define DESC92S_RATE24M 0x08 | ||
45 | #define DESC92S_RATE36M 0x09 | ||
46 | #define DESC92S_RATE48M 0x0a | ||
47 | #define DESC92S_RATE54M 0x0b | ||
48 | #define DESC92S_RATEMCS0 0x0c | ||
49 | #define DESC92S_RATEMCS1 0x0d | ||
50 | #define DESC92S_RATEMCS2 0x0e | ||
51 | #define DESC92S_RATEMCS3 0x0f | ||
52 | #define DESC92S_RATEMCS4 0x10 | ||
53 | #define DESC92S_RATEMCS5 0x11 | ||
54 | #define DESC92S_RATEMCS6 0x12 | ||
55 | #define DESC92S_RATEMCS7 0x13 | ||
56 | #define DESC92S_RATEMCS8 0x14 | ||
57 | #define DESC92S_RATEMCS9 0x15 | ||
58 | #define DESC92S_RATEMCS10 0x16 | ||
59 | #define DESC92S_RATEMCS11 0x17 | ||
60 | #define DESC92S_RATEMCS12 0x18 | ||
61 | #define DESC92S_RATEMCS13 0x19 | ||
62 | #define DESC92S_RATEMCS14 0x1a | ||
63 | #define DESC92S_RATEMCS15 0x1b | ||
64 | #define DESC92S_RATEMCS15_SG 0x1c | ||
65 | #define DESC92S_RATEMCS32 0x20 | ||
66 | |||
67 | #define SHORT_SLOT_TIME 9 | ||
68 | #define NON_SHORT_SLOT_TIME 20 | ||
69 | |||
70 | /* Rx smooth factor */ | ||
71 | #define RX_SMOOTH_FACTOR 20 | ||
72 | |||
73 | /* Queue Select Value in TxDesc */ | ||
74 | #define QSLT_BK 0x2 | ||
75 | #define QSLT_BE 0x0 | ||
76 | #define QSLT_VI 0x5 | ||
77 | #define QSLT_VO 0x6 | ||
78 | #define QSLT_BEACON 0x10 | ||
79 | #define QSLT_HIGH 0x11 | ||
80 | #define QSLT_MGNT 0x12 | ||
81 | #define QSLT_CMD 0x13 | ||
82 | |||
83 | #define PHY_RSSI_SLID_WIN_MAX 100 | ||
84 | #define PHY_LINKQUALITY_SLID_WIN_MAX 20 | ||
85 | #define PHY_BEACON_RSSI_SLID_WIN_MAX 10 | ||
86 | |||
87 | /* Tx Desc */ | ||
88 | #define TX_DESC_SIZE_RTL8192S (16 * 4) | ||
89 | #define TX_CMDDESC_SIZE_RTL8192S (16 * 4) | ||
90 | |||
91 | /* Define a macro that takes a le32 word, converts it to host ordering, | ||
92 | * right shifts by a specified count, creates a mask of the specified | ||
93 | * bit count, and extracts that number of bits. | ||
94 | */ | ||
95 | |||
96 | #define SHIFT_AND_MASK_LE(__pdesc, __shift, __mask) \ | ||
97 | ((le32_to_cpu(*(((__le32 *)(__pdesc)))) >> (__shift)) & \ | ||
98 | BIT_LEN_MASK_32(__mask)) | ||
99 | |||
100 | /* Define a macro that clears a bit field in an le32 word and | ||
101 | * sets the specified value into that bit field. The resulting | ||
102 | * value remains in le32 ordering; however, it is properly converted | ||
103 | * to host ordering for the clear and set operations before conversion | ||
104 | * back to le32. | ||
105 | */ | ||
106 | |||
107 | #define SET_BITS_OFFSET_LE(__pdesc, __shift, __len, __val) \ | ||
108 | (*(__le32 *)(__pdesc) = \ | ||
109 | (cpu_to_le32((le32_to_cpu(*((__le32 *)(__pdesc))) & \ | ||
110 | (~(BIT_OFFSET_LEN_MASK_32((__shift), __len)))) | \ | ||
111 | (((u32)(__val) & BIT_LEN_MASK_32(__len)) << (__shift))))); | ||
112 | |||
113 | /* macros to read/write various fields in RX or TX descriptors */ | ||
114 | |||
115 | /* Dword 0 */ | ||
116 | #define SET_TX_DESC_PKT_SIZE(__pdesc, __val) \ | ||
117 | SET_BITS_OFFSET_LE(__pdesc, 0, 16, __val) | ||
118 | #define SET_TX_DESC_OFFSET(__pdesc, __val) \ | ||
119 | SET_BITS_OFFSET_LE(__pdesc, 16, 8, __val) | ||
120 | #define SET_TX_DESC_TYPE(__pdesc, __val) \ | ||
121 | SET_BITS_OFFSET_LE(__pdesc, 24, 2, __val) | ||
122 | #define SET_TX_DESC_LAST_SEG(__pdesc, __val) \ | ||
123 | SET_BITS_OFFSET_LE(__pdesc, 26, 1, __val) | ||
124 | #define SET_TX_DESC_FIRST_SEG(__pdesc, __val) \ | ||
125 | SET_BITS_OFFSET_LE(__pdesc, 27, 1, __val) | ||
126 | #define SET_TX_DESC_LINIP(__pdesc, __val) \ | ||
127 | SET_BITS_OFFSET_LE(__pdesc, 28, 1, __val) | ||
128 | #define SET_TX_DESC_AMSDU(__pdesc, __val) \ | ||
129 | SET_BITS_OFFSET_LE(__pdesc, 29, 1, __val) | ||
130 | #define SET_TX_DESC_GREEN_FIELD(__pdesc, __val) \ | ||
131 | SET_BITS_OFFSET_LE(__pdesc, 30, 1, __val) | ||
132 | #define SET_TX_DESC_OWN(__pdesc, __val) \ | ||
133 | SET_BITS_OFFSET_LE(__pdesc, 31, 1, __val) | ||
134 | |||
135 | #define GET_TX_DESC_OWN(__pdesc) \ | ||
136 | SHIFT_AND_MASK_LE(__pdesc, 31, 1) | ||
137 | |||
138 | /* Dword 1 */ | ||
139 | #define SET_TX_DESC_MACID(__pdesc, __val) \ | ||
140 | SET_BITS_OFFSET_LE(__pdesc + 4, 0, 5, __val) | ||
141 | #define SET_TX_DESC_MORE_DATA(__pdesc, __val) \ | ||
142 | SET_BITS_OFFSET_LE(__pdesc + 4, 5, 1, __val) | ||
143 | #define SET_TX_DESC_MORE_FRAG(__pdesc, __val) \ | ||
144 | SET_BITS_OFFSET_LE(__pdesc + 4, 6, 1, __val) | ||
145 | #define SET_TX_DESC_PIFS(__pdesc, __val) \ | ||
146 | SET_BITS_OFFSET_LE(__pdesc + 4, 7, 1, __val) | ||
147 | #define SET_TX_DESC_QUEUE_SEL(__pdesc, __val) \ | ||
148 | SET_BITS_OFFSET_LE(__pdesc + 4, 8, 5, __val) | ||
149 | #define SET_TX_DESC_ACK_POLICY(__pdesc, __val) \ | ||
150 | SET_BITS_OFFSET_LE(__pdesc + 4, 13, 2, __val) | ||
151 | #define SET_TX_DESC_NO_ACM(__pdesc, __val) \ | ||
152 | SET_BITS_OFFSET_LE(__pdesc + 4, 15, 1, __val) | ||
153 | #define SET_TX_DESC_NON_QOS(__pdesc, __val) \ | ||
154 | SET_BITS_OFFSET_LE(__pdesc + 4, 16, 1, __val) | ||
155 | #define SET_TX_DESC_KEY_ID(__pdesc, __val) \ | ||
156 | SET_BITS_OFFSET_LE(__pdesc + 4, 17, 2, __val) | ||
157 | #define SET_TX_DESC_OUI(__pdesc, __val) \ | ||
158 | SET_BITS_OFFSET_LE(__pdesc + 4, 19, 1, __val) | ||
159 | #define SET_TX_DESC_PKT_TYPE(__pdesc, __val) \ | ||
160 | SET_BITS_OFFSET_LE(__pdesc + 4, 20, 1, __val) | ||
161 | #define SET_TX_DESC_EN_DESC_ID(__pdesc, __val) \ | ||
162 | SET_BITS_OFFSET_LE(__pdesc + 4, 21, 1, __val) | ||
163 | #define SET_TX_DESC_SEC_TYPE(__pdesc, __val) \ | ||
164 | SET_BITS_OFFSET_LE(__pdesc + 4, 22, 2, __val) | ||
165 | #define SET_TX_DESC_WDS(__pdesc, __val) \ | ||
166 | SET_BITS_OFFSET_LE(__pdesc + 4, 24, 1, __val) | ||
167 | #define SET_TX_DESC_HTC(__pdesc, __val) \ | ||
168 | SET_BITS_OFFSET_LE(__pdesc + 4, 25, 1, __val) | ||
169 | #define SET_TX_DESC_PKT_OFFSET(__pdesc, __val) \ | ||
170 | SET_BITS_OFFSET_LE(__pdesc + 4, 26, 5, __val) | ||
171 | #define SET_TX_DESC_HWPC(__pdesc, __val) \ | ||
172 | SET_BITS_OFFSET_LE(__pdesc + 4, 27, 1, __val) | ||
173 | |||
174 | /* Dword 2 */ | ||
175 | #define SET_TX_DESC_DATA_RETRY_LIMIT(__pdesc, __val) \ | ||
176 | SET_BITS_OFFSET_LE(__pdesc + 8, 0, 6, __val) | ||
177 | #define SET_TX_DESC_RETRY_LIMIT_ENABLE(__pdesc, __val) \ | ||
178 | SET_BITS_OFFSET_LE(__pdesc + 8, 6, 1, __val) | ||
179 | #define SET_TX_DESC_TSFL(__pdesc, __val) \ | ||
180 | SET_BITS_OFFSET_LE(__pdesc + 8, 7, 5, __val) | ||
181 | #define SET_TX_DESC_RTS_RETRY_COUNT(__pdesc, __val) \ | ||
182 | SET_BITS_OFFSET_LE(__pdesc + 8, 12, 6, __val) | ||
183 | #define SET_TX_DESC_DATA_RETRY_COUNT(__pdesc, __val) \ | ||
184 | SET_BITS_OFFSET_LE(__pdesc + 8, 18, 6, __val) | ||
185 | #define SET_TX_DESC_RSVD_MACID(__pdesc, __val) \ | ||
186 | SET_BITS_OFFSET_LE(((__pdesc) + 8), 24, 5, __val) | ||
187 | #define SET_TX_DESC_AGG_ENABLE(__pdesc, __val) \ | ||
188 | SET_BITS_OFFSET_LE(__pdesc + 8, 29, 1, __val) | ||
189 | #define SET_TX_DESC_AGG_BREAK(__pdesc, __val) \ | ||
190 | SET_BITS_OFFSET_LE(__pdesc + 8, 30, 1, __val) | ||
191 | #define SET_TX_DESC_OWN_MAC(__pdesc, __val) \ | ||
192 | SET_BITS_OFFSET_LE(__pdesc + 8, 31, 1, __val) | ||
193 | |||
194 | /* Dword 3 */ | ||
195 | #define SET_TX_DESC_NEXT_HEAP_PAGE(__pdesc, __val) \ | ||
196 | SET_BITS_OFFSET_LE(__pdesc + 12, 0, 8, __val) | ||
197 | #define SET_TX_DESC_TAIL_PAGE(__pdesc, __val) \ | ||
198 | SET_BITS_OFFSET_LE(__pdesc + 12, 8, 8, __val) | ||
199 | #define SET_TX_DESC_SEQ(__pdesc, __val) \ | ||
200 | SET_BITS_OFFSET_LE(__pdesc + 12, 16, 12, __val) | ||
201 | #define SET_TX_DESC_FRAG(__pdesc, __val) \ | ||
202 | SET_BITS_OFFSET_LE(__pdesc + 12, 28, 4, __val) | ||
203 | |||
204 | /* Dword 4 */ | ||
205 | #define SET_TX_DESC_RTS_RATE(__pdesc, __val) \ | ||
206 | SET_BITS_OFFSET_LE(__pdesc + 16, 0, 6, __val) | ||
207 | #define SET_TX_DESC_DISABLE_RTS_FB(__pdesc, __val) \ | ||
208 | SET_BITS_OFFSET_LE(__pdesc + 16, 6, 1, __val) | ||
209 | #define SET_TX_DESC_RTS_RATE_FB_LIMIT(__pdesc, __val) \ | ||
210 | SET_BITS_OFFSET_LE(__pdesc + 16, 7, 4, __val) | ||
211 | #define SET_TX_DESC_CTS_ENABLE(__pdesc, __val) \ | ||
212 | SET_BITS_OFFSET_LE(__pdesc + 16, 11, 1, __val) | ||
213 | #define SET_TX_DESC_RTS_ENABLE(__pdesc, __val) \ | ||
214 | SET_BITS_OFFSET_LE(__pdesc + 16, 12, 1, __val) | ||
215 | #define SET_TX_DESC_RA_BRSR_ID(__pdesc, __val) \ | ||
216 | SET_BITS_OFFSET_LE(__pdesc + 16, 13, 3, __val) | ||
217 | #define SET_TX_DESC_TXHT(__pdesc, __val) \ | ||
218 | SET_BITS_OFFSET_LE(__pdesc + 16, 16, 1, __val) | ||
219 | #define SET_TX_DESC_TX_SHORT(__pdesc, __val) \ | ||
220 | SET_BITS_OFFSET_LE(__pdesc + 16, 17, 1, __val) | ||
221 | #define SET_TX_DESC_TX_BANDWIDTH(__pdesc, __val) \ | ||
222 | SET_BITS_OFFSET_LE(__pdesc + 16, 18, 1, __val) | ||
223 | #define SET_TX_DESC_TX_SUB_CARRIER(__pdesc, __val) \ | ||
224 | SET_BITS_OFFSET_LE(__pdesc + 16, 19, 2, __val) | ||
225 | #define SET_TX_DESC_TX_STBC(__pdesc, __val) \ | ||
226 | SET_BITS_OFFSET_LE(__pdesc + 16, 21, 2, __val) | ||
227 | #define SET_TX_DESC_TX_REVERSE_DIRECTION(__pdesc, __val) \ | ||
228 | SET_BITS_OFFSET_LE(__pdesc + 16, 23, 1, __val) | ||
229 | #define SET_TX_DESC_RTS_HT(__pdesc, __val) \ | ||
230 | SET_BITS_OFFSET_LE(__pdesc + 16, 24, 1, __val) | ||
231 | #define SET_TX_DESC_RTS_SHORT(__pdesc, __val) \ | ||
232 | SET_BITS_OFFSET_LE(__pdesc + 16, 25, 1, __val) | ||
233 | #define SET_TX_DESC_RTS_BANDWIDTH(__pdesc, __val) \ | ||
234 | SET_BITS_OFFSET_LE(__pdesc + 16, 26, 1, __val) | ||
235 | #define SET_TX_DESC_RTS_SUB_CARRIER(__pdesc, __val) \ | ||
236 | SET_BITS_OFFSET_LE(__pdesc + 16, 27, 2, __val) | ||
237 | #define SET_TX_DESC_RTS_STBC(__pdesc, __val) \ | ||
238 | SET_BITS_OFFSET_LE(__pdesc + 16, 29, 2, __val) | ||
239 | #define SET_TX_DESC_USER_RATE(__pdesc, __val) \ | ||
240 | SET_BITS_OFFSET_LE(__pdesc + 16, 31, 1, __val) | ||
241 | |||
242 | /* Dword 5 */ | ||
243 | #define SET_TX_DESC_PACKET_ID(__pdesc, __val) \ | ||
244 | SET_BITS_OFFSET_LE(__pdesc + 20, 0, 9, __val) | ||
245 | #define SET_TX_DESC_TX_RATE(__pdesc, __val) \ | ||
246 | SET_BITS_OFFSET_LE(__pdesc + 20, 9, 6, __val) | ||
247 | #define SET_TX_DESC_DISABLE_FB(__pdesc, __val) \ | ||
248 | SET_BITS_OFFSET_LE(__pdesc + 20, 15, 1, __val) | ||
249 | #define SET_TX_DESC_DATA_RATE_FB_LIMIT(__pdesc, __val) \ | ||
250 | SET_BITS_OFFSET_LE(__pdesc + 20, 16, 5, __val) | ||
251 | #define SET_TX_DESC_TX_AGC(__pdesc, __val) \ | ||
252 | SET_BITS_OFFSET_LE(__pdesc + 20, 21, 11, __val) | ||
253 | |||
254 | /* Dword 6 */ | ||
255 | #define SET_TX_DESC_IP_CHECK_SUM(__pdesc, __val) \ | ||
256 | SET_BITS_OFFSET_LE(__pdesc + 24, 0, 16, __val) | ||
257 | #define SET_TX_DESC_TCP_CHECK_SUM(__pdesc, __val) \ | ||
258 | SET_BITS_OFFSET_LE(__pdesc + 24, 16, 16, __val) | ||
259 | |||
260 | /* Dword 7 */ | ||
261 | #define SET_TX_DESC_TX_BUFFER_SIZE(__pdesc, __val) \ | ||
262 | SET_BITS_OFFSET_LE(__pdesc + 28, 0, 16, __val) | ||
263 | #define SET_TX_DESC_IP_HEADER_OFFSET(__pdesc, __val) \ | ||
264 | SET_BITS_OFFSET_LE(__pdesc + 28, 16, 8, __val) | ||
265 | #define SET_TX_DESC_TCP_ENABLE(__pdesc, __val) \ | ||
266 | SET_BITS_OFFSET_LE(__pdesc + 28, 31, 1, __val) | ||
267 | |||
268 | /* Dword 8 */ | ||
269 | #define SET_TX_DESC_TX_BUFFER_ADDRESS(__pdesc, __val) \ | ||
270 | SET_BITS_OFFSET_LE(__pdesc + 32, 0, 32, __val) | ||
271 | #define GET_TX_DESC_TX_BUFFER_ADDRESS(__pdesc) \ | ||
272 | SHIFT_AND_MASK_LE(__pdesc + 32, 0, 32) | ||
273 | |||
274 | /* Dword 9 */ | ||
275 | #define SET_TX_DESC_NEXT_DESC_ADDRESS(__pdesc, __val) \ | ||
276 | SET_BITS_OFFSET_LE(__pdesc + 36, 0, 32, __val) | ||
277 | |||
278 | /* Because the PCI Tx descriptors are chaied at the | ||
279 | * initialization and all the NextDescAddresses in | ||
280 | * these descriptors cannot not be cleared (,or | ||
281 | * driver/HW cannot find the next descriptor), the | ||
282 | * offset 36 (NextDescAddresses) is reserved when | ||
283 | * the desc is cleared. */ | ||
284 | #define TX_DESC_NEXT_DESC_OFFSET 36 | ||
285 | #define CLEAR_PCI_TX_DESC_CONTENT(__pdesc, _size) \ | ||
286 | do { \ | ||
287 | if (_size > TX_DESC_NEXT_DESC_OFFSET) \ | ||
288 | memset(__pdesc, 0, TX_DESC_NEXT_DESC_OFFSET); \ | ||
289 | else \ | ||
290 | memset(__pdesc, 0, _size); \ | ||
291 | } while (0); | ||
292 | |||
293 | /* Rx Desc */ | ||
294 | #define RX_STATUS_DESC_SIZE 24 | ||
295 | #define RX_DRV_INFO_SIZE_UNIT 8 | ||
296 | |||
297 | /* DWORD 0 */ | ||
298 | #define SET_RX_STATUS_DESC_PKT_LEN(__pdesc, __val) \ | ||
299 | SET_BITS_OFFSET_LE(__pdesc, 0, 14, __val) | ||
300 | #define SET_RX_STATUS_DESC_CRC32(__pdesc, __val) \ | ||
301 | SET_BITS_OFFSET_LE(__pdesc, 14, 1, __val) | ||
302 | #define SET_RX_STATUS_DESC_ICV(__pdesc, __val) \ | ||
303 | SET_BITS_OFFSET_LE(__pdesc, 15, 1, __val) | ||
304 | #define SET_RX_STATUS_DESC_DRVINFO_SIZE(__pdesc, __val) \ | ||
305 | SET_BITS_OFFSET_LE(__pdesc, 16, 4, __val) | ||
306 | #define SET_RX_STATUS_DESC_SECURITY(__pdesc, __val) \ | ||
307 | SET_BITS_OFFSET_LE(__pdesc, 20, 3, __val) | ||
308 | #define SET_RX_STATUS_DESC_QOS(__pdesc, __val) \ | ||
309 | SET_BITS_OFFSET_LE(__pdesc, 23, 1, __val) | ||
310 | #define SET_RX_STATUS_DESC_SHIFT(__pdesc, __val) \ | ||
311 | SET_BITS_OFFSET_LE(__pdesc, 24, 2, __val) | ||
312 | #define SET_RX_STATUS_DESC_PHY_STATUS(__pdesc, __val) \ | ||
313 | SET_BITS_OFFSET_LE(__pdesc, 26, 1, __val) | ||
314 | #define SET_RX_STATUS_DESC_SWDEC(__pdesc, __val) \ | ||
315 | SET_BITS_OFFSET_LE(__pdesc, 27, 1, __val) | ||
316 | #define SET_RX_STATUS_DESC_LAST_SEG(__pdesc, __val) \ | ||
317 | SET_BITS_OFFSET_LE(__pdesc, 28, 1, __val) | ||
318 | #define SET_RX_STATUS_DESC_FIRST_SEG(__pdesc, __val) \ | ||
319 | SET_BITS_OFFSET_LE(__pdesc, 29, 1, __val) | ||
320 | #define SET_RX_STATUS_DESC_EOR(__pdesc, __val) \ | ||
321 | SET_BITS_OFFSET_LE(__pdesc, 30, 1, __val) | ||
322 | #define SET_RX_STATUS_DESC_OWN(__pdesc, __val) \ | ||
323 | SET_BITS_OFFSET_LE(__pdesc, 31, 1, __val) | ||
324 | |||
325 | #define GET_RX_STATUS_DESC_PKT_LEN(__pdesc) \ | ||
326 | SHIFT_AND_MASK_LE(__pdesc, 0, 14) | ||
327 | #define GET_RX_STATUS_DESC_CRC32(__pdesc) \ | ||
328 | SHIFT_AND_MASK_LE(__pdesc, 14, 1) | ||
329 | #define GET_RX_STATUS_DESC_ICV(__pdesc) \ | ||
330 | SHIFT_AND_MASK_LE(__pdesc, 15, 1) | ||
331 | #define GET_RX_STATUS_DESC_DRVINFO_SIZE(__pdesc) \ | ||
332 | SHIFT_AND_MASK_LE(__pdesc, 16, 4) | ||
333 | #define GET_RX_STATUS_DESC_SECURITY(__pdesc) \ | ||
334 | SHIFT_AND_MASK_LE(__pdesc, 20, 3) | ||
335 | #define GET_RX_STATUS_DESC_QOS(__pdesc) \ | ||
336 | SHIFT_AND_MASK_LE(__pdesc, 23, 1) | ||
337 | #define GET_RX_STATUS_DESC_SHIFT(__pdesc) \ | ||
338 | SHIFT_AND_MASK_LE(__pdesc, 24, 2) | ||
339 | #define GET_RX_STATUS_DESC_PHY_STATUS(__pdesc) \ | ||
340 | SHIFT_AND_MASK_LE(__pdesc, 26, 1) | ||
341 | #define GET_RX_STATUS_DESC_SWDEC(__pdesc) \ | ||
342 | SHIFT_AND_MASK_LE(__pdesc, 27, 1) | ||
343 | #define GET_RX_STATUS_DESC_LAST_SEG(__pdesc) \ | ||
344 | SHIFT_AND_MASK_LE(__pdesc, 28, 1) | ||
345 | #define GET_RX_STATUS_DESC_FIRST_SEG(__pdesc) \ | ||
346 | SHIFT_AND_MASK_LE(__pdesc, 29, 1) | ||
347 | #define GET_RX_STATUS_DESC_EOR(__pdesc) \ | ||
348 | SHIFT_AND_MASK_LE(__pdesc, 30, 1) | ||
349 | #define GET_RX_STATUS_DESC_OWN(__pdesc) \ | ||
350 | SHIFT_AND_MASK_LE(__pdesc, 31, 1) | ||
351 | |||
352 | /* DWORD 1 */ | ||
353 | #define SET_RX_STATUS_DESC_MACID(__pdesc, __val) \ | ||
354 | SET_BITS_OFFSET_LE(__pdesc + 4, 0, 5, __val) | ||
355 | #define SET_RX_STATUS_DESC_TID(__pdesc, __val) \ | ||
356 | SET_BITS_OFFSET_LE(__pdesc + 4, 5, 4, __val) | ||
357 | #define SET_RX_STATUS_DESC_PAGGR(__pdesc, __val) \ | ||
358 | SET_BITS_OFFSET_LE(__pdesc + 4, 14, 1, __val) | ||
359 | #define SET_RX_STATUS_DESC_FAGGR(__pdesc, __val) \ | ||
360 | SET_BITS_OFFSET_LE(__pdesc + 4, 15, 1, __val) | ||
361 | #define SET_RX_STATUS_DESC_A1_FIT(__pdesc, __val) \ | ||
362 | SET_BITS_OFFSET_LE(__pdesc + 4, 16, 4, __val) | ||
363 | #define SET_RX_STATUS_DESC_A2_FIT(__pdesc, __val) \ | ||
364 | SET_BITS_OFFSET_LE(__pdesc + 4, 20, 4, __val) | ||
365 | #define SET_RX_STATUS_DESC_PAM(__pdesc, __val) \ | ||
366 | SET_BITS_OFFSET_LE(__pdesc + 4, 24, 1, __val) | ||
367 | #define SET_RX_STATUS_DESC_PWR(__pdesc, __val) \ | ||
368 | SET_BITS_OFFSET_LE(__pdesc + 4, 25, 1, __val) | ||
369 | #define SET_RX_STATUS_DESC_MOREDATA(__pdesc, __val) \ | ||
370 | SET_BITS_OFFSET_LE(__pdesc + 4, 26, 1, __val) | ||
371 | #define SET_RX_STATUS_DESC_MOREFRAG(__pdesc, __val) \ | ||
372 | SET_BITS_OFFSET_LE(__pdesc + 4, 27, 1, __val) | ||
373 | #define SET_RX_STATUS_DESC_TYPE(__pdesc, __val) \ | ||
374 | SET_BITS_OFFSET_LE(__pdesc + 4, 28, 2, __val) | ||
375 | #define SET_RX_STATUS_DESC_MC(__pdesc, __val) \ | ||
376 | SET_BITS_OFFSET_LE(__pdesc + 4, 30, 1, __val) | ||
377 | #define SET_RX_STATUS_DESC_BC(__pdesc, __val) \ | ||
378 | SET_BITS_OFFSET_LE(__pdesc + 4, 31, 1, __val) | ||
379 | |||
380 | #define GET_RX_STATUS_DEC_MACID(__pdesc) \ | ||
381 | SHIFT_AND_MASK_LE(__pdesc + 4, 0, 5) | ||
382 | #define GET_RX_STATUS_DESC_TID(__pdesc) \ | ||
383 | SHIFT_AND_MASK_LE(__pdesc + 4, 5, 4) | ||
384 | #define GET_RX_STATUS_DESC_PAGGR(__pdesc) \ | ||
385 | SHIFT_AND_MASK_LE(__pdesc + 4, 14, 1) | ||
386 | #define GET_RX_STATUS_DESC_FAGGR(__pdesc) \ | ||
387 | SHIFT_AND_MASK_LE(__pdesc + 4, 15, 1) | ||
388 | #define GET_RX_STATUS_DESC_A1_FIT(__pdesc) \ | ||
389 | SHIFT_AND_MASK_LE(__pdesc + 4, 16, 4) | ||
390 | #define GET_RX_STATUS_DESC_A2_FIT(__pdesc) \ | ||
391 | SHIFT_AND_MASK_LE(__pdesc + 4, 20, 4) | ||
392 | #define GET_RX_STATUS_DESC_PAM(__pdesc) \ | ||
393 | SHIFT_AND_MASK_LE(__pdesc + 4, 24, 1) | ||
394 | #define GET_RX_STATUS_DESC_PWR(__pdesc) \ | ||
395 | SHIFT_AND_MASK_LE(__pdesc + 4, 25, 1) | ||
396 | #define GET_RX_STATUS_DESC_MORE_DATA(__pdesc) \ | ||
397 | SHIFT_AND_MASK_LE(__pdesc + 4, 26, 1) | ||
398 | #define GET_RX_STATUS_DESC_MORE_FRAG(__pdesc) \ | ||
399 | SHIFT_AND_MASK_LE(__pdesc + 4, 27, 1) | ||
400 | #define GET_RX_STATUS_DESC_TYPE(__pdesc) \ | ||
401 | SHIFT_AND_MASK_LE(__pdesc + 4, 28, 2) | ||
402 | #define GET_RX_STATUS_DESC_MC(__pdesc) \ | ||
403 | SHIFT_AND_MASK_LE(__pdesc + 4, 30, 1) | ||
404 | #define GET_RX_STATUS_DESC_BC(__pdesc) \ | ||
405 | SHIFT_AND_MASK_LE(__pdesc + 4, 31, 1) | ||
406 | |||
407 | /* DWORD 2 */ | ||
408 | #define SET_RX_STATUS_DESC_SEQ(__pdesc, __val) \ | ||
409 | SET_BITS_OFFSET_LE(__pdesc + 8, 0, 12, __val) | ||
410 | #define SET_RX_STATUS_DESC_FRAG(__pdesc, __val) \ | ||
411 | SET_BITS_OFFSET_LE(__pdesc + 8, 12, 4, __val) | ||
412 | #define SET_RX_STATUS_DESC_NEXT_PKTLEN(__pdesc, __val) \ | ||
413 | SET_BITS_OFFSET_LE(__pdesc + 8, 16, 8, __val) | ||
414 | #define SET_RX_STATUS_DESC_NEXT_IND(__pdesc, __val) \ | ||
415 | SET_BITS_OFFSET_LE(__pdesc + 8, 30, 1, __val) | ||
416 | |||
417 | #define GET_RX_STATUS_DESC_SEQ(__pdesc) \ | ||
418 | SHIFT_AND_MASK_LE(__pdesc + 8, 0, 12) | ||
419 | #define GET_RX_STATUS_DESC_FRAG(__pdesc) \ | ||
420 | SHIFT_AND_MASK_LE(__pdesc + 8, 12, 4) | ||
421 | #define GET_RX_STATUS_DESC_NEXT_PKTLEN(__pdesc) \ | ||
422 | SHIFT_AND_MASK_LE(__pdesc + 8, 16, 8) | ||
423 | #define GET_RX_STATUS_DESC_NEXT_IND(__pdesc) \ | ||
424 | SHIFT_AND_MASK_LE(__pdesc + 8, 30, 1) | ||
425 | |||
426 | /* DWORD 3 */ | ||
427 | #define SET_RX_STATUS_DESC_RX_MCS(__pdesc, __val) \ | ||
428 | SET_BITS_OFFSET_LE(__pdesc + 12, 0, 6, __val) | ||
429 | #define SET_RX_STATUS_DESC_RX_HT(__pdesc, __val) \ | ||
430 | SET_BITS_OFFSET_LE(__pdesc + 12, 6, 1, __val) | ||
431 | #define SET_RX_STATUS_DESC_AMSDU(__pdesc, __val) \ | ||
432 | SET_BITS_OFFSET_LE(__pdesc + 12, 7, 1, __val) | ||
433 | #define SET_RX_STATUS_DESC_SPLCP(__pdesc, __val) \ | ||
434 | SET_BITS_OFFSET_LE(__pdesc + 12, 8, 1, __val) | ||
435 | #define SET_RX_STATUS_DESC_BW(__pdesc, __val) \ | ||
436 | SET_BITS_OFFSET_LE(__pdesc + 12, 9, 1, __val) | ||
437 | #define SET_RX_STATUS_DESC_HTC(__pdesc, __val) \ | ||
438 | SET_BITS_OFFSET_LE(__pdesc + 12, 10, 1, __val) | ||
439 | #define SET_RX_STATUS_DESC_TCP_CHK_RPT(__pdesc, __val) \ | ||
440 | SET_BITS_OFFSET_LE(__pdesc + 12, 11, 1, __val) | ||
441 | #define SET_RX_STATUS_DESC_IP_CHK_RPT(__pdesc, __val) \ | ||
442 | SET_BITS_OFFSET_LE(__pdesc + 12, 12, 1, __val) | ||
443 | #define SET_RX_STATUS_DESC_TCP_CHK_VALID(__pdesc, __val) \ | ||
444 | SET_BITS_OFFSET_LE(__pdesc + 12, 13, 1, __val) | ||
445 | #define SET_RX_STATUS_DESC_HWPC_ERR(__pdesc, __val) \ | ||
446 | SET_BITS_OFFSET_LE(__pdesc + 12, 14, 1, __val) | ||
447 | #define SET_RX_STATUS_DESC_HWPC_IND(__pdesc, __val) \ | ||
448 | SET_BITS_OFFSET_LE(__pdesc + 12, 15, 1, __val) | ||
449 | #define SET_RX_STATUS_DESC_IV0(__pdesc, __val) \ | ||
450 | SET_BITS_OFFSET_LE(__pdesc + 12, 16, 16, __val) | ||
451 | |||
452 | #define GET_RX_STATUS_DESC_RX_MCS(__pdesc) \ | ||
453 | SHIFT_AND_MASK_LE(__pdesc + 12, 0, 6) | ||
454 | #define GET_RX_STATUS_DESC_RX_HT(__pdesc) \ | ||
455 | SHIFT_AND_MASK_LE(__pdesc + 12, 6, 1) | ||
456 | #define GET_RX_STATUS_DESC_AMSDU(__pdesc) \ | ||
457 | SHIFT_AND_MASK_LE(__pdesc + 12, 7, 1) | ||
458 | #define GET_RX_STATUS_DESC_SPLCP(__pdesc) \ | ||
459 | SHIFT_AND_MASK_LE(__pdesc + 12, 8, 1) | ||
460 | #define GET_RX_STATUS_DESC_BW(__pdesc) \ | ||
461 | SHIFT_AND_MASK_LE(__pdesc + 12, 9, 1) | ||
462 | #define GET_RX_STATUS_DESC_HTC(__pdesc) \ | ||
463 | SHIFT_AND_MASK_LE(__pdesc + 12, 10, 1) | ||
464 | #define GET_RX_STATUS_DESC_TCP_CHK_RPT(__pdesc) \ | ||
465 | SHIFT_AND_MASK_LE(__pdesc + 12, 11, 1) | ||
466 | #define GET_RX_STATUS_DESC_IP_CHK_RPT(__pdesc) \ | ||
467 | SHIFT_AND_MASK_LE(__pdesc + 12, 12, 1) | ||
468 | #define GET_RX_STATUS_DESC_TCP_CHK_VALID(__pdesc) \ | ||
469 | SHIFT_AND_MASK_LE(__pdesc + 12, 13, 1) | ||
470 | #define GET_RX_STATUS_DESC_HWPC_ERR(__pdesc) \ | ||
471 | SHIFT_AND_MASK_LE(__pdesc + 12, 14, 1) | ||
472 | #define GET_RX_STATUS_DESC_HWPC_IND(__pdesc) \ | ||
473 | SHIFT_AND_MASK_LE(__pdesc + 12, 15, 1) | ||
474 | #define GET_RX_STATUS_DESC_IV0(__pdesc) \ | ||
475 | SHIFT_AND_MASK_LE(__pdesc + 12, 16, 16) | ||
476 | |||
477 | /* DWORD 4 */ | ||
478 | #define SET_RX_STATUS_DESC_IV1(__pdesc, __val) \ | ||
479 | SET_BITS_OFFSET_LE(__pdesc + 16, 0, 32, __val) | ||
480 | #define GET_RX_STATUS_DESC_IV1(__pdesc) \ | ||
481 | SHIFT_AND_MASK_LE(__pdesc + 16, 0, 32) | ||
482 | |||
483 | /* DWORD 5 */ | ||
484 | #define SET_RX_STATUS_DESC_TSFL(__pdesc, __val) \ | ||
485 | SET_BITS_OFFSET_LE(__pdesc + 20, 0, 32, __val) | ||
486 | #define GET_RX_STATUS_DESC_TSFL(__pdesc) \ | ||
487 | SHIFT_AND_MASK_LE(__pdesc + 20, 0, 32) | ||
488 | |||
489 | /* DWORD 6 */ | ||
490 | #define SET_RX_STATUS__DESC_BUFF_ADDR(__pdesc, __val) \ | ||
491 | SET_BITS_OFFSET_LE(__pdesc + 24, 0, 32, __val) | ||
492 | |||
493 | #define RX_HAL_IS_CCK_RATE(_pdesc)\ | ||
494 | (GET_RX_STATUS_DESC_RX_MCS(_pdesc) == DESC92S_RATE1M || \ | ||
495 | GET_RX_STATUS_DESC_RX_MCS(_pdesc) == DESC92S_RATE2M || \ | ||
496 | GET_RX_STATUS_DESC_RX_MCS(_pdesc) == DESC92S_RATE5_5M ||\ | ||
497 | GET_RX_STATUS_DESC_RX_MCS(_pdesc) == DESC92S_RATE11M) | ||
498 | |||
499 | enum rf_optype { | ||
500 | RF_OP_BY_SW_3WIRE = 0, | ||
501 | RF_OP_BY_FW, | ||
502 | RF_OP_MAX | ||
503 | }; | ||
504 | |||
505 | enum ic_inferiority { | ||
506 | IC_INFERIORITY_A = 0, | ||
507 | IC_INFERIORITY_B = 1, | ||
508 | }; | ||
509 | |||
510 | enum fwcmd_iotype { | ||
511 | /* For DIG DM */ | ||
512 | FW_CMD_DIG_ENABLE = 0, | ||
513 | FW_CMD_DIG_DISABLE = 1, | ||
514 | FW_CMD_DIG_HALT = 2, | ||
515 | FW_CMD_DIG_RESUME = 3, | ||
516 | /* For High Power DM */ | ||
517 | FW_CMD_HIGH_PWR_ENABLE = 4, | ||
518 | FW_CMD_HIGH_PWR_DISABLE = 5, | ||
519 | /* For Rate adaptive DM */ | ||
520 | FW_CMD_RA_RESET = 6, | ||
521 | FW_CMD_RA_ACTIVE = 7, | ||
522 | FW_CMD_RA_REFRESH_N = 8, | ||
523 | FW_CMD_RA_REFRESH_BG = 9, | ||
524 | FW_CMD_RA_INIT = 10, | ||
525 | /* For FW supported IQK */ | ||
526 | FW_CMD_IQK_INIT = 11, | ||
527 | /* Tx power tracking switch, | ||
528 | * MP driver only */ | ||
529 | FW_CMD_TXPWR_TRACK_ENABLE = 12, | ||
530 | /* Tx power tracking switch, | ||
531 | * MP driver only */ | ||
532 | FW_CMD_TXPWR_TRACK_DISABLE = 13, | ||
533 | /* Tx power tracking with thermal | ||
534 | * indication, for Normal driver */ | ||
535 | FW_CMD_TXPWR_TRACK_THERMAL = 14, | ||
536 | FW_CMD_PAUSE_DM_BY_SCAN = 15, | ||
537 | FW_CMD_RESUME_DM_BY_SCAN = 16, | ||
538 | FW_CMD_RA_REFRESH_N_COMB = 17, | ||
539 | FW_CMD_RA_REFRESH_BG_COMB = 18, | ||
540 | FW_CMD_ANTENNA_SW_ENABLE = 19, | ||
541 | FW_CMD_ANTENNA_SW_DISABLE = 20, | ||
542 | /* Tx Status report for CCX from FW */ | ||
543 | FW_CMD_TX_FEEDBACK_CCX_ENABLE = 21, | ||
544 | /* Indifate firmware that driver | ||
545 | * enters LPS, For PS-Poll issue */ | ||
546 | FW_CMD_LPS_ENTER = 22, | ||
547 | /* Indicate firmware that driver | ||
548 | * leave LPS*/ | ||
549 | FW_CMD_LPS_LEAVE = 23, | ||
550 | /* Set DIG mode to signal strength */ | ||
551 | FW_CMD_DIG_MODE_SS = 24, | ||
552 | /* Set DIG mode to false alarm. */ | ||
553 | FW_CMD_DIG_MODE_FA = 25, | ||
554 | FW_CMD_ADD_A2_ENTRY = 26, | ||
555 | FW_CMD_CTRL_DM_BY_DRIVER = 27, | ||
556 | FW_CMD_CTRL_DM_BY_DRIVER_NEW = 28, | ||
557 | FW_CMD_PAPE_CONTROL = 29, | ||
558 | FW_CMD_IQK_ENABLE = 30, | ||
559 | }; | ||
560 | |||
561 | /* | ||
562 | * Driver info contain PHY status | ||
563 | * and other variabel size info | ||
564 | * PHY Status content as below | ||
565 | */ | ||
566 | struct rx_fwinfo { | ||
567 | /* DWORD 0 */ | ||
568 | u8 gain_trsw[4]; | ||
569 | /* DWORD 1 */ | ||
570 | u8 pwdb_all; | ||
571 | u8 cfosho[4]; | ||
572 | /* DWORD 2 */ | ||
573 | u8 cfotail[4]; | ||
574 | /* DWORD 3 */ | ||
575 | s8 rxevm[2]; | ||
576 | s8 rxsnr[4]; | ||
577 | /* DWORD 4 */ | ||
578 | u8 pdsnr[2]; | ||
579 | /* DWORD 5 */ | ||
580 | u8 csi_current[2]; | ||
581 | u8 csi_target[2]; | ||
582 | /* DWORD 6 */ | ||
583 | u8 sigevm; | ||
584 | u8 max_ex_pwr; | ||
585 | u8 ex_intf_flag:1; | ||
586 | u8 sgi_en:1; | ||
587 | u8 rxsc:2; | ||
588 | u8 reserve:4; | ||
589 | }; | ||
590 | |||
591 | struct phy_sts_cck_8192s_t { | ||
592 | u8 adc_pwdb_x[4]; | ||
593 | u8 sq_rpt; | ||
594 | u8 cck_agc_rpt; | ||
595 | }; | ||
596 | |||
597 | #endif | ||
598 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/dm.c b/drivers/net/wireless/rtlwifi/rtl8192se/dm.c new file mode 100644 index 000000000000..da86db86fa4a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/dm.c | |||
@@ -0,0 +1,733 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../base.h" | ||
32 | #include "reg.h" | ||
33 | #include "def.h" | ||
34 | #include "phy.h" | ||
35 | #include "dm.h" | ||
36 | #include "fw.h" | ||
37 | |||
38 | struct dig_t digtable; | ||
39 | static const u32 edca_setting_dl[PEER_MAX] = { | ||
40 | 0xa44f, /* 0 UNKNOWN */ | ||
41 | 0x5ea44f, /* 1 REALTEK_90 */ | ||
42 | 0x5ea44f, /* 2 REALTEK_92SE */ | ||
43 | 0xa630, /* 3 BROAD */ | ||
44 | 0xa44f, /* 4 RAL */ | ||
45 | 0xa630, /* 5 ATH */ | ||
46 | 0xa630, /* 6 CISCO */ | ||
47 | 0xa42b, /* 7 MARV */ | ||
48 | }; | ||
49 | |||
50 | static const u32 edca_setting_dl_gmode[PEER_MAX] = { | ||
51 | 0x4322, /* 0 UNKNOWN */ | ||
52 | 0xa44f, /* 1 REALTEK_90 */ | ||
53 | 0x5ea44f, /* 2 REALTEK_92SE */ | ||
54 | 0xa42b, /* 3 BROAD */ | ||
55 | 0x5e4322, /* 4 RAL */ | ||
56 | 0x4322, /* 5 ATH */ | ||
57 | 0xa430, /* 6 CISCO */ | ||
58 | 0x5ea44f, /* 7 MARV */ | ||
59 | }; | ||
60 | |||
61 | static const u32 edca_setting_ul[PEER_MAX] = { | ||
62 | 0x5e4322, /* 0 UNKNOWN */ | ||
63 | 0xa44f, /* 1 REALTEK_90 */ | ||
64 | 0x5ea44f, /* 2 REALTEK_92SE */ | ||
65 | 0x5ea322, /* 3 BROAD */ | ||
66 | 0x5ea422, /* 4 RAL */ | ||
67 | 0x5ea322, /* 5 ATH */ | ||
68 | 0x3ea44f, /* 6 CISCO */ | ||
69 | 0x5ea44f, /* 7 MARV */ | ||
70 | }; | ||
71 | |||
72 | static void _rtl92s_dm_check_edca_turbo(struct ieee80211_hw *hw) | ||
73 | { | ||
74 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
75 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
76 | |||
77 | static u64 last_txok_cnt; | ||
78 | static u64 last_rxok_cnt; | ||
79 | u64 cur_txok_cnt = 0; | ||
80 | u64 cur_rxok_cnt = 0; | ||
81 | |||
82 | u32 edca_be_ul = edca_setting_ul[mac->vendor]; | ||
83 | u32 edca_be_dl = edca_setting_dl[mac->vendor]; | ||
84 | u32 edca_gmode = edca_setting_dl_gmode[mac->vendor]; | ||
85 | |||
86 | if (mac->link_state != MAC80211_LINKED) { | ||
87 | rtlpriv->dm.current_turbo_edca = false; | ||
88 | goto dm_checkedcaturbo_exit; | ||
89 | } | ||
90 | |||
91 | if ((!rtlpriv->dm.is_any_nonbepkts) && | ||
92 | (!rtlpriv->dm.disable_framebursting)) { | ||
93 | cur_txok_cnt = rtlpriv->stats.txbytesunicast - last_txok_cnt; | ||
94 | cur_rxok_cnt = rtlpriv->stats.rxbytesunicast - last_rxok_cnt; | ||
95 | |||
96 | if (rtlpriv->phy.rf_type == RF_1T2R) { | ||
97 | if (cur_txok_cnt > 4 * cur_rxok_cnt) { | ||
98 | /* Uplink TP is present. */ | ||
99 | if (rtlpriv->dm.is_cur_rdlstate || | ||
100 | !rtlpriv->dm.current_turbo_edca) { | ||
101 | rtl_write_dword(rtlpriv, EDCAPARA_BE, | ||
102 | edca_be_ul); | ||
103 | rtlpriv->dm.is_cur_rdlstate = false; | ||
104 | } | ||
105 | } else {/* Balance TP is present. */ | ||
106 | if (!rtlpriv->dm.is_cur_rdlstate || | ||
107 | !rtlpriv->dm.current_turbo_edca) { | ||
108 | if (mac->mode == WIRELESS_MODE_G || | ||
109 | mac->mode == WIRELESS_MODE_B) | ||
110 | rtl_write_dword(rtlpriv, | ||
111 | EDCAPARA_BE, | ||
112 | edca_gmode); | ||
113 | else | ||
114 | rtl_write_dword(rtlpriv, | ||
115 | EDCAPARA_BE, | ||
116 | edca_be_dl); | ||
117 | rtlpriv->dm.is_cur_rdlstate = true; | ||
118 | } | ||
119 | } | ||
120 | rtlpriv->dm.current_turbo_edca = true; | ||
121 | } else { | ||
122 | if (cur_rxok_cnt > 4 * cur_txok_cnt) { | ||
123 | if (!rtlpriv->dm.is_cur_rdlstate || | ||
124 | !rtlpriv->dm.current_turbo_edca) { | ||
125 | if (mac->mode == WIRELESS_MODE_G || | ||
126 | mac->mode == WIRELESS_MODE_B) | ||
127 | rtl_write_dword(rtlpriv, | ||
128 | EDCAPARA_BE, | ||
129 | edca_gmode); | ||
130 | else | ||
131 | rtl_write_dword(rtlpriv, | ||
132 | EDCAPARA_BE, | ||
133 | edca_be_dl); | ||
134 | rtlpriv->dm.is_cur_rdlstate = true; | ||
135 | } | ||
136 | } else { | ||
137 | if (rtlpriv->dm.is_cur_rdlstate || | ||
138 | !rtlpriv->dm.current_turbo_edca) { | ||
139 | rtl_write_dword(rtlpriv, EDCAPARA_BE, | ||
140 | edca_be_ul); | ||
141 | rtlpriv->dm.is_cur_rdlstate = false; | ||
142 | } | ||
143 | } | ||
144 | rtlpriv->dm.current_turbo_edca = true; | ||
145 | } | ||
146 | } else { | ||
147 | if (rtlpriv->dm.current_turbo_edca) { | ||
148 | u8 tmp = AC0_BE; | ||
149 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_AC_PARAM, | ||
150 | (u8 *)(&tmp)); | ||
151 | rtlpriv->dm.current_turbo_edca = false; | ||
152 | } | ||
153 | } | ||
154 | |||
155 | dm_checkedcaturbo_exit: | ||
156 | rtlpriv->dm.is_any_nonbepkts = false; | ||
157 | last_txok_cnt = rtlpriv->stats.txbytesunicast; | ||
158 | last_rxok_cnt = rtlpriv->stats.rxbytesunicast; | ||
159 | } | ||
160 | |||
161 | static void _rtl92s_dm_txpowertracking_callback_thermalmeter( | ||
162 | struct ieee80211_hw *hw) | ||
163 | { | ||
164 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
165 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
166 | u8 thermalvalue = 0; | ||
167 | |||
168 | rtlpriv->dm.txpower_trackinginit = true; | ||
169 | |||
170 | thermalvalue = (u8)rtl_get_rfreg(hw, RF90_PATH_A, RF_T_METER, 0x1f); | ||
171 | |||
172 | RT_TRACE(rtlpriv, COMP_POWER_TRACKING, DBG_LOUD, | ||
173 | ("Readback Thermal Meter = 0x%x pre thermal meter 0x%x " | ||
174 | "eeprom_thermalmeter 0x%x\n", thermalvalue, | ||
175 | rtlpriv->dm.thermalvalue, rtlefuse->eeprom_thermalmeter)); | ||
176 | |||
177 | if (thermalvalue) { | ||
178 | rtlpriv->dm.thermalvalue = thermalvalue; | ||
179 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_TXPWR_TRACK_THERMAL); | ||
180 | } | ||
181 | |||
182 | rtlpriv->dm.txpowercount = 0; | ||
183 | } | ||
184 | |||
185 | static void _rtl92s_dm_check_txpowertracking_thermalmeter( | ||
186 | struct ieee80211_hw *hw) | ||
187 | { | ||
188 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
189 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
190 | static u8 tm_trigger; | ||
191 | u8 tx_power_checkcnt = 5; | ||
192 | |||
193 | /* 2T2R TP issue */ | ||
194 | if (rtlphy->rf_type == RF_2T2R) | ||
195 | return; | ||
196 | |||
197 | if (!rtlpriv->dm.txpower_tracking) | ||
198 | return; | ||
199 | |||
200 | if (rtlpriv->dm.txpowercount <= tx_power_checkcnt) { | ||
201 | rtlpriv->dm.txpowercount++; | ||
202 | return; | ||
203 | } | ||
204 | |||
205 | if (!tm_trigger) { | ||
206 | rtl_set_rfreg(hw, RF90_PATH_A, RF_T_METER, | ||
207 | RFREG_OFFSET_MASK, 0x60); | ||
208 | tm_trigger = 1; | ||
209 | } else { | ||
210 | _rtl92s_dm_txpowertracking_callback_thermalmeter(hw); | ||
211 | tm_trigger = 0; | ||
212 | } | ||
213 | } | ||
214 | |||
215 | static void _rtl92s_dm_refresh_rateadaptive_mask(struct ieee80211_hw *hw) | ||
216 | { | ||
217 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
218 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
219 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
220 | struct rate_adaptive *ra = &(rtlpriv->ra); | ||
221 | |||
222 | u32 low_rssi_thresh = 0; | ||
223 | u32 middle_rssi_thresh = 0; | ||
224 | u32 high_rssi_thresh = 0; | ||
225 | u8 rssi_level; | ||
226 | struct ieee80211_sta *sta = NULL; | ||
227 | |||
228 | if (is_hal_stop(rtlhal)) | ||
229 | return; | ||
230 | |||
231 | if (!rtlpriv->dm.useramask) | ||
232 | return; | ||
233 | |||
234 | if (!rtlpriv->dm.inform_fw_driverctrldm) { | ||
235 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_CTRL_DM_BY_DRIVER); | ||
236 | rtlpriv->dm.inform_fw_driverctrldm = true; | ||
237 | } | ||
238 | |||
239 | rcu_read_lock(); | ||
240 | if (mac->opmode == NL80211_IFTYPE_STATION) | ||
241 | sta = get_sta(hw, mac->vif, mac->bssid); | ||
242 | if ((mac->link_state == MAC80211_LINKED) && | ||
243 | (mac->opmode == NL80211_IFTYPE_STATION)) { | ||
244 | switch (ra->pre_ratr_state) { | ||
245 | case DM_RATR_STA_HIGH: | ||
246 | high_rssi_thresh = 40; | ||
247 | middle_rssi_thresh = 30; | ||
248 | low_rssi_thresh = 20; | ||
249 | break; | ||
250 | case DM_RATR_STA_MIDDLE: | ||
251 | high_rssi_thresh = 44; | ||
252 | middle_rssi_thresh = 30; | ||
253 | low_rssi_thresh = 20; | ||
254 | break; | ||
255 | case DM_RATR_STA_LOW: | ||
256 | high_rssi_thresh = 44; | ||
257 | middle_rssi_thresh = 34; | ||
258 | low_rssi_thresh = 20; | ||
259 | break; | ||
260 | case DM_RATR_STA_ULTRALOW: | ||
261 | high_rssi_thresh = 44; | ||
262 | middle_rssi_thresh = 34; | ||
263 | low_rssi_thresh = 24; | ||
264 | break; | ||
265 | default: | ||
266 | high_rssi_thresh = 44; | ||
267 | middle_rssi_thresh = 34; | ||
268 | low_rssi_thresh = 24; | ||
269 | break; | ||
270 | } | ||
271 | |||
272 | if (rtlpriv->dm.undecorated_smoothed_pwdb > | ||
273 | (long)high_rssi_thresh) { | ||
274 | ra->ratr_state = DM_RATR_STA_HIGH; | ||
275 | rssi_level = 1; | ||
276 | } else if (rtlpriv->dm.undecorated_smoothed_pwdb > | ||
277 | (long)middle_rssi_thresh) { | ||
278 | ra->ratr_state = DM_RATR_STA_LOW; | ||
279 | rssi_level = 3; | ||
280 | } else if (rtlpriv->dm.undecorated_smoothed_pwdb > | ||
281 | (long)low_rssi_thresh) { | ||
282 | ra->ratr_state = DM_RATR_STA_LOW; | ||
283 | rssi_level = 5; | ||
284 | } else { | ||
285 | ra->ratr_state = DM_RATR_STA_ULTRALOW; | ||
286 | rssi_level = 6; | ||
287 | } | ||
288 | |||
289 | if (ra->pre_ratr_state != ra->ratr_state) { | ||
290 | RT_TRACE(rtlpriv, COMP_RATE, DBG_LOUD, ("RSSI = %ld " | ||
291 | "RSSI_LEVEL = %d PreState = %d, CurState = %d\n", | ||
292 | rtlpriv->dm.undecorated_smoothed_pwdb, | ||
293 | ra->ratr_state, | ||
294 | ra->pre_ratr_state, ra->ratr_state)); | ||
295 | |||
296 | rtlpriv->cfg->ops->update_rate_tbl(hw, sta, | ||
297 | ra->ratr_state); | ||
298 | ra->pre_ratr_state = ra->ratr_state; | ||
299 | } | ||
300 | } | ||
301 | rcu_read_unlock(); | ||
302 | } | ||
303 | |||
304 | static void _rtl92s_dm_switch_baseband_mrc(struct ieee80211_hw *hw) | ||
305 | { | ||
306 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
307 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
308 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
309 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
310 | bool current_mrc; | ||
311 | bool enable_mrc = true; | ||
312 | long tmpentry_maxpwdb = 0; | ||
313 | u8 rssi_a = 0; | ||
314 | u8 rssi_b = 0; | ||
315 | |||
316 | if (is_hal_stop(rtlhal)) | ||
317 | return; | ||
318 | |||
319 | if ((rtlphy->rf_type == RF_1T1R) || (rtlphy->rf_type == RF_2T2R)) | ||
320 | return; | ||
321 | |||
322 | rtlpriv->cfg->ops->get_hw_reg(hw, HW_VAR_MRC, (u8 *)(¤t_mrc)); | ||
323 | |||
324 | if (mac->link_state >= MAC80211_LINKED) { | ||
325 | if (rtlpriv->dm.undecorated_smoothed_pwdb > tmpentry_maxpwdb) { | ||
326 | rssi_a = rtlpriv->stats.rx_rssi_percentage[RF90_PATH_A]; | ||
327 | rssi_b = rtlpriv->stats.rx_rssi_percentage[RF90_PATH_B]; | ||
328 | } | ||
329 | } | ||
330 | |||
331 | /* MRC settings would NOT affect TP on Wireless B mode. */ | ||
332 | if (mac->mode != WIRELESS_MODE_B) { | ||
333 | if ((rssi_a == 0) && (rssi_b == 0)) { | ||
334 | enable_mrc = true; | ||
335 | } else if (rssi_b > 30) { | ||
336 | /* Turn on B-Path */ | ||
337 | enable_mrc = true; | ||
338 | } else if (rssi_b < 5) { | ||
339 | /* Turn off B-path */ | ||
340 | enable_mrc = false; | ||
341 | /* Take care of RSSI differentiation. */ | ||
342 | } else if (rssi_a > 15 && (rssi_a >= rssi_b)) { | ||
343 | if ((rssi_a - rssi_b) > 15) | ||
344 | /* Turn off B-path */ | ||
345 | enable_mrc = false; | ||
346 | else if ((rssi_a - rssi_b) < 10) | ||
347 | /* Turn on B-Path */ | ||
348 | enable_mrc = true; | ||
349 | else | ||
350 | enable_mrc = current_mrc; | ||
351 | } else { | ||
352 | /* Turn on B-Path */ | ||
353 | enable_mrc = true; | ||
354 | } | ||
355 | } | ||
356 | |||
357 | /* Update MRC settings if needed. */ | ||
358 | if (enable_mrc != current_mrc) | ||
359 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_MRC, | ||
360 | (u8 *)&enable_mrc); | ||
361 | |||
362 | } | ||
363 | |||
364 | void rtl92s_dm_init_edca_turbo(struct ieee80211_hw *hw) | ||
365 | { | ||
366 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
367 | |||
368 | rtlpriv->dm.current_turbo_edca = false; | ||
369 | rtlpriv->dm.is_any_nonbepkts = false; | ||
370 | rtlpriv->dm.is_cur_rdlstate = false; | ||
371 | } | ||
372 | |||
373 | static void _rtl92s_dm_init_rate_adaptive_mask(struct ieee80211_hw *hw) | ||
374 | { | ||
375 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
376 | struct rate_adaptive *ra = &(rtlpriv->ra); | ||
377 | |||
378 | ra->ratr_state = DM_RATR_STA_MAX; | ||
379 | ra->pre_ratr_state = DM_RATR_STA_MAX; | ||
380 | |||
381 | if (rtlpriv->dm.dm_type == DM_TYPE_BYDRIVER) | ||
382 | rtlpriv->dm.useramask = true; | ||
383 | else | ||
384 | rtlpriv->dm.useramask = false; | ||
385 | |||
386 | rtlpriv->dm.useramask = false; | ||
387 | rtlpriv->dm.inform_fw_driverctrldm = false; | ||
388 | } | ||
389 | |||
390 | static void _rtl92s_dm_init_txpowertracking_thermalmeter( | ||
391 | struct ieee80211_hw *hw) | ||
392 | { | ||
393 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
394 | |||
395 | rtlpriv->dm.txpower_tracking = true; | ||
396 | rtlpriv->dm.txpowercount = 0; | ||
397 | rtlpriv->dm.txpower_trackinginit = false; | ||
398 | } | ||
399 | |||
400 | static void _rtl92s_dm_false_alarm_counter_statistics(struct ieee80211_hw *hw) | ||
401 | { | ||
402 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
403 | struct false_alarm_statistics *falsealm_cnt = &(rtlpriv->falsealm_cnt); | ||
404 | u32 ret_value; | ||
405 | |||
406 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER1, MASKDWORD); | ||
407 | falsealm_cnt->cnt_parity_fail = ((ret_value & 0xffff0000) >> 16); | ||
408 | |||
409 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER2, MASKDWORD); | ||
410 | falsealm_cnt->cnt_rate_illegal = (ret_value & 0xffff); | ||
411 | falsealm_cnt->cnt_crc8_fail = ((ret_value & 0xffff0000) >> 16); | ||
412 | ret_value = rtl_get_bbreg(hw, ROFDM_PHYCOUNTER3, MASKDWORD); | ||
413 | falsealm_cnt->cnt_mcs_fail = (ret_value & 0xffff); | ||
414 | |||
415 | falsealm_cnt->cnt_ofdm_fail = falsealm_cnt->cnt_parity_fail + | ||
416 | falsealm_cnt->cnt_rate_illegal + falsealm_cnt->cnt_crc8_fail + | ||
417 | falsealm_cnt->cnt_mcs_fail; | ||
418 | |||
419 | /* read CCK false alarm */ | ||
420 | ret_value = rtl_get_bbreg(hw, 0xc64, MASKDWORD); | ||
421 | falsealm_cnt->cnt_cck_fail = (ret_value & 0xffff); | ||
422 | falsealm_cnt->cnt_all = falsealm_cnt->cnt_ofdm_fail + | ||
423 | falsealm_cnt->cnt_cck_fail; | ||
424 | } | ||
425 | |||
426 | static void rtl92s_backoff_enable_flag(struct ieee80211_hw *hw) | ||
427 | { | ||
428 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
429 | struct false_alarm_statistics *falsealm_cnt = &(rtlpriv->falsealm_cnt); | ||
430 | |||
431 | if (falsealm_cnt->cnt_all > digtable.fa_highthresh) { | ||
432 | if ((digtable.backoff_val - 6) < | ||
433 | digtable.backoffval_range_min) | ||
434 | digtable.backoff_val = digtable.backoffval_range_min; | ||
435 | else | ||
436 | digtable.backoff_val -= 6; | ||
437 | } else if (falsealm_cnt->cnt_all < digtable.fa_lowthresh) { | ||
438 | if ((digtable.backoff_val + 6) > | ||
439 | digtable.backoffval_range_max) | ||
440 | digtable.backoff_val = | ||
441 | digtable.backoffval_range_max; | ||
442 | else | ||
443 | digtable.backoff_val += 6; | ||
444 | } | ||
445 | } | ||
446 | |||
447 | static void _rtl92s_dm_initial_gain_sta_beforeconnect(struct ieee80211_hw *hw) | ||
448 | { | ||
449 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
450 | struct false_alarm_statistics *falsealm_cnt = &(rtlpriv->falsealm_cnt); | ||
451 | static u8 initialized, force_write; | ||
452 | u8 initial_gain = 0; | ||
453 | |||
454 | if ((digtable.pre_sta_connectstate == digtable.cur_sta_connectstate) || | ||
455 | (digtable.cur_sta_connectstate == DIG_STA_BEFORE_CONNECT)) { | ||
456 | if (digtable.cur_sta_connectstate == DIG_STA_BEFORE_CONNECT) { | ||
457 | if (rtlpriv->psc.rfpwr_state != ERFON) | ||
458 | return; | ||
459 | |||
460 | if (digtable.backoff_enable_flag == true) | ||
461 | rtl92s_backoff_enable_flag(hw); | ||
462 | else | ||
463 | digtable.backoff_val = DM_DIG_BACKOFF; | ||
464 | |||
465 | if ((digtable.rssi_val + 10 - digtable.backoff_val) > | ||
466 | digtable.rx_gain_range_max) | ||
467 | digtable.cur_igvalue = | ||
468 | digtable.rx_gain_range_max; | ||
469 | else if ((digtable.rssi_val + 10 - digtable.backoff_val) | ||
470 | < digtable.rx_gain_range_min) | ||
471 | digtable.cur_igvalue = | ||
472 | digtable.rx_gain_range_min; | ||
473 | else | ||
474 | digtable.cur_igvalue = digtable.rssi_val + 10 - | ||
475 | digtable.backoff_val; | ||
476 | |||
477 | if (falsealm_cnt->cnt_all > 10000) | ||
478 | digtable.cur_igvalue = | ||
479 | (digtable.cur_igvalue > 0x33) ? | ||
480 | digtable.cur_igvalue : 0x33; | ||
481 | |||
482 | if (falsealm_cnt->cnt_all > 16000) | ||
483 | digtable.cur_igvalue = | ||
484 | digtable.rx_gain_range_max; | ||
485 | /* connected -> connected or disconnected -> disconnected */ | ||
486 | } else { | ||
487 | /* Firmware control DIG, do nothing in driver dm */ | ||
488 | return; | ||
489 | } | ||
490 | /* disconnected -> connected or connected -> | ||
491 | * disconnected or beforeconnect->(dis)connected */ | ||
492 | } else { | ||
493 | /* Enable FW DIG */ | ||
494 | digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_MAX; | ||
495 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_DIG_ENABLE); | ||
496 | |||
497 | digtable.backoff_val = DM_DIG_BACKOFF; | ||
498 | digtable.cur_igvalue = rtlpriv->phy.default_initialgain[0]; | ||
499 | digtable.pre_igvalue = 0; | ||
500 | return; | ||
501 | } | ||
502 | |||
503 | /* Forced writing to prevent from fw-dig overwriting. */ | ||
504 | if (digtable.pre_igvalue != rtl_get_bbreg(hw, ROFDM0_XAAGCCORE1, | ||
505 | MASKBYTE0)) | ||
506 | force_write = 1; | ||
507 | |||
508 | if ((digtable.pre_igvalue != digtable.cur_igvalue) || | ||
509 | !initialized || force_write) { | ||
510 | /* Disable FW DIG */ | ||
511 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_DIG_DISABLE); | ||
512 | |||
513 | initial_gain = (u8)digtable.cur_igvalue; | ||
514 | |||
515 | /* Set initial gain. */ | ||
516 | rtl_set_bbreg(hw, ROFDM0_XAAGCCORE1, MASKBYTE0, initial_gain); | ||
517 | rtl_set_bbreg(hw, ROFDM0_XBAGCCORE1, MASKBYTE0, initial_gain); | ||
518 | digtable.pre_igvalue = digtable.cur_igvalue; | ||
519 | initialized = 1; | ||
520 | force_write = 0; | ||
521 | } | ||
522 | } | ||
523 | |||
524 | static void _rtl92s_dm_ctrl_initgain_bytwoport(struct ieee80211_hw *hw) | ||
525 | { | ||
526 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
527 | |||
528 | if (rtlpriv->mac80211.act_scanning) | ||
529 | return; | ||
530 | |||
531 | /* Decide the current status and if modify initial gain or not */ | ||
532 | if (rtlpriv->mac80211.link_state >= MAC80211_LINKED || | ||
533 | rtlpriv->mac80211.opmode == NL80211_IFTYPE_ADHOC) | ||
534 | digtable.cur_sta_connectstate = DIG_STA_CONNECT; | ||
535 | else | ||
536 | digtable.cur_sta_connectstate = DIG_STA_DISCONNECT; | ||
537 | |||
538 | digtable.rssi_val = rtlpriv->dm.undecorated_smoothed_pwdb; | ||
539 | |||
540 | /* Change dig mode to rssi */ | ||
541 | if (digtable.cur_sta_connectstate != DIG_STA_DISCONNECT) { | ||
542 | if (digtable.dig_twoport_algorithm == | ||
543 | DIG_TWO_PORT_ALGO_FALSE_ALARM) { | ||
544 | digtable.dig_twoport_algorithm = DIG_TWO_PORT_ALGO_RSSI; | ||
545 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_DIG_MODE_SS); | ||
546 | } | ||
547 | } | ||
548 | |||
549 | _rtl92s_dm_false_alarm_counter_statistics(hw); | ||
550 | _rtl92s_dm_initial_gain_sta_beforeconnect(hw); | ||
551 | |||
552 | digtable.pre_sta_connectstate = digtable.cur_sta_connectstate; | ||
553 | } | ||
554 | |||
555 | static void _rtl92s_dm_ctrl_initgain_byrssi(struct ieee80211_hw *hw) | ||
556 | { | ||
557 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
558 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
559 | |||
560 | /* 2T2R TP issue */ | ||
561 | if (rtlphy->rf_type == RF_2T2R) | ||
562 | return; | ||
563 | |||
564 | if (!rtlpriv->dm.dm_initialgain_enable) | ||
565 | return; | ||
566 | |||
567 | if (digtable.dig_enable_flag == false) | ||
568 | return; | ||
569 | |||
570 | _rtl92s_dm_ctrl_initgain_bytwoport(hw); | ||
571 | } | ||
572 | |||
573 | static void _rtl92s_dm_dynamic_txpower(struct ieee80211_hw *hw) | ||
574 | { | ||
575 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
576 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
577 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
578 | long undecorated_smoothed_pwdb; | ||
579 | long txpwr_threshold_lv1, txpwr_threshold_lv2; | ||
580 | |||
581 | /* 2T2R TP issue */ | ||
582 | if (rtlphy->rf_type == RF_2T2R) | ||
583 | return; | ||
584 | |||
585 | if (!rtlpriv->dm.dynamic_txpower_enable || | ||
586 | rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) { | ||
587 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
588 | return; | ||
589 | } | ||
590 | |||
591 | if ((mac->link_state < MAC80211_LINKED) && | ||
592 | (rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb == 0)) { | ||
593 | RT_TRACE(rtlpriv, COMP_POWER, DBG_TRACE, | ||
594 | ("Not connected to any\n")); | ||
595 | |||
596 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
597 | |||
598 | rtlpriv->dm.last_dtp_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
599 | return; | ||
600 | } | ||
601 | |||
602 | if (mac->link_state >= MAC80211_LINKED) { | ||
603 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
604 | undecorated_smoothed_pwdb = | ||
605 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
606 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
607 | ("AP Client PWDB = 0x%lx\n", | ||
608 | undecorated_smoothed_pwdb)); | ||
609 | } else { | ||
610 | undecorated_smoothed_pwdb = | ||
611 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
612 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
613 | ("STA Default Port PWDB = 0x%lx\n", | ||
614 | undecorated_smoothed_pwdb)); | ||
615 | } | ||
616 | } else { | ||
617 | undecorated_smoothed_pwdb = | ||
618 | rtlpriv->dm.entry_min_undecoratedsmoothed_pwdb; | ||
619 | |||
620 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
621 | ("AP Ext Port PWDB = 0x%lx\n", | ||
622 | undecorated_smoothed_pwdb)); | ||
623 | } | ||
624 | |||
625 | txpwr_threshold_lv2 = TX_POWER_NEAR_FIELD_THRESH_LVL2; | ||
626 | txpwr_threshold_lv1 = TX_POWER_NEAR_FIELD_THRESH_LVL1; | ||
627 | |||
628 | if (rtl_get_bbreg(hw, 0xc90, MASKBYTE0) == 1) | ||
629 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
630 | else if (undecorated_smoothed_pwdb >= txpwr_threshold_lv2) | ||
631 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL2; | ||
632 | else if ((undecorated_smoothed_pwdb < (txpwr_threshold_lv2 - 3)) && | ||
633 | (undecorated_smoothed_pwdb >= txpwr_threshold_lv1)) | ||
634 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL1; | ||
635 | else if (undecorated_smoothed_pwdb < (txpwr_threshold_lv1 - 3)) | ||
636 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
637 | |||
638 | if ((rtlpriv->dm.dynamic_txhighpower_lvl != rtlpriv->dm.last_dtp_lvl)) | ||
639 | rtl92s_phy_set_txpower(hw, rtlphy->current_channel); | ||
640 | |||
641 | rtlpriv->dm.last_dtp_lvl = rtlpriv->dm.dynamic_txhighpower_lvl; | ||
642 | } | ||
643 | |||
644 | static void _rtl92s_dm_init_dig(struct ieee80211_hw *hw) | ||
645 | { | ||
646 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
647 | |||
648 | /* Disable DIG scheme now.*/ | ||
649 | digtable.dig_enable_flag = true; | ||
650 | digtable.backoff_enable_flag = true; | ||
651 | |||
652 | if ((rtlpriv->dm.dm_type == DM_TYPE_BYDRIVER) && | ||
653 | (hal_get_firmwareversion(rtlpriv) >= 0x3c)) | ||
654 | digtable.dig_algorithm = DIG_ALGO_BY_TOW_PORT; | ||
655 | else | ||
656 | digtable.dig_algorithm = | ||
657 | DIG_ALGO_BEFORE_CONNECT_BY_RSSI_AND_ALARM; | ||
658 | |||
659 | digtable.dig_twoport_algorithm = DIG_TWO_PORT_ALGO_RSSI; | ||
660 | digtable.dig_ext_port_stage = DIG_EXT_PORT_STAGE_MAX; | ||
661 | /* off=by real rssi value, on=by digtable.rssi_val for new dig */ | ||
662 | digtable.dig_dbgmode = DM_DBG_OFF; | ||
663 | digtable.dig_slgorithm_switch = 0; | ||
664 | |||
665 | /* 2007/10/04 MH Define init gain threshol. */ | ||
666 | digtable.dig_state = DM_STA_DIG_MAX; | ||
667 | digtable.dig_highpwrstate = DM_STA_DIG_MAX; | ||
668 | |||
669 | digtable.cur_sta_connectstate = DIG_STA_DISCONNECT; | ||
670 | digtable.pre_sta_connectstate = DIG_STA_DISCONNECT; | ||
671 | digtable.cur_ap_connectstate = DIG_AP_DISCONNECT; | ||
672 | digtable.pre_ap_connectstate = DIG_AP_DISCONNECT; | ||
673 | |||
674 | digtable.rssi_lowthresh = DM_DIG_THRESH_LOW; | ||
675 | digtable.rssi_highthresh = DM_DIG_THRESH_HIGH; | ||
676 | |||
677 | digtable.fa_lowthresh = DM_FALSEALARM_THRESH_LOW; | ||
678 | digtable.fa_highthresh = DM_FALSEALARM_THRESH_HIGH; | ||
679 | |||
680 | digtable.rssi_highpower_lowthresh = DM_DIG_HIGH_PWR_THRESH_LOW; | ||
681 | digtable.rssi_highpower_highthresh = DM_DIG_HIGH_PWR_THRESH_HIGH; | ||
682 | |||
683 | /* for dig debug rssi value */ | ||
684 | digtable.rssi_val = 50; | ||
685 | digtable.backoff_val = DM_DIG_BACKOFF; | ||
686 | digtable.rx_gain_range_max = DM_DIG_MAX; | ||
687 | |||
688 | digtable.rx_gain_range_min = DM_DIG_MIN; | ||
689 | |||
690 | digtable.backoffval_range_max = DM_DIG_BACKOFF_MAX; | ||
691 | digtable.backoffval_range_min = DM_DIG_BACKOFF_MIN; | ||
692 | } | ||
693 | |||
694 | static void _rtl92s_dm_init_dynamic_txpower(struct ieee80211_hw *hw) | ||
695 | { | ||
696 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
697 | |||
698 | if ((hal_get_firmwareversion(rtlpriv) >= 60) && | ||
699 | (rtlpriv->dm.dm_type == DM_TYPE_BYDRIVER)) | ||
700 | rtlpriv->dm.dynamic_txpower_enable = true; | ||
701 | else | ||
702 | rtlpriv->dm.dynamic_txpower_enable = false; | ||
703 | |||
704 | rtlpriv->dm.last_dtp_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
705 | rtlpriv->dm.dynamic_txhighpower_lvl = TX_HIGHPWR_LEVEL_NORMAL; | ||
706 | } | ||
707 | |||
708 | void rtl92s_dm_init(struct ieee80211_hw *hw) | ||
709 | { | ||
710 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
711 | |||
712 | rtlpriv->dm.dm_type = DM_TYPE_BYDRIVER; | ||
713 | rtlpriv->dm.undecorated_smoothed_pwdb = -1; | ||
714 | |||
715 | _rtl92s_dm_init_dynamic_txpower(hw); | ||
716 | rtl92s_dm_init_edca_turbo(hw); | ||
717 | _rtl92s_dm_init_rate_adaptive_mask(hw); | ||
718 | _rtl92s_dm_init_txpowertracking_thermalmeter(hw); | ||
719 | _rtl92s_dm_init_dig(hw); | ||
720 | |||
721 | rtl_write_dword(rtlpriv, WFM5, FW_CCA_CHK_ENABLE); | ||
722 | } | ||
723 | |||
724 | void rtl92s_dm_watchdog(struct ieee80211_hw *hw) | ||
725 | { | ||
726 | _rtl92s_dm_check_edca_turbo(hw); | ||
727 | _rtl92s_dm_check_txpowertracking_thermalmeter(hw); | ||
728 | _rtl92s_dm_ctrl_initgain_byrssi(hw); | ||
729 | _rtl92s_dm_dynamic_txpower(hw); | ||
730 | _rtl92s_dm_refresh_rateadaptive_mask(hw); | ||
731 | _rtl92s_dm_switch_baseband_mrc(hw); | ||
732 | } | ||
733 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/dm.h b/drivers/net/wireless/rtlwifi/rtl8192se/dm.h new file mode 100644 index 000000000000..9051a556acc4 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/dm.h | |||
@@ -0,0 +1,164 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __RTL_92S_DM_H__ | ||
30 | #define __RTL_92S_DM_H__ | ||
31 | |||
32 | struct dig_t { | ||
33 | u8 dig_enable_flag; | ||
34 | u8 dig_algorithm; | ||
35 | u8 dig_twoport_algorithm; | ||
36 | u8 dig_ext_port_stage; | ||
37 | u8 dig_dbgmode; | ||
38 | u8 dig_slgorithm_switch; | ||
39 | |||
40 | long rssi_lowthresh; | ||
41 | long rssi_highthresh; | ||
42 | |||
43 | u32 fa_lowthresh; | ||
44 | u32 fa_highthresh; | ||
45 | |||
46 | long rssi_highpower_lowthresh; | ||
47 | long rssi_highpower_highthresh; | ||
48 | |||
49 | u8 dig_state; | ||
50 | u8 dig_highpwrstate; | ||
51 | u8 cur_sta_connectstate; | ||
52 | u8 pre_sta_connectstate; | ||
53 | u8 cur_ap_connectstate; | ||
54 | u8 pre_ap_connectstate; | ||
55 | |||
56 | u8 cur_pd_thstate; | ||
57 | u8 pre_pd_thstate; | ||
58 | u8 cur_cs_ratiostate; | ||
59 | u8 pre_cs_ratiostate; | ||
60 | |||
61 | u32 pre_igvalue; | ||
62 | u32 cur_igvalue; | ||
63 | |||
64 | u8 backoff_enable_flag; | ||
65 | char backoff_val; | ||
66 | char backoffval_range_max; | ||
67 | char backoffval_range_min; | ||
68 | u8 rx_gain_range_max; | ||
69 | u8 rx_gain_range_min; | ||
70 | |||
71 | long rssi_val; | ||
72 | }; | ||
73 | |||
74 | enum dm_dig_alg { | ||
75 | DIG_ALGO_BY_FALSE_ALARM = 0, | ||
76 | DIG_ALGO_BY_RSSI = 1, | ||
77 | DIG_ALGO_BEFORE_CONNECT_BY_RSSI_AND_ALARM = 2, | ||
78 | DIG_ALGO_BY_TOW_PORT = 3, | ||
79 | DIG_ALGO_MAX | ||
80 | }; | ||
81 | |||
82 | enum dm_dig_two_port_alg { | ||
83 | DIG_TWO_PORT_ALGO_RSSI = 0, | ||
84 | DIG_TWO_PORT_ALGO_FALSE_ALARM = 1, | ||
85 | }; | ||
86 | |||
87 | enum dm_dig_dbg { | ||
88 | DM_DBG_OFF = 0, | ||
89 | DM_DBG_ON = 1, | ||
90 | DM_DBG_MAX | ||
91 | }; | ||
92 | |||
93 | enum dm_dig_sta { | ||
94 | DM_STA_DIG_OFF = 0, | ||
95 | DM_STA_DIG_ON, | ||
96 | DM_STA_DIG_MAX | ||
97 | }; | ||
98 | |||
99 | enum dm_dig_connect { | ||
100 | DIG_STA_DISCONNECT = 0, | ||
101 | DIG_STA_CONNECT = 1, | ||
102 | DIG_STA_BEFORE_CONNECT = 2, | ||
103 | DIG_AP_DISCONNECT = 3, | ||
104 | DIG_AP_CONNECT = 4, | ||
105 | DIG_AP_ADD_STATION = 5, | ||
106 | DIG_CONNECT_MAX | ||
107 | }; | ||
108 | |||
109 | enum dm_dig_ext_port_alg { | ||
110 | DIG_EXT_PORT_STAGE_0 = 0, | ||
111 | DIG_EXT_PORT_STAGE_1 = 1, | ||
112 | DIG_EXT_PORT_STAGE_2 = 2, | ||
113 | DIG_EXT_PORT_STAGE_3 = 3, | ||
114 | DIG_EXT_PORT_STAGE_MAX = 4, | ||
115 | }; | ||
116 | |||
117 | enum dm_ratr_sta { | ||
118 | DM_RATR_STA_HIGH = 0, | ||
119 | DM_RATR_STA_MIDDLEHIGH = 1, | ||
120 | DM_RATR_STA_MIDDLE = 2, | ||
121 | DM_RATR_STA_MIDDLELOW = 3, | ||
122 | DM_RATR_STA_LOW = 4, | ||
123 | DM_RATR_STA_ULTRALOW = 5, | ||
124 | DM_RATR_STA_MAX | ||
125 | }; | ||
126 | |||
127 | #define DM_TYPE_BYFW 0 | ||
128 | #define DM_TYPE_BYDRIVER 1 | ||
129 | |||
130 | #define TX_HIGH_PWR_LEVEL_NORMAL 0 | ||
131 | #define TX_HIGH_PWR_LEVEL_LEVEL1 1 | ||
132 | #define TX_HIGH_PWR_LEVEL_LEVEL2 2 | ||
133 | |||
134 | #define HAL_DM_DIG_DISABLE BIT(0) /* Disable Dig */ | ||
135 | #define HAL_DM_HIPWR_DISABLE BIT(1) /* Disable High Power */ | ||
136 | |||
137 | #define TX_HIGHPWR_LEVEL_NORMAL 0 | ||
138 | #define TX_HIGHPWR_LEVEL_NORMAL1 1 | ||
139 | #define TX_HIGHPWR_LEVEL_NORMAL2 2 | ||
140 | |||
141 | #define TX_POWER_NEAR_FIELD_THRESH_LVL2 74 | ||
142 | #define TX_POWER_NEAR_FIELD_THRESH_LVL1 67 | ||
143 | |||
144 | #define DM_DIG_THRESH_HIGH 40 | ||
145 | #define DM_DIG_THRESH_LOW 35 | ||
146 | #define DM_FALSEALARM_THRESH_LOW 40 | ||
147 | #define DM_FALSEALARM_THRESH_HIGH 1000 | ||
148 | #define DM_DIG_HIGH_PWR_THRESH_HIGH 75 | ||
149 | #define DM_DIG_HIGH_PWR_THRESH_LOW 70 | ||
150 | #define DM_DIG_BACKOFF 12 | ||
151 | #define DM_DIG_MAX 0x3e | ||
152 | #define DM_DIG_MIN 0x1c | ||
153 | #define DM_DIG_MIN_Netcore 0x12 | ||
154 | #define DM_DIG_BACKOFF_MAX 12 | ||
155 | #define DM_DIG_BACKOFF_MIN -4 | ||
156 | |||
157 | extern struct dig_t digtable; | ||
158 | |||
159 | void rtl92s_dm_watchdog(struct ieee80211_hw *hw); | ||
160 | void rtl92s_dm_init(struct ieee80211_hw *hw); | ||
161 | void rtl92s_dm_init_edca_turbo(struct ieee80211_hw *hw); | ||
162 | |||
163 | #endif | ||
164 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/fw.c b/drivers/net/wireless/rtlwifi/rtl8192se/fw.c new file mode 100644 index 000000000000..3b5af0113d7f --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/fw.c | |||
@@ -0,0 +1,654 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../base.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "fw.h" | ||
36 | |||
37 | static void _rtl92s_fw_set_rqpn(struct ieee80211_hw *hw) | ||
38 | { | ||
39 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
40 | |||
41 | rtl_write_dword(rtlpriv, RQPN, 0xffffffff); | ||
42 | rtl_write_dword(rtlpriv, RQPN + 4, 0xffffffff); | ||
43 | rtl_write_byte(rtlpriv, RQPN + 8, 0xff); | ||
44 | rtl_write_byte(rtlpriv, RQPN + 0xB, 0x80); | ||
45 | } | ||
46 | |||
47 | static bool _rtl92s_firmware_enable_cpu(struct ieee80211_hw *hw) | ||
48 | { | ||
49 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
50 | u32 ichecktime = 200; | ||
51 | u16 tmpu2b; | ||
52 | u8 tmpu1b, cpustatus = 0; | ||
53 | |||
54 | _rtl92s_fw_set_rqpn(hw); | ||
55 | |||
56 | /* Enable CPU. */ | ||
57 | tmpu1b = rtl_read_byte(rtlpriv, SYS_CLKR); | ||
58 | /* AFE source */ | ||
59 | rtl_write_byte(rtlpriv, SYS_CLKR, (tmpu1b | SYS_CPU_CLKSEL)); | ||
60 | |||
61 | tmpu2b = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
62 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (tmpu2b | FEN_CPUEN)); | ||
63 | |||
64 | /* Polling IMEM Ready after CPU has refilled. */ | ||
65 | do { | ||
66 | cpustatus = rtl_read_byte(rtlpriv, TCR); | ||
67 | if (cpustatus & IMEM_RDY) { | ||
68 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
69 | ("IMEM Ready after CPU has refilled.\n")); | ||
70 | break; | ||
71 | } | ||
72 | |||
73 | udelay(100); | ||
74 | } while (ichecktime--); | ||
75 | |||
76 | if (!(cpustatus & IMEM_RDY)) | ||
77 | return false; | ||
78 | |||
79 | return true; | ||
80 | } | ||
81 | |||
82 | static enum fw_status _rtl92s_firmware_get_nextstatus( | ||
83 | enum fw_status fw_currentstatus) | ||
84 | { | ||
85 | enum fw_status next_fwstatus = 0; | ||
86 | |||
87 | switch (fw_currentstatus) { | ||
88 | case FW_STATUS_INIT: | ||
89 | next_fwstatus = FW_STATUS_LOAD_IMEM; | ||
90 | break; | ||
91 | case FW_STATUS_LOAD_IMEM: | ||
92 | next_fwstatus = FW_STATUS_LOAD_EMEM; | ||
93 | break; | ||
94 | case FW_STATUS_LOAD_EMEM: | ||
95 | next_fwstatus = FW_STATUS_LOAD_DMEM; | ||
96 | break; | ||
97 | case FW_STATUS_LOAD_DMEM: | ||
98 | next_fwstatus = FW_STATUS_READY; | ||
99 | break; | ||
100 | default: | ||
101 | break; | ||
102 | } | ||
103 | |||
104 | return next_fwstatus; | ||
105 | } | ||
106 | |||
107 | static u8 _rtl92s_firmware_header_map_rftype(struct ieee80211_hw *hw) | ||
108 | { | ||
109 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
110 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
111 | |||
112 | switch (rtlphy->rf_type) { | ||
113 | case RF_1T1R: | ||
114 | return 0x11; | ||
115 | break; | ||
116 | case RF_1T2R: | ||
117 | return 0x12; | ||
118 | break; | ||
119 | case RF_2T2R: | ||
120 | return 0x22; | ||
121 | break; | ||
122 | default: | ||
123 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
124 | ("Unknown RF type(%x)\n", | ||
125 | rtlphy->rf_type)); | ||
126 | break; | ||
127 | } | ||
128 | return 0x22; | ||
129 | } | ||
130 | |||
131 | static void _rtl92s_firmwareheader_priveupdate(struct ieee80211_hw *hw, | ||
132 | struct fw_priv *pfw_priv) | ||
133 | { | ||
134 | /* Update RF types for RATR settings. */ | ||
135 | pfw_priv->rf_config = _rtl92s_firmware_header_map_rftype(hw); | ||
136 | } | ||
137 | |||
138 | |||
139 | |||
140 | static bool _rtl92s_cmd_send_packet(struct ieee80211_hw *hw, | ||
141 | struct sk_buff *skb, u8 last) | ||
142 | { | ||
143 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
144 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
145 | struct rtl8192_tx_ring *ring; | ||
146 | struct rtl_tx_desc *pdesc; | ||
147 | unsigned long flags; | ||
148 | u8 idx = 0; | ||
149 | |||
150 | ring = &rtlpci->tx_ring[TXCMD_QUEUE]; | ||
151 | |||
152 | spin_lock_irqsave(&rtlpriv->locks.irq_th_lock, flags); | ||
153 | |||
154 | idx = (ring->idx + skb_queue_len(&ring->queue)) % ring->entries; | ||
155 | pdesc = &ring->desc[idx]; | ||
156 | rtlpriv->cfg->ops->fill_tx_cmddesc(hw, (u8 *)pdesc, 1, 1, skb); | ||
157 | __skb_queue_tail(&ring->queue, skb); | ||
158 | |||
159 | spin_unlock_irqrestore(&rtlpriv->locks.irq_th_lock, flags); | ||
160 | |||
161 | return true; | ||
162 | } | ||
163 | |||
164 | static bool _rtl92s_firmware_downloadcode(struct ieee80211_hw *hw, | ||
165 | u8 *code_virtual_address, u32 buffer_len) | ||
166 | { | ||
167 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
168 | struct sk_buff *skb; | ||
169 | struct rtl_tcb_desc *tcb_desc; | ||
170 | unsigned char *seg_ptr; | ||
171 | u16 frag_threshold = MAX_FIRMWARE_CODE_SIZE; | ||
172 | u16 frag_length, frag_offset = 0; | ||
173 | u16 extra_descoffset = 0; | ||
174 | u8 last_inipkt = 0; | ||
175 | |||
176 | _rtl92s_fw_set_rqpn(hw); | ||
177 | |||
178 | if (buffer_len >= MAX_FIRMWARE_CODE_SIZE) { | ||
179 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
180 | ("Size over FIRMWARE_CODE_SIZE!\n")); | ||
181 | |||
182 | return false; | ||
183 | } | ||
184 | |||
185 | extra_descoffset = 0; | ||
186 | |||
187 | do { | ||
188 | if ((buffer_len - frag_offset) > frag_threshold) { | ||
189 | frag_length = frag_threshold + extra_descoffset; | ||
190 | } else { | ||
191 | frag_length = (u16)(buffer_len - frag_offset + | ||
192 | extra_descoffset); | ||
193 | last_inipkt = 1; | ||
194 | } | ||
195 | |||
196 | /* Allocate skb buffer to contain firmware */ | ||
197 | /* info and tx descriptor info. */ | ||
198 | skb = dev_alloc_skb(frag_length); | ||
199 | skb_reserve(skb, extra_descoffset); | ||
200 | seg_ptr = (u8 *)skb_put(skb, (u32)(frag_length - | ||
201 | extra_descoffset)); | ||
202 | memcpy(seg_ptr, code_virtual_address + frag_offset, | ||
203 | (u32)(frag_length - extra_descoffset)); | ||
204 | |||
205 | tcb_desc = (struct rtl_tcb_desc *)(skb->cb); | ||
206 | tcb_desc->queue_index = TXCMD_QUEUE; | ||
207 | tcb_desc->cmd_or_init = DESC_PACKET_TYPE_INIT; | ||
208 | tcb_desc->last_inipkt = last_inipkt; | ||
209 | |||
210 | _rtl92s_cmd_send_packet(hw, skb, last_inipkt); | ||
211 | |||
212 | frag_offset += (frag_length - extra_descoffset); | ||
213 | |||
214 | } while (frag_offset < buffer_len); | ||
215 | |||
216 | rtl_write_byte(rtlpriv, TP_POLL, TPPOLL_CQ); | ||
217 | |||
218 | return true ; | ||
219 | } | ||
220 | |||
221 | static bool _rtl92s_firmware_checkready(struct ieee80211_hw *hw, | ||
222 | u8 loadfw_status) | ||
223 | { | ||
224 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
225 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
226 | struct rt_firmware *firmware = (struct rt_firmware *)rtlhal->pfirmware; | ||
227 | u32 tmpu4b; | ||
228 | u8 cpustatus = 0; | ||
229 | short pollingcnt = 1000; | ||
230 | bool rtstatus = true; | ||
231 | |||
232 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("LoadStaus(%d)\n", | ||
233 | loadfw_status)); | ||
234 | |||
235 | firmware->fwstatus = (enum fw_status)loadfw_status; | ||
236 | |||
237 | switch (loadfw_status) { | ||
238 | case FW_STATUS_LOAD_IMEM: | ||
239 | /* Polling IMEM code done. */ | ||
240 | do { | ||
241 | cpustatus = rtl_read_byte(rtlpriv, TCR); | ||
242 | if (cpustatus & IMEM_CODE_DONE) | ||
243 | break; | ||
244 | udelay(5); | ||
245 | } while (pollingcnt--); | ||
246 | |||
247 | if (!(cpustatus & IMEM_CHK_RPT) || (pollingcnt <= 0)) { | ||
248 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
249 | ("FW_STATUS_LOAD_IMEM" | ||
250 | " FAIL CPU, Status=%x\r\n", cpustatus)); | ||
251 | goto status_check_fail; | ||
252 | } | ||
253 | break; | ||
254 | |||
255 | case FW_STATUS_LOAD_EMEM: | ||
256 | /* Check Put Code OK and Turn On CPU */ | ||
257 | /* Polling EMEM code done. */ | ||
258 | do { | ||
259 | cpustatus = rtl_read_byte(rtlpriv, TCR); | ||
260 | if (cpustatus & EMEM_CODE_DONE) | ||
261 | break; | ||
262 | udelay(5); | ||
263 | } while (pollingcnt--); | ||
264 | |||
265 | if (!(cpustatus & EMEM_CHK_RPT) || (pollingcnt <= 0)) { | ||
266 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
267 | ("FW_STATUS_LOAD_EMEM" | ||
268 | " FAIL CPU, Status=%x\r\n", cpustatus)); | ||
269 | goto status_check_fail; | ||
270 | } | ||
271 | |||
272 | /* Turn On CPU */ | ||
273 | rtstatus = _rtl92s_firmware_enable_cpu(hw); | ||
274 | if (rtstatus != true) { | ||
275 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
276 | ("Enable CPU fail!\n")); | ||
277 | goto status_check_fail; | ||
278 | } | ||
279 | break; | ||
280 | |||
281 | case FW_STATUS_LOAD_DMEM: | ||
282 | /* Polling DMEM code done */ | ||
283 | do { | ||
284 | cpustatus = rtl_read_byte(rtlpriv, TCR); | ||
285 | if (cpustatus & DMEM_CODE_DONE) | ||
286 | break; | ||
287 | udelay(5); | ||
288 | } while (pollingcnt--); | ||
289 | |||
290 | if (!(cpustatus & DMEM_CODE_DONE) || (pollingcnt <= 0)) { | ||
291 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
292 | ("Polling DMEM code done" | ||
293 | " fail ! cpustatus(%#x)\n", cpustatus)); | ||
294 | goto status_check_fail; | ||
295 | } | ||
296 | |||
297 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
298 | ("DMEM code download success," | ||
299 | " cpustatus(%#x)\n", cpustatus)); | ||
300 | |||
301 | /* Prevent Delay too much and being scheduled out */ | ||
302 | /* Polling Load Firmware ready */ | ||
303 | pollingcnt = 2000; | ||
304 | do { | ||
305 | cpustatus = rtl_read_byte(rtlpriv, TCR); | ||
306 | if (cpustatus & FWRDY) | ||
307 | break; | ||
308 | udelay(40); | ||
309 | } while (pollingcnt--); | ||
310 | |||
311 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
312 | ("Polling Load Firmware ready," | ||
313 | " cpustatus(%x)\n", cpustatus)); | ||
314 | |||
315 | if (((cpustatus & LOAD_FW_READY) != LOAD_FW_READY) || | ||
316 | (pollingcnt <= 0)) { | ||
317 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
318 | ("Polling Load Firmware" | ||
319 | " ready fail ! cpustatus(%x)\n", cpustatus)); | ||
320 | goto status_check_fail; | ||
321 | } | ||
322 | |||
323 | /* If right here, we can set TCR/RCR to desired value */ | ||
324 | /* and config MAC lookback mode to normal mode */ | ||
325 | tmpu4b = rtl_read_dword(rtlpriv, TCR); | ||
326 | rtl_write_dword(rtlpriv, TCR, (tmpu4b & (~TCR_ICV))); | ||
327 | |||
328 | tmpu4b = rtl_read_dword(rtlpriv, RCR); | ||
329 | rtl_write_dword(rtlpriv, RCR, (tmpu4b | RCR_APPFCS | | ||
330 | RCR_APP_ICV | RCR_APP_MIC)); | ||
331 | |||
332 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
333 | ("Current RCR settings(%#x)\n", tmpu4b)); | ||
334 | |||
335 | /* Set to normal mode. */ | ||
336 | rtl_write_byte(rtlpriv, LBKMD_SEL, LBK_NORMAL); | ||
337 | break; | ||
338 | |||
339 | default: | ||
340 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
341 | ("Unknown status check!\n")); | ||
342 | rtstatus = false; | ||
343 | break; | ||
344 | } | ||
345 | |||
346 | status_check_fail: | ||
347 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("loadfw_status(%d), " | ||
348 | "rtstatus(%x)\n", loadfw_status, rtstatus)); | ||
349 | return rtstatus; | ||
350 | } | ||
351 | |||
352 | int rtl92s_download_fw(struct ieee80211_hw *hw) | ||
353 | { | ||
354 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
355 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
356 | struct rt_firmware *firmware = NULL; | ||
357 | struct fw_hdr *pfwheader; | ||
358 | struct fw_priv *pfw_priv = NULL; | ||
359 | u8 *puc_mappedfile = NULL; | ||
360 | u32 ul_filelength = 0; | ||
361 | u32 file_length = 0; | ||
362 | u8 fwhdr_size = RT_8192S_FIRMWARE_HDR_SIZE; | ||
363 | u8 fwstatus = FW_STATUS_INIT; | ||
364 | bool rtstatus = true; | ||
365 | |||
366 | if (!rtlhal->pfirmware) | ||
367 | return 1; | ||
368 | |||
369 | firmware = (struct rt_firmware *)rtlhal->pfirmware; | ||
370 | firmware->fwstatus = FW_STATUS_INIT; | ||
371 | |||
372 | puc_mappedfile = firmware->sz_fw_tmpbuffer; | ||
373 | file_length = firmware->sz_fw_tmpbufferlen; | ||
374 | |||
375 | /* 1. Retrieve FW header. */ | ||
376 | firmware->pfwheader = (struct fw_hdr *) puc_mappedfile; | ||
377 | pfwheader = firmware->pfwheader; | ||
378 | firmware->firmwareversion = byte(pfwheader->version, 0); | ||
379 | firmware->pfwheader->fwpriv.hci_sel = 1;/* pcie */ | ||
380 | |||
381 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("signature:%x, version:" | ||
382 | "%x, size:%x," | ||
383 | "imemsize:%x, sram size:%x\n", pfwheader->signature, | ||
384 | pfwheader->version, pfwheader->dmem_size, | ||
385 | pfwheader->img_imem_size, pfwheader->img_sram_size)); | ||
386 | |||
387 | /* 2. Retrieve IMEM image. */ | ||
388 | if ((pfwheader->img_imem_size == 0) || (pfwheader->img_imem_size > | ||
389 | sizeof(firmware->fw_imem))) { | ||
390 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
391 | ("memory for data image is less than IMEM required\n")); | ||
392 | goto fail; | ||
393 | } else { | ||
394 | puc_mappedfile += fwhdr_size; | ||
395 | |||
396 | memcpy(firmware->fw_imem, puc_mappedfile, | ||
397 | pfwheader->img_imem_size); | ||
398 | firmware->fw_imem_len = pfwheader->img_imem_size; | ||
399 | } | ||
400 | |||
401 | /* 3. Retriecve EMEM image. */ | ||
402 | if (pfwheader->img_sram_size > sizeof(firmware->fw_emem)) { | ||
403 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
404 | ("memory for data image is less than EMEM required\n")); | ||
405 | goto fail; | ||
406 | } else { | ||
407 | puc_mappedfile += firmware->fw_imem_len; | ||
408 | |||
409 | memcpy(firmware->fw_emem, puc_mappedfile, | ||
410 | pfwheader->img_sram_size); | ||
411 | firmware->fw_emem_len = pfwheader->img_sram_size; | ||
412 | } | ||
413 | |||
414 | /* 4. download fw now */ | ||
415 | fwstatus = _rtl92s_firmware_get_nextstatus(firmware->fwstatus); | ||
416 | while (fwstatus != FW_STATUS_READY) { | ||
417 | /* Image buffer redirection. */ | ||
418 | switch (fwstatus) { | ||
419 | case FW_STATUS_LOAD_IMEM: | ||
420 | puc_mappedfile = firmware->fw_imem; | ||
421 | ul_filelength = firmware->fw_imem_len; | ||
422 | break; | ||
423 | case FW_STATUS_LOAD_EMEM: | ||
424 | puc_mappedfile = firmware->fw_emem; | ||
425 | ul_filelength = firmware->fw_emem_len; | ||
426 | break; | ||
427 | case FW_STATUS_LOAD_DMEM: | ||
428 | /* Partial update the content of header private. */ | ||
429 | pfwheader = firmware->pfwheader; | ||
430 | pfw_priv = &pfwheader->fwpriv; | ||
431 | _rtl92s_firmwareheader_priveupdate(hw, pfw_priv); | ||
432 | puc_mappedfile = (u8 *)(firmware->pfwheader) + | ||
433 | RT_8192S_FIRMWARE_HDR_EXCLUDE_PRI_SIZE; | ||
434 | ul_filelength = fwhdr_size - | ||
435 | RT_8192S_FIRMWARE_HDR_EXCLUDE_PRI_SIZE; | ||
436 | break; | ||
437 | default: | ||
438 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
439 | ("Unexpected Download step!!\n")); | ||
440 | goto fail; | ||
441 | break; | ||
442 | } | ||
443 | |||
444 | /* <2> Download image file */ | ||
445 | rtstatus = _rtl92s_firmware_downloadcode(hw, puc_mappedfile, | ||
446 | ul_filelength); | ||
447 | |||
448 | if (rtstatus != true) { | ||
449 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("fail!\n")); | ||
450 | goto fail; | ||
451 | } | ||
452 | |||
453 | /* <3> Check whether load FW process is ready */ | ||
454 | rtstatus = _rtl92s_firmware_checkready(hw, fwstatus); | ||
455 | if (rtstatus != true) { | ||
456 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("fail!\n")); | ||
457 | goto fail; | ||
458 | } | ||
459 | |||
460 | fwstatus = _rtl92s_firmware_get_nextstatus(firmware->fwstatus); | ||
461 | } | ||
462 | |||
463 | return rtstatus; | ||
464 | fail: | ||
465 | return 0; | ||
466 | } | ||
467 | |||
468 | static u32 _rtl92s_fill_h2c_cmd(struct sk_buff *skb, u32 h2cbufferlen, | ||
469 | u32 cmd_num, u32 *pelement_id, u32 *pcmd_len, | ||
470 | u8 **pcmb_buffer, u8 *cmd_start_seq) | ||
471 | { | ||
472 | u32 totallen = 0, len = 0, tx_desclen = 0; | ||
473 | u32 pre_continueoffset = 0; | ||
474 | u8 *ph2c_buffer; | ||
475 | u8 i = 0; | ||
476 | |||
477 | do { | ||
478 | /* 8 - Byte aligment */ | ||
479 | len = H2C_TX_CMD_HDR_LEN + N_BYTE_ALIGMENT(pcmd_len[i], 8); | ||
480 | |||
481 | /* Buffer length is not enough */ | ||
482 | if (h2cbufferlen < totallen + len + tx_desclen) | ||
483 | break; | ||
484 | |||
485 | /* Clear content */ | ||
486 | ph2c_buffer = (u8 *)skb_put(skb, (u32)len); | ||
487 | memset((ph2c_buffer + totallen + tx_desclen), 0, len); | ||
488 | |||
489 | /* CMD len */ | ||
490 | SET_BITS_TO_LE_4BYTE((ph2c_buffer + totallen + tx_desclen), | ||
491 | 0, 16, pcmd_len[i]); | ||
492 | |||
493 | /* CMD ID */ | ||
494 | SET_BITS_TO_LE_4BYTE((ph2c_buffer + totallen + tx_desclen), | ||
495 | 16, 8, pelement_id[i]); | ||
496 | |||
497 | /* CMD Sequence */ | ||
498 | *cmd_start_seq = *cmd_start_seq % 0x80; | ||
499 | SET_BITS_TO_LE_4BYTE((ph2c_buffer + totallen + tx_desclen), | ||
500 | 24, 7, *cmd_start_seq); | ||
501 | ++*cmd_start_seq; | ||
502 | |||
503 | /* Copy memory */ | ||
504 | memcpy((ph2c_buffer + totallen + tx_desclen + | ||
505 | H2C_TX_CMD_HDR_LEN), pcmb_buffer[i], pcmd_len[i]); | ||
506 | |||
507 | /* CMD continue */ | ||
508 | /* set the continue in prevoius cmd. */ | ||
509 | if (i < cmd_num - 1) | ||
510 | SET_BITS_TO_LE_4BYTE((ph2c_buffer + pre_continueoffset), | ||
511 | 31, 1, 1); | ||
512 | |||
513 | pre_continueoffset = totallen; | ||
514 | |||
515 | totallen += len; | ||
516 | } while (++i < cmd_num); | ||
517 | |||
518 | return totallen; | ||
519 | } | ||
520 | |||
521 | static u32 _rtl92s_get_h2c_cmdlen(u32 h2cbufferlen, u32 cmd_num, u32 *pcmd_len) | ||
522 | { | ||
523 | u32 totallen = 0, len = 0, tx_desclen = 0; | ||
524 | u8 i = 0; | ||
525 | |||
526 | do { | ||
527 | /* 8 - Byte aligment */ | ||
528 | len = H2C_TX_CMD_HDR_LEN + N_BYTE_ALIGMENT(pcmd_len[i], 8); | ||
529 | |||
530 | /* Buffer length is not enough */ | ||
531 | if (h2cbufferlen < totallen + len + tx_desclen) | ||
532 | break; | ||
533 | |||
534 | totallen += len; | ||
535 | } while (++i < cmd_num); | ||
536 | |||
537 | return totallen + tx_desclen; | ||
538 | } | ||
539 | |||
540 | static bool _rtl92s_firmware_set_h2c_cmd(struct ieee80211_hw *hw, u8 h2c_cmd, | ||
541 | u8 *pcmd_buffer) | ||
542 | { | ||
543 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
544 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
545 | struct rtl_tcb_desc *cb_desc; | ||
546 | struct sk_buff *skb; | ||
547 | u32 element_id = 0; | ||
548 | u32 cmd_len = 0; | ||
549 | u32 len; | ||
550 | |||
551 | switch (h2c_cmd) { | ||
552 | case FW_H2C_SETPWRMODE: | ||
553 | element_id = H2C_SETPWRMODE_CMD ; | ||
554 | cmd_len = sizeof(struct h2c_set_pwrmode_parm); | ||
555 | break; | ||
556 | case FW_H2C_JOINBSSRPT: | ||
557 | element_id = H2C_JOINBSSRPT_CMD; | ||
558 | cmd_len = sizeof(struct h2c_joinbss_rpt_parm); | ||
559 | break; | ||
560 | case FW_H2C_WOWLAN_UPDATE_GTK: | ||
561 | element_id = H2C_WOWLAN_UPDATE_GTK_CMD; | ||
562 | cmd_len = sizeof(struct h2c_wpa_two_way_parm); | ||
563 | break; | ||
564 | case FW_H2C_WOWLAN_UPDATE_IV: | ||
565 | element_id = H2C_WOWLAN_UPDATE_IV_CMD; | ||
566 | cmd_len = sizeof(unsigned long long); | ||
567 | break; | ||
568 | case FW_H2C_WOWLAN_OFFLOAD: | ||
569 | element_id = H2C_WOWLAN_FW_OFFLOAD; | ||
570 | cmd_len = sizeof(u8); | ||
571 | break; | ||
572 | default: | ||
573 | break; | ||
574 | } | ||
575 | |||
576 | len = _rtl92s_get_h2c_cmdlen(MAX_TRANSMIT_BUFFER_SIZE, 1, &cmd_len); | ||
577 | skb = dev_alloc_skb(len); | ||
578 | cb_desc = (struct rtl_tcb_desc *)(skb->cb); | ||
579 | cb_desc->queue_index = TXCMD_QUEUE; | ||
580 | cb_desc->cmd_or_init = DESC_PACKET_TYPE_NORMAL; | ||
581 | cb_desc->last_inipkt = false; | ||
582 | |||
583 | _rtl92s_fill_h2c_cmd(skb, MAX_TRANSMIT_BUFFER_SIZE, 1, &element_id, | ||
584 | &cmd_len, &pcmd_buffer, &rtlhal->h2c_txcmd_seq); | ||
585 | _rtl92s_cmd_send_packet(hw, skb, false); | ||
586 | rtlpriv->cfg->ops->tx_polling(hw, TXCMD_QUEUE); | ||
587 | |||
588 | return true; | ||
589 | } | ||
590 | |||
591 | void rtl92s_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 Mode) | ||
592 | { | ||
593 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
594 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
595 | struct h2c_set_pwrmode_parm pwrmode; | ||
596 | u16 max_wakeup_period = 0; | ||
597 | |||
598 | pwrmode.mode = Mode; | ||
599 | pwrmode.flag_low_traffic_en = 0; | ||
600 | pwrmode.flag_lpnav_en = 0; | ||
601 | pwrmode.flag_rf_low_snr_en = 0; | ||
602 | pwrmode.flag_dps_en = 0; | ||
603 | pwrmode.bcn_rx_en = 0; | ||
604 | pwrmode.bcn_to = 0; | ||
605 | SET_BITS_TO_LE_2BYTE((u8 *)(&pwrmode) + 8, 0, 16, | ||
606 | mac->vif->bss_conf.beacon_int); | ||
607 | pwrmode.app_itv = 0; | ||
608 | pwrmode.awake_bcn_itvl = ppsc->reg_max_lps_awakeintvl; | ||
609 | pwrmode.smart_ps = 1; | ||
610 | pwrmode.bcn_pass_period = 10; | ||
611 | |||
612 | /* Set beacon pass count */ | ||
613 | if (pwrmode.mode == FW_PS_MIN_MODE) | ||
614 | max_wakeup_period = mac->vif->bss_conf.beacon_int; | ||
615 | else if (pwrmode.mode == FW_PS_MAX_MODE) | ||
616 | max_wakeup_period = mac->vif->bss_conf.beacon_int * | ||
617 | mac->vif->bss_conf.dtim_period; | ||
618 | |||
619 | if (max_wakeup_period >= 500) | ||
620 | pwrmode.bcn_pass_cnt = 1; | ||
621 | else if ((max_wakeup_period >= 300) && (max_wakeup_period < 500)) | ||
622 | pwrmode.bcn_pass_cnt = 2; | ||
623 | else if ((max_wakeup_period >= 200) && (max_wakeup_period < 300)) | ||
624 | pwrmode.bcn_pass_cnt = 3; | ||
625 | else if ((max_wakeup_period >= 20) && (max_wakeup_period < 200)) | ||
626 | pwrmode.bcn_pass_cnt = 5; | ||
627 | else | ||
628 | pwrmode.bcn_pass_cnt = 1; | ||
629 | |||
630 | _rtl92s_firmware_set_h2c_cmd(hw, FW_H2C_SETPWRMODE, (u8 *)&pwrmode); | ||
631 | |||
632 | } | ||
633 | |||
634 | void rtl92s_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, | ||
635 | u8 mstatus, u8 ps_qosinfo) | ||
636 | { | ||
637 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
638 | struct h2c_joinbss_rpt_parm joinbss_rpt; | ||
639 | |||
640 | joinbss_rpt.opmode = mstatus; | ||
641 | joinbss_rpt.ps_qos_info = ps_qosinfo; | ||
642 | joinbss_rpt.bssid[0] = mac->bssid[0]; | ||
643 | joinbss_rpt.bssid[1] = mac->bssid[1]; | ||
644 | joinbss_rpt.bssid[2] = mac->bssid[2]; | ||
645 | joinbss_rpt.bssid[3] = mac->bssid[3]; | ||
646 | joinbss_rpt.bssid[4] = mac->bssid[4]; | ||
647 | joinbss_rpt.bssid[5] = mac->bssid[5]; | ||
648 | SET_BITS_TO_LE_2BYTE((u8 *)(&joinbss_rpt) + 8, 0, 16, | ||
649 | mac->vif->bss_conf.beacon_int); | ||
650 | SET_BITS_TO_LE_2BYTE((u8 *)(&joinbss_rpt) + 10, 0, 16, mac->assoc_id); | ||
651 | |||
652 | _rtl92s_firmware_set_h2c_cmd(hw, FW_H2C_JOINBSSRPT, (u8 *)&joinbss_rpt); | ||
653 | } | ||
654 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/fw.h b/drivers/net/wireless/rtlwifi/rtl8192se/fw.h new file mode 100644 index 000000000000..74cc503efe8a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/fw.h | |||
@@ -0,0 +1,375 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_FIRMWARE92S_H__ | ||
30 | #define __REALTEK_FIRMWARE92S_H__ | ||
31 | |||
32 | #define RTL8190_MAX_FIRMWARE_CODE_SIZE 64000 | ||
33 | #define RTL8190_CPU_START_OFFSET 0x80 | ||
34 | /* Firmware Local buffer size. 64k */ | ||
35 | #define MAX_FIRMWARE_CODE_SIZE 0xFF00 | ||
36 | |||
37 | #define RT_8192S_FIRMWARE_HDR_SIZE 80 | ||
38 | #define RT_8192S_FIRMWARE_HDR_EXCLUDE_PRI_SIZE 32 | ||
39 | |||
40 | /* support till 64 bit bus width OS */ | ||
41 | #define MAX_DEV_ADDR_SIZE 8 | ||
42 | #define MAX_FIRMWARE_INFORMATION_SIZE 32 | ||
43 | #define MAX_802_11_HEADER_LENGTH (40 + \ | ||
44 | MAX_FIRMWARE_INFORMATION_SIZE) | ||
45 | #define ENCRYPTION_MAX_OVERHEAD 128 | ||
46 | #define MAX_FRAGMENT_COUNT 8 | ||
47 | #define MAX_TRANSMIT_BUFFER_SIZE (1600 + \ | ||
48 | (MAX_802_11_HEADER_LENGTH + \ | ||
49 | ENCRYPTION_MAX_OVERHEAD) *\ | ||
50 | MAX_FRAGMENT_COUNT) | ||
51 | |||
52 | #define H2C_TX_CMD_HDR_LEN 8 | ||
53 | |||
54 | /* The following DM control code are for Reg0x364, */ | ||
55 | #define FW_DIG_ENABLE_CTL BIT(0) | ||
56 | #define FW_HIGH_PWR_ENABLE_CTL BIT(1) | ||
57 | #define FW_SS_CTL BIT(2) | ||
58 | #define FW_RA_INIT_CTL BIT(3) | ||
59 | #define FW_RA_BG_CTL BIT(4) | ||
60 | #define FW_RA_N_CTL BIT(5) | ||
61 | #define FW_PWR_TRK_CTL BIT(6) | ||
62 | #define FW_IQK_CTL BIT(7) | ||
63 | #define FW_FA_CTL BIT(8) | ||
64 | #define FW_DRIVER_CTRL_DM_CTL BIT(9) | ||
65 | #define FW_PAPE_CTL_BY_SW_HW BIT(10) | ||
66 | #define FW_DISABLE_ALL_DM 0 | ||
67 | #define FW_PWR_TRK_PARAM_CLR 0x0000ffff | ||
68 | #define FW_RA_PARAM_CLR 0xffff0000 | ||
69 | |||
70 | enum desc_packet_type { | ||
71 | DESC_PACKET_TYPE_INIT = 0, | ||
72 | DESC_PACKET_TYPE_NORMAL = 1, | ||
73 | }; | ||
74 | |||
75 | /* 8-bytes alignment required */ | ||
76 | struct fw_priv { | ||
77 | /* --- long word 0 ---- */ | ||
78 | /* 0x12: CE product, 0x92: IT product */ | ||
79 | u8 signature_0; | ||
80 | /* 0x87: CE product, 0x81: IT product */ | ||
81 | u8 signature_1; | ||
82 | /* 0x81: PCI-AP, 01:PCIe, 02: 92S-U, | ||
83 | * 0x82: USB-AP, 0x12: 72S-U, 03:SDIO */ | ||
84 | u8 hci_sel; | ||
85 | /* the same value as reigster value */ | ||
86 | u8 chip_version; | ||
87 | /* customer ID low byte */ | ||
88 | u8 customer_id_0; | ||
89 | /* customer ID high byte */ | ||
90 | u8 customer_id_1; | ||
91 | /* 0x11: 1T1R, 0x12: 1T2R, | ||
92 | * 0x92: 1T2R turbo, 0x22: 2T2R */ | ||
93 | u8 rf_config; | ||
94 | /* 4: 4EP, 6: 6EP, 11: 11EP */ | ||
95 | u8 usb_ep_num; | ||
96 | |||
97 | /* --- long word 1 ---- */ | ||
98 | /* regulatory class bit map 0 */ | ||
99 | u8 regulatory_class_0; | ||
100 | /* regulatory class bit map 1 */ | ||
101 | u8 regulatory_class_1; | ||
102 | /* regulatory class bit map 2 */ | ||
103 | u8 regulatory_class_2; | ||
104 | /* regulatory class bit map 3 */ | ||
105 | u8 regulatory_class_3; | ||
106 | /* 0:SWSI, 1:HWSI, 2:HWPI */ | ||
107 | u8 rfintfs; | ||
108 | u8 def_nettype; | ||
109 | u8 rsvd010; | ||
110 | u8 rsvd011; | ||
111 | |||
112 | /* --- long word 2 ---- */ | ||
113 | /* 0x00: normal, 0x03: MACLBK, 0x01: PHYLBK */ | ||
114 | u8 lbk_mode; | ||
115 | /* 1: for MP use, 0: for normal | ||
116 | * driver (to be discussed) */ | ||
117 | u8 mp_mode; | ||
118 | u8 rsvd020; | ||
119 | u8 rsvd021; | ||
120 | u8 rsvd022; | ||
121 | u8 rsvd023; | ||
122 | u8 rsvd024; | ||
123 | u8 rsvd025; | ||
124 | |||
125 | /* --- long word 3 ---- */ | ||
126 | /* QoS enable */ | ||
127 | u8 qos_en; | ||
128 | /* 40MHz BW enable */ | ||
129 | /* 4181 convert AMSDU to AMPDU, 0: disable */ | ||
130 | u8 bw_40mhz_en; | ||
131 | u8 amsdu2ampdu_en; | ||
132 | /* 11n AMPDU enable */ | ||
133 | u8 ampdu_en; | ||
134 | /* FW offloads, 0: driver handles */ | ||
135 | u8 rate_control_offload; | ||
136 | /* FW offloads, 0: driver handles */ | ||
137 | u8 aggregation_offload; | ||
138 | u8 rsvd030; | ||
139 | u8 rsvd031; | ||
140 | |||
141 | /* --- long word 4 ---- */ | ||
142 | /* 1. FW offloads, 0: driver handles */ | ||
143 | u8 beacon_offload; | ||
144 | /* 2. FW offloads, 0: driver handles */ | ||
145 | u8 mlme_offload; | ||
146 | /* 3. FW offloads, 0: driver handles */ | ||
147 | u8 hwpc_offload; | ||
148 | /* 4. FW offloads, 0: driver handles */ | ||
149 | u8 tcp_checksum_offload; | ||
150 | /* 5. FW offloads, 0: driver handles */ | ||
151 | u8 tcp_offload; | ||
152 | /* 6. FW offloads, 0: driver handles */ | ||
153 | u8 ps_control_offload; | ||
154 | /* 7. FW offloads, 0: driver handles */ | ||
155 | u8 wwlan_offload; | ||
156 | u8 rsvd040; | ||
157 | |||
158 | /* --- long word 5 ---- */ | ||
159 | /* tcp tx packet length low byte */ | ||
160 | u8 tcp_tx_frame_len_L; | ||
161 | /* tcp tx packet length high byte */ | ||
162 | u8 tcp_tx_frame_len_H; | ||
163 | /* tcp rx packet length low byte */ | ||
164 | u8 tcp_rx_frame_len_L; | ||
165 | /* tcp rx packet length high byte */ | ||
166 | u8 tcp_rx_frame_len_H; | ||
167 | u8 rsvd050; | ||
168 | u8 rsvd051; | ||
169 | u8 rsvd052; | ||
170 | u8 rsvd053; | ||
171 | }; | ||
172 | |||
173 | /* 8-byte alinment required */ | ||
174 | struct fw_hdr { | ||
175 | |||
176 | /* --- LONG WORD 0 ---- */ | ||
177 | u16 signature; | ||
178 | /* 0x8000 ~ 0x8FFF for FPGA version, | ||
179 | * 0x0000 ~ 0x7FFF for ASIC version, */ | ||
180 | u16 version; | ||
181 | /* define the size of boot loader */ | ||
182 | u32 dmem_size; | ||
183 | |||
184 | |||
185 | /* --- LONG WORD 1 ---- */ | ||
186 | /* define the size of FW in IMEM */ | ||
187 | u32 img_imem_size; | ||
188 | /* define the size of FW in SRAM */ | ||
189 | u32 img_sram_size; | ||
190 | |||
191 | /* --- LONG WORD 2 ---- */ | ||
192 | /* define the size of DMEM variable */ | ||
193 | u32 fw_priv_size; | ||
194 | u32 rsvd0; | ||
195 | |||
196 | /* --- LONG WORD 3 ---- */ | ||
197 | u32 rsvd1; | ||
198 | u32 rsvd2; | ||
199 | |||
200 | struct fw_priv fwpriv; | ||
201 | |||
202 | } ; | ||
203 | |||
204 | enum fw_status { | ||
205 | FW_STATUS_INIT = 0, | ||
206 | FW_STATUS_LOAD_IMEM = 1, | ||
207 | FW_STATUS_LOAD_EMEM = 2, | ||
208 | FW_STATUS_LOAD_DMEM = 3, | ||
209 | FW_STATUS_READY = 4, | ||
210 | }; | ||
211 | |||
212 | struct rt_firmware { | ||
213 | struct fw_hdr *pfwheader; | ||
214 | enum fw_status fwstatus; | ||
215 | u16 firmwareversion; | ||
216 | u8 fw_imem[RTL8190_MAX_FIRMWARE_CODE_SIZE]; | ||
217 | u8 fw_emem[RTL8190_MAX_FIRMWARE_CODE_SIZE]; | ||
218 | u32 fw_imem_len; | ||
219 | u32 fw_emem_len; | ||
220 | u8 sz_fw_tmpbuffer[164000]; | ||
221 | u32 sz_fw_tmpbufferlen; | ||
222 | u16 cmdpacket_fragthresold; | ||
223 | }; | ||
224 | |||
225 | struct h2c_set_pwrmode_parm { | ||
226 | u8 mode; | ||
227 | u8 flag_low_traffic_en; | ||
228 | u8 flag_lpnav_en; | ||
229 | u8 flag_rf_low_snr_en; | ||
230 | /* 1: dps, 0: 32k */ | ||
231 | u8 flag_dps_en; | ||
232 | u8 bcn_rx_en; | ||
233 | u8 bcn_pass_cnt; | ||
234 | /* beacon TO (ms). ¡§=0¡¨ no limit. */ | ||
235 | u8 bcn_to; | ||
236 | u16 bcn_itv; | ||
237 | /* only for VOIP mode. */ | ||
238 | u8 app_itv; | ||
239 | u8 awake_bcn_itvl; | ||
240 | u8 smart_ps; | ||
241 | /* unit: 100 ms */ | ||
242 | u8 bcn_pass_period; | ||
243 | }; | ||
244 | |||
245 | struct h2c_joinbss_rpt_parm { | ||
246 | u8 opmode; | ||
247 | u8 ps_qos_info; | ||
248 | u8 bssid[6]; | ||
249 | u16 bcnitv; | ||
250 | u16 aid; | ||
251 | } ; | ||
252 | |||
253 | struct h2c_wpa_ptk { | ||
254 | /* EAPOL-Key Key Confirmation Key (KCK) */ | ||
255 | u8 kck[16]; | ||
256 | /* EAPOL-Key Key Encryption Key (KEK) */ | ||
257 | u8 kek[16]; | ||
258 | /* Temporal Key 1 (TK1) */ | ||
259 | u8 tk1[16]; | ||
260 | union { | ||
261 | /* Temporal Key 2 (TK2) */ | ||
262 | u8 tk2[16]; | ||
263 | struct { | ||
264 | u8 tx_mic_key[8]; | ||
265 | u8 rx_mic_key[8]; | ||
266 | } athu; | ||
267 | } u; | ||
268 | }; | ||
269 | |||
270 | struct h2c_wpa_two_way_parm { | ||
271 | /* algorithm TKIP or AES */ | ||
272 | u8 pairwise_en_alg; | ||
273 | u8 group_en_alg; | ||
274 | struct h2c_wpa_ptk wpa_ptk_value; | ||
275 | } ; | ||
276 | |||
277 | enum h2c_cmd { | ||
278 | FW_H2C_SETPWRMODE = 0, | ||
279 | FW_H2C_JOINBSSRPT = 1, | ||
280 | FW_H2C_WOWLAN_UPDATE_GTK = 2, | ||
281 | FW_H2C_WOWLAN_UPDATE_IV = 3, | ||
282 | FW_H2C_WOWLAN_OFFLOAD = 4, | ||
283 | }; | ||
284 | |||
285 | enum fw_h2c_cmd { | ||
286 | H2C_READ_MACREG_CMD, /*0*/ | ||
287 | H2C_WRITE_MACREG_CMD, | ||
288 | H2C_READBB_CMD, | ||
289 | H2C_WRITEBB_CMD, | ||
290 | H2C_READRF_CMD, | ||
291 | H2C_WRITERF_CMD, /*5*/ | ||
292 | H2C_READ_EEPROM_CMD, | ||
293 | H2C_WRITE_EEPROM_CMD, | ||
294 | H2C_READ_EFUSE_CMD, | ||
295 | H2C_WRITE_EFUSE_CMD, | ||
296 | H2C_READ_CAM_CMD, /*10*/ | ||
297 | H2C_WRITE_CAM_CMD, | ||
298 | H2C_SETBCNITV_CMD, | ||
299 | H2C_SETMBIDCFG_CMD, | ||
300 | H2C_JOINBSS_CMD, | ||
301 | H2C_DISCONNECT_CMD, /*15*/ | ||
302 | H2C_CREATEBSS_CMD, | ||
303 | H2C_SETOPMode_CMD, | ||
304 | H2C_SITESURVEY_CMD, | ||
305 | H2C_SETAUTH_CMD, | ||
306 | H2C_SETKEY_CMD, /*20*/ | ||
307 | H2C_SETSTAKEY_CMD, | ||
308 | H2C_SETASSOCSTA_CMD, | ||
309 | H2C_DELASSOCSTA_CMD, | ||
310 | H2C_SETSTAPWRSTATE_CMD, | ||
311 | H2C_SETBASICRATE_CMD, /*25*/ | ||
312 | H2C_GETBASICRATE_CMD, | ||
313 | H2C_SETDATARATE_CMD, | ||
314 | H2C_GETDATARATE_CMD, | ||
315 | H2C_SETPHYINFO_CMD, | ||
316 | H2C_GETPHYINFO_CMD, /*30*/ | ||
317 | H2C_SETPHY_CMD, | ||
318 | H2C_GETPHY_CMD, | ||
319 | H2C_READRSSI_CMD, | ||
320 | H2C_READGAIN_CMD, | ||
321 | H2C_SETATIM_CMD, /*35*/ | ||
322 | H2C_SETPWRMODE_CMD, | ||
323 | H2C_JOINBSSRPT_CMD, | ||
324 | H2C_SETRATABLE_CMD, | ||
325 | H2C_GETRATABLE_CMD, | ||
326 | H2C_GETCCXREPORT_CMD, /*40*/ | ||
327 | H2C_GETDTMREPORT_CMD, | ||
328 | H2C_GETTXRATESTATICS_CMD, | ||
329 | H2C_SETUSBSUSPEND_CMD, | ||
330 | H2C_SETH2CLBK_CMD, | ||
331 | H2C_TMP1, /*45*/ | ||
332 | H2C_WOWLAN_UPDATE_GTK_CMD, | ||
333 | H2C_WOWLAN_FW_OFFLOAD, | ||
334 | H2C_TMP2, | ||
335 | H2C_TMP3, | ||
336 | H2C_WOWLAN_UPDATE_IV_CMD, /*50*/ | ||
337 | H2C_TMP4, | ||
338 | MAX_H2CCMD /*52*/ | ||
339 | }; | ||
340 | |||
341 | /* The following macros are used for FW | ||
342 | * CMD map and parameter updated. */ | ||
343 | #define FW_CMD_IO_CLR(rtlpriv, _Bit) \ | ||
344 | do { \ | ||
345 | udelay(1000); \ | ||
346 | rtlpriv->rtlhal.fwcmd_iomap &= (~_Bit); \ | ||
347 | } while (0); | ||
348 | |||
349 | #define FW_CMD_IO_UPDATE(rtlpriv, _val) \ | ||
350 | rtlpriv->rtlhal.fwcmd_iomap = _val; | ||
351 | |||
352 | #define FW_CMD_IO_SET(rtlpriv, _val) \ | ||
353 | do { \ | ||
354 | rtl_write_word(rtlpriv, LBUS_MON_ADDR, (u16)_val); \ | ||
355 | FW_CMD_IO_UPDATE(rtlpriv, _val); \ | ||
356 | } while (0); | ||
357 | |||
358 | #define FW_CMD_PARA_SET(rtlpriv, _val) \ | ||
359 | do { \ | ||
360 | rtl_write_dword(rtlpriv, LBUS_ADDR_MASK, _val); \ | ||
361 | rtlpriv->rtlhal.fwcmd_ioparam = _val; \ | ||
362 | } while (0); | ||
363 | |||
364 | #define FW_CMD_IO_QUERY(rtlpriv) \ | ||
365 | (u16)(rtlpriv->rtlhal.fwcmd_iomap) | ||
366 | #define FW_CMD_IO_PARA_QUERY(rtlpriv) \ | ||
367 | ((u32)(rtlpriv->rtlhal.fwcmd_ioparam)) | ||
368 | |||
369 | int rtl92s_download_fw(struct ieee80211_hw *hw); | ||
370 | void rtl92s_set_fw_pwrmode_cmd(struct ieee80211_hw *hw, u8 mode); | ||
371 | void rtl92s_set_fw_joinbss_report_cmd(struct ieee80211_hw *hw, | ||
372 | u8 mstatus, u8 ps_qosinfo); | ||
373 | |||
374 | #endif | ||
375 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/hw.c b/drivers/net/wireless/rtlwifi/rtl8192se/hw.c new file mode 100644 index 000000000000..2e9005d0454b --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/hw.c | |||
@@ -0,0 +1,2512 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../efuse.h" | ||
32 | #include "../base.h" | ||
33 | #include "../regd.h" | ||
34 | #include "../cam.h" | ||
35 | #include "../ps.h" | ||
36 | #include "../pci.h" | ||
37 | #include "reg.h" | ||
38 | #include "def.h" | ||
39 | #include "phy.h" | ||
40 | #include "dm.h" | ||
41 | #include "fw.h" | ||
42 | #include "led.h" | ||
43 | #include "hw.h" | ||
44 | |||
45 | void rtl92se_get_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
46 | { | ||
47 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
48 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
49 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
50 | |||
51 | switch (variable) { | ||
52 | case HW_VAR_RCR: { | ||
53 | *((u32 *) (val)) = rtlpci->receive_config; | ||
54 | break; | ||
55 | } | ||
56 | case HW_VAR_RF_STATE: { | ||
57 | *((enum rf_pwrstate *)(val)) = ppsc->rfpwr_state; | ||
58 | break; | ||
59 | } | ||
60 | case HW_VAR_FW_PSMODE_STATUS: { | ||
61 | *((bool *) (val)) = ppsc->fw_current_inpsmode; | ||
62 | break; | ||
63 | } | ||
64 | case HW_VAR_CORRECT_TSF: { | ||
65 | u64 tsf; | ||
66 | u32 *ptsf_low = (u32 *)&tsf; | ||
67 | u32 *ptsf_high = ((u32 *)&tsf) + 1; | ||
68 | |||
69 | *ptsf_high = rtl_read_dword(rtlpriv, (TSFR + 4)); | ||
70 | *ptsf_low = rtl_read_dword(rtlpriv, TSFR); | ||
71 | |||
72 | *((u64 *) (val)) = tsf; | ||
73 | |||
74 | break; | ||
75 | } | ||
76 | case HW_VAR_MRC: { | ||
77 | *((bool *)(val)) = rtlpriv->dm.current_mrc_switch; | ||
78 | break; | ||
79 | } | ||
80 | default: { | ||
81 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
82 | ("switch case not process\n")); | ||
83 | break; | ||
84 | } | ||
85 | } | ||
86 | } | ||
87 | |||
88 | void rtl92se_set_hw_reg(struct ieee80211_hw *hw, u8 variable, u8 *val) | ||
89 | { | ||
90 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
91 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
92 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
93 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
94 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
95 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
96 | |||
97 | switch (variable) { | ||
98 | case HW_VAR_ETHER_ADDR:{ | ||
99 | rtl_write_dword(rtlpriv, IDR0, ((u32 *)(val))[0]); | ||
100 | rtl_write_word(rtlpriv, IDR4, ((u16 *)(val + 4))[0]); | ||
101 | break; | ||
102 | } | ||
103 | case HW_VAR_BASIC_RATE:{ | ||
104 | u16 rate_cfg = ((u16 *) val)[0]; | ||
105 | u8 rate_index = 0; | ||
106 | |||
107 | if (rtlhal->version == VERSION_8192S_ACUT) | ||
108 | rate_cfg = rate_cfg & 0x150; | ||
109 | else | ||
110 | rate_cfg = rate_cfg & 0x15f; | ||
111 | |||
112 | rate_cfg |= 0x01; | ||
113 | |||
114 | rtl_write_byte(rtlpriv, RRSR, rate_cfg & 0xff); | ||
115 | rtl_write_byte(rtlpriv, RRSR + 1, | ||
116 | (rate_cfg >> 8) & 0xff); | ||
117 | |||
118 | while (rate_cfg > 0x1) { | ||
119 | rate_cfg = (rate_cfg >> 1); | ||
120 | rate_index++; | ||
121 | } | ||
122 | rtl_write_byte(rtlpriv, INIRTSMCS_SEL, rate_index); | ||
123 | |||
124 | break; | ||
125 | } | ||
126 | case HW_VAR_BSSID:{ | ||
127 | rtl_write_dword(rtlpriv, BSSIDR, ((u32 *)(val))[0]); | ||
128 | rtl_write_word(rtlpriv, BSSIDR + 4, | ||
129 | ((u16 *)(val + 4))[0]); | ||
130 | break; | ||
131 | } | ||
132 | case HW_VAR_SIFS:{ | ||
133 | rtl_write_byte(rtlpriv, SIFS_OFDM, val[0]); | ||
134 | rtl_write_byte(rtlpriv, SIFS_OFDM + 1, val[1]); | ||
135 | break; | ||
136 | } | ||
137 | case HW_VAR_SLOT_TIME:{ | ||
138 | u8 e_aci; | ||
139 | |||
140 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
141 | ("HW_VAR_SLOT_TIME %x\n", val[0])); | ||
142 | |||
143 | rtl_write_byte(rtlpriv, SLOT_TIME, val[0]); | ||
144 | |||
145 | for (e_aci = 0; e_aci < AC_MAX; e_aci++) { | ||
146 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
147 | HW_VAR_AC_PARAM, | ||
148 | (u8 *)(&e_aci)); | ||
149 | } | ||
150 | break; | ||
151 | } | ||
152 | case HW_VAR_ACK_PREAMBLE:{ | ||
153 | u8 reg_tmp; | ||
154 | u8 short_preamble = (bool) (*(u8 *) val); | ||
155 | reg_tmp = (mac->cur_40_prime_sc) << 5; | ||
156 | if (short_preamble) | ||
157 | reg_tmp |= 0x80; | ||
158 | |||
159 | rtl_write_byte(rtlpriv, RRSR + 2, reg_tmp); | ||
160 | break; | ||
161 | } | ||
162 | case HW_VAR_AMPDU_MIN_SPACE:{ | ||
163 | u8 min_spacing_to_set; | ||
164 | u8 sec_min_space; | ||
165 | |||
166 | min_spacing_to_set = *((u8 *)val); | ||
167 | if (min_spacing_to_set <= 7) { | ||
168 | if (rtlpriv->sec.pairwise_enc_algorithm == | ||
169 | NO_ENCRYPTION) | ||
170 | sec_min_space = 0; | ||
171 | else | ||
172 | sec_min_space = 1; | ||
173 | |||
174 | if (min_spacing_to_set < sec_min_space) | ||
175 | min_spacing_to_set = sec_min_space; | ||
176 | if (min_spacing_to_set > 5) | ||
177 | min_spacing_to_set = 5; | ||
178 | |||
179 | mac->min_space_cfg = | ||
180 | ((mac->min_space_cfg & 0xf8) | | ||
181 | min_spacing_to_set); | ||
182 | |||
183 | *val = min_spacing_to_set; | ||
184 | |||
185 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
186 | ("Set HW_VAR_AMPDU_MIN_SPACE: %#x\n", | ||
187 | mac->min_space_cfg)); | ||
188 | |||
189 | rtl_write_byte(rtlpriv, AMPDU_MIN_SPACE, | ||
190 | mac->min_space_cfg); | ||
191 | } | ||
192 | break; | ||
193 | } | ||
194 | case HW_VAR_SHORTGI_DENSITY:{ | ||
195 | u8 density_to_set; | ||
196 | |||
197 | density_to_set = *((u8 *) val); | ||
198 | mac->min_space_cfg = rtlpriv->rtlhal.minspace_cfg; | ||
199 | mac->min_space_cfg |= (density_to_set << 3); | ||
200 | |||
201 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
202 | ("Set HW_VAR_SHORTGI_DENSITY: %#x\n", | ||
203 | mac->min_space_cfg)); | ||
204 | |||
205 | rtl_write_byte(rtlpriv, AMPDU_MIN_SPACE, | ||
206 | mac->min_space_cfg); | ||
207 | |||
208 | break; | ||
209 | } | ||
210 | case HW_VAR_AMPDU_FACTOR:{ | ||
211 | u8 factor_toset; | ||
212 | u8 regtoset; | ||
213 | u8 factorlevel[18] = { | ||
214 | 2, 4, 4, 7, 7, 13, 13, | ||
215 | 13, 2, 7, 7, 13, 13, | ||
216 | 15, 15, 15, 15, 0}; | ||
217 | u8 index = 0; | ||
218 | |||
219 | factor_toset = *((u8 *) val); | ||
220 | if (factor_toset <= 3) { | ||
221 | factor_toset = (1 << (factor_toset + 2)); | ||
222 | if (factor_toset > 0xf) | ||
223 | factor_toset = 0xf; | ||
224 | |||
225 | for (index = 0; index < 17; index++) { | ||
226 | if (factorlevel[index] > factor_toset) | ||
227 | factorlevel[index] = | ||
228 | factor_toset; | ||
229 | } | ||
230 | |||
231 | for (index = 0; index < 8; index++) { | ||
232 | regtoset = ((factorlevel[index * 2]) | | ||
233 | (factorlevel[index * | ||
234 | 2 + 1] << 4)); | ||
235 | rtl_write_byte(rtlpriv, | ||
236 | AGGLEN_LMT_L + index, | ||
237 | regtoset); | ||
238 | } | ||
239 | |||
240 | regtoset = ((factorlevel[16]) | | ||
241 | (factorlevel[17] << 4)); | ||
242 | rtl_write_byte(rtlpriv, AGGLEN_LMT_H, regtoset); | ||
243 | |||
244 | RT_TRACE(rtlpriv, COMP_MLME, DBG_LOUD, | ||
245 | ("Set HW_VAR_AMPDU_FACTOR: %#x\n", | ||
246 | factor_toset)); | ||
247 | } | ||
248 | break; | ||
249 | } | ||
250 | case HW_VAR_AC_PARAM:{ | ||
251 | u8 e_aci = *((u8 *) val); | ||
252 | rtl92s_dm_init_edca_turbo(hw); | ||
253 | |||
254 | if (rtlpci->acm_method != eAcmWay2_SW) | ||
255 | rtlpriv->cfg->ops->set_hw_reg(hw, | ||
256 | HW_VAR_ACM_CTRL, | ||
257 | (u8 *)(&e_aci)); | ||
258 | break; | ||
259 | } | ||
260 | case HW_VAR_ACM_CTRL:{ | ||
261 | u8 e_aci = *((u8 *) val); | ||
262 | union aci_aifsn *p_aci_aifsn = (union aci_aifsn *)(&( | ||
263 | mac->ac[0].aifs)); | ||
264 | u8 acm = p_aci_aifsn->f.acm; | ||
265 | u8 acm_ctrl = rtl_read_byte(rtlpriv, AcmHwCtrl); | ||
266 | |||
267 | acm_ctrl = acm_ctrl | ((rtlpci->acm_method == 2) ? | ||
268 | 0x0 : 0x1); | ||
269 | |||
270 | if (acm) { | ||
271 | switch (e_aci) { | ||
272 | case AC0_BE: | ||
273 | acm_ctrl |= AcmHw_BeqEn; | ||
274 | break; | ||
275 | case AC2_VI: | ||
276 | acm_ctrl |= AcmHw_ViqEn; | ||
277 | break; | ||
278 | case AC3_VO: | ||
279 | acm_ctrl |= AcmHw_VoqEn; | ||
280 | break; | ||
281 | default: | ||
282 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
283 | ("HW_VAR_ACM_CTRL acm set " | ||
284 | "failed: eACI is %d\n", acm)); | ||
285 | break; | ||
286 | } | ||
287 | } else { | ||
288 | switch (e_aci) { | ||
289 | case AC0_BE: | ||
290 | acm_ctrl &= (~AcmHw_BeqEn); | ||
291 | break; | ||
292 | case AC2_VI: | ||
293 | acm_ctrl &= (~AcmHw_ViqEn); | ||
294 | break; | ||
295 | case AC3_VO: | ||
296 | acm_ctrl &= (~AcmHw_BeqEn); | ||
297 | break; | ||
298 | default: | ||
299 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
300 | ("switch case not process\n")); | ||
301 | break; | ||
302 | } | ||
303 | } | ||
304 | |||
305 | RT_TRACE(rtlpriv, COMP_QOS, DBG_TRACE, | ||
306 | ("HW_VAR_ACM_CTRL Write 0x%X\n", acm_ctrl)); | ||
307 | rtl_write_byte(rtlpriv, AcmHwCtrl, acm_ctrl); | ||
308 | break; | ||
309 | } | ||
310 | case HW_VAR_RCR:{ | ||
311 | rtl_write_dword(rtlpriv, RCR, ((u32 *) (val))[0]); | ||
312 | rtlpci->receive_config = ((u32 *) (val))[0]; | ||
313 | break; | ||
314 | } | ||
315 | case HW_VAR_RETRY_LIMIT:{ | ||
316 | u8 retry_limit = ((u8 *) (val))[0]; | ||
317 | |||
318 | rtl_write_word(rtlpriv, RETRY_LIMIT, | ||
319 | retry_limit << RETRY_LIMIT_SHORT_SHIFT | | ||
320 | retry_limit << RETRY_LIMIT_LONG_SHIFT); | ||
321 | break; | ||
322 | } | ||
323 | case HW_VAR_DUAL_TSF_RST: { | ||
324 | break; | ||
325 | } | ||
326 | case HW_VAR_EFUSE_BYTES: { | ||
327 | rtlefuse->efuse_usedbytes = *((u16 *) val); | ||
328 | break; | ||
329 | } | ||
330 | case HW_VAR_EFUSE_USAGE: { | ||
331 | rtlefuse->efuse_usedpercentage = *((u8 *) val); | ||
332 | break; | ||
333 | } | ||
334 | case HW_VAR_IO_CMD: { | ||
335 | break; | ||
336 | } | ||
337 | case HW_VAR_WPA_CONFIG: { | ||
338 | rtl_write_byte(rtlpriv, REG_SECR, *((u8 *) val)); | ||
339 | break; | ||
340 | } | ||
341 | case HW_VAR_SET_RPWM:{ | ||
342 | break; | ||
343 | } | ||
344 | case HW_VAR_H2C_FW_PWRMODE:{ | ||
345 | break; | ||
346 | } | ||
347 | case HW_VAR_FW_PSMODE_STATUS: { | ||
348 | ppsc->fw_current_inpsmode = *((bool *) val); | ||
349 | break; | ||
350 | } | ||
351 | case HW_VAR_H2C_FW_JOINBSSRPT:{ | ||
352 | break; | ||
353 | } | ||
354 | case HW_VAR_AID:{ | ||
355 | break; | ||
356 | } | ||
357 | case HW_VAR_CORRECT_TSF:{ | ||
358 | break; | ||
359 | } | ||
360 | case HW_VAR_MRC: { | ||
361 | bool bmrc_toset = *((bool *)val); | ||
362 | u8 u1bdata = 0; | ||
363 | |||
364 | if (bmrc_toset) { | ||
365 | rtl_set_bbreg(hw, ROFDM0_TRXPATHENABLE, | ||
366 | MASKBYTE0, 0x33); | ||
367 | u1bdata = (u8)rtl_get_bbreg(hw, | ||
368 | ROFDM1_TRXPATHENABLE, | ||
369 | MASKBYTE0); | ||
370 | rtl_set_bbreg(hw, ROFDM1_TRXPATHENABLE, | ||
371 | MASKBYTE0, | ||
372 | ((u1bdata & 0xf0) | 0x03)); | ||
373 | u1bdata = (u8)rtl_get_bbreg(hw, | ||
374 | ROFDM0_TRXPATHENABLE, | ||
375 | MASKBYTE1); | ||
376 | rtl_set_bbreg(hw, ROFDM0_TRXPATHENABLE, | ||
377 | MASKBYTE1, | ||
378 | (u1bdata | 0x04)); | ||
379 | |||
380 | /* Update current settings. */ | ||
381 | rtlpriv->dm.current_mrc_switch = bmrc_toset; | ||
382 | } else { | ||
383 | rtl_set_bbreg(hw, ROFDM0_TRXPATHENABLE, | ||
384 | MASKBYTE0, 0x13); | ||
385 | u1bdata = (u8)rtl_get_bbreg(hw, | ||
386 | ROFDM1_TRXPATHENABLE, | ||
387 | MASKBYTE0); | ||
388 | rtl_set_bbreg(hw, ROFDM1_TRXPATHENABLE, | ||
389 | MASKBYTE0, | ||
390 | ((u1bdata & 0xf0) | 0x01)); | ||
391 | u1bdata = (u8)rtl_get_bbreg(hw, | ||
392 | ROFDM0_TRXPATHENABLE, | ||
393 | MASKBYTE1); | ||
394 | rtl_set_bbreg(hw, ROFDM0_TRXPATHENABLE, | ||
395 | MASKBYTE1, (u1bdata & 0xfb)); | ||
396 | |||
397 | /* Update current settings. */ | ||
398 | rtlpriv->dm.current_mrc_switch = bmrc_toset; | ||
399 | } | ||
400 | |||
401 | break; | ||
402 | } | ||
403 | default: | ||
404 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
405 | ("switch case not process\n")); | ||
406 | break; | ||
407 | } | ||
408 | |||
409 | } | ||
410 | |||
411 | void rtl92se_enable_hw_security_config(struct ieee80211_hw *hw) | ||
412 | { | ||
413 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
414 | u8 sec_reg_value = 0x0; | ||
415 | |||
416 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("PairwiseEncAlgorithm = %d " | ||
417 | "GroupEncAlgorithm = %d\n", | ||
418 | rtlpriv->sec.pairwise_enc_algorithm, | ||
419 | rtlpriv->sec.group_enc_algorithm)); | ||
420 | |||
421 | if (rtlpriv->cfg->mod_params->sw_crypto || rtlpriv->sec.use_sw_sec) { | ||
422 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
423 | ("not open hw encryption\n")); | ||
424 | return; | ||
425 | } | ||
426 | |||
427 | sec_reg_value = SCR_TXENCENABLE | SCR_RXENCENABLE; | ||
428 | |||
429 | if (rtlpriv->sec.use_defaultkey) { | ||
430 | sec_reg_value |= SCR_TXUSEDK; | ||
431 | sec_reg_value |= SCR_RXUSEDK; | ||
432 | } | ||
433 | |||
434 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, ("The SECR-value %x\n", | ||
435 | sec_reg_value)); | ||
436 | |||
437 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_WPA_CONFIG, &sec_reg_value); | ||
438 | |||
439 | } | ||
440 | |||
441 | static u8 _rtl92ce_halset_sysclk(struct ieee80211_hw *hw, u8 data) | ||
442 | { | ||
443 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
444 | u8 waitcount = 100; | ||
445 | bool bresult = false; | ||
446 | u8 tmpvalue; | ||
447 | |||
448 | rtl_write_byte(rtlpriv, SYS_CLKR + 1, data); | ||
449 | |||
450 | /* Wait the MAC synchronized. */ | ||
451 | udelay(400); | ||
452 | |||
453 | /* Check if it is set ready. */ | ||
454 | tmpvalue = rtl_read_byte(rtlpriv, SYS_CLKR + 1); | ||
455 | bresult = ((tmpvalue & BIT(7)) == (data & BIT(7))); | ||
456 | |||
457 | if ((data & (BIT(6) | BIT(7))) == false) { | ||
458 | waitcount = 100; | ||
459 | tmpvalue = 0; | ||
460 | |||
461 | while (1) { | ||
462 | waitcount--; | ||
463 | |||
464 | tmpvalue = rtl_read_byte(rtlpriv, SYS_CLKR + 1); | ||
465 | if ((tmpvalue & BIT(6))) | ||
466 | break; | ||
467 | |||
468 | printk(KERN_ERR "wait for BIT(6) return value %x\n", | ||
469 | tmpvalue); | ||
470 | if (waitcount == 0) | ||
471 | break; | ||
472 | |||
473 | udelay(10); | ||
474 | } | ||
475 | |||
476 | if (waitcount == 0) | ||
477 | bresult = false; | ||
478 | else | ||
479 | bresult = true; | ||
480 | } | ||
481 | |||
482 | return bresult; | ||
483 | } | ||
484 | |||
485 | void rtl8192se_gpiobit3_cfg_inputmode(struct ieee80211_hw *hw) | ||
486 | { | ||
487 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
488 | u8 u1tmp; | ||
489 | |||
490 | /* The following config GPIO function */ | ||
491 | rtl_write_byte(rtlpriv, MAC_PINMUX_CFG, (GPIOMUX_EN | GPIOSEL_GPIO)); | ||
492 | u1tmp = rtl_read_byte(rtlpriv, GPIO_IO_SEL); | ||
493 | |||
494 | /* config GPIO3 to input */ | ||
495 | u1tmp &= HAL_8192S_HW_GPIO_OFF_MASK; | ||
496 | rtl_write_byte(rtlpriv, GPIO_IO_SEL, u1tmp); | ||
497 | |||
498 | } | ||
499 | |||
500 | static u8 _rtl92se_rf_onoff_detect(struct ieee80211_hw *hw) | ||
501 | { | ||
502 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
503 | u8 u1tmp; | ||
504 | u8 retval = ERFON; | ||
505 | |||
506 | /* The following config GPIO function */ | ||
507 | rtl_write_byte(rtlpriv, MAC_PINMUX_CFG, (GPIOMUX_EN | GPIOSEL_GPIO)); | ||
508 | u1tmp = rtl_read_byte(rtlpriv, GPIO_IO_SEL); | ||
509 | |||
510 | /* config GPIO3 to input */ | ||
511 | u1tmp &= HAL_8192S_HW_GPIO_OFF_MASK; | ||
512 | rtl_write_byte(rtlpriv, GPIO_IO_SEL, u1tmp); | ||
513 | |||
514 | /* On some of the platform, driver cannot read correct | ||
515 | * value without delay between Write_GPIO_SEL and Read_GPIO_IN */ | ||
516 | mdelay(10); | ||
517 | |||
518 | /* check GPIO3 */ | ||
519 | u1tmp = rtl_read_byte(rtlpriv, GPIO_IN); | ||
520 | retval = (u1tmp & HAL_8192S_HW_GPIO_OFF_BIT) ? ERFON : ERFOFF; | ||
521 | |||
522 | return retval; | ||
523 | } | ||
524 | |||
525 | static void _rtl92se_macconfig_before_fwdownload(struct ieee80211_hw *hw) | ||
526 | { | ||
527 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
528 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
529 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
530 | |||
531 | u8 i; | ||
532 | u8 tmpu1b; | ||
533 | u16 tmpu2b; | ||
534 | u8 pollingcnt = 20; | ||
535 | |||
536 | if (rtlpci->first_init) { | ||
537 | /* Reset PCIE Digital */ | ||
538 | tmpu1b = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
539 | tmpu1b &= 0xFE; | ||
540 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, tmpu1b); | ||
541 | udelay(1); | ||
542 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, tmpu1b | BIT(0)); | ||
543 | } | ||
544 | |||
545 | /* Switch to SW IO control */ | ||
546 | tmpu1b = rtl_read_byte(rtlpriv, (SYS_CLKR + 1)); | ||
547 | if (tmpu1b & BIT(7)) { | ||
548 | tmpu1b &= ~(BIT(6) | BIT(7)); | ||
549 | |||
550 | /* Set failed, return to prevent hang. */ | ||
551 | if (!_rtl92ce_halset_sysclk(hw, tmpu1b)) | ||
552 | return; | ||
553 | } | ||
554 | |||
555 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, 0x0); | ||
556 | udelay(50); | ||
557 | rtl_write_byte(rtlpriv, LDOA15_CTRL, 0x34); | ||
558 | udelay(50); | ||
559 | |||
560 | /* Clear FW RPWM for FW control LPS.*/ | ||
561 | rtl_write_byte(rtlpriv, RPWM, 0x0); | ||
562 | |||
563 | /* Reset MAC-IO and CPU and Core Digital BIT(10)/11/15 */ | ||
564 | tmpu1b = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
565 | tmpu1b &= 0x73; | ||
566 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, tmpu1b); | ||
567 | /* wait for BIT 10/11/15 to pull high automatically!! */ | ||
568 | mdelay(1); | ||
569 | |||
570 | rtl_write_byte(rtlpriv, CMDR, 0); | ||
571 | rtl_write_byte(rtlpriv, TCR, 0); | ||
572 | |||
573 | /* Data sheet not define 0x562!!! Copy from WMAC!!!!! */ | ||
574 | tmpu1b = rtl_read_byte(rtlpriv, 0x562); | ||
575 | tmpu1b |= 0x08; | ||
576 | rtl_write_byte(rtlpriv, 0x562, tmpu1b); | ||
577 | tmpu1b &= ~(BIT(3)); | ||
578 | rtl_write_byte(rtlpriv, 0x562, tmpu1b); | ||
579 | |||
580 | /* Enable AFE clock source */ | ||
581 | tmpu1b = rtl_read_byte(rtlpriv, AFE_XTAL_CTRL); | ||
582 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL, (tmpu1b | 0x01)); | ||
583 | /* Delay 1.5ms */ | ||
584 | mdelay(2); | ||
585 | tmpu1b = rtl_read_byte(rtlpriv, AFE_XTAL_CTRL + 1); | ||
586 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL + 1, (tmpu1b & 0xfb)); | ||
587 | |||
588 | /* Enable AFE Macro Block's Bandgap */ | ||
589 | tmpu1b = rtl_read_byte(rtlpriv, AFE_MISC); | ||
590 | rtl_write_byte(rtlpriv, AFE_MISC, (tmpu1b | BIT(0))); | ||
591 | mdelay(1); | ||
592 | |||
593 | /* Enable AFE Mbias */ | ||
594 | tmpu1b = rtl_read_byte(rtlpriv, AFE_MISC); | ||
595 | rtl_write_byte(rtlpriv, AFE_MISC, (tmpu1b | 0x02)); | ||
596 | mdelay(1); | ||
597 | |||
598 | /* Enable LDOA15 block */ | ||
599 | tmpu1b = rtl_read_byte(rtlpriv, LDOA15_CTRL); | ||
600 | rtl_write_byte(rtlpriv, LDOA15_CTRL, (tmpu1b | BIT(0))); | ||
601 | |||
602 | /* Set Digital Vdd to Retention isolation Path. */ | ||
603 | tmpu2b = rtl_read_word(rtlpriv, REG_SYS_ISO_CTRL); | ||
604 | rtl_write_word(rtlpriv, REG_SYS_ISO_CTRL, (tmpu2b | BIT(11))); | ||
605 | |||
606 | /* For warm reboot NIC disappera bug. */ | ||
607 | tmpu2b = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
608 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (tmpu2b | BIT(13))); | ||
609 | |||
610 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL + 1, 0x68); | ||
611 | |||
612 | /* Enable AFE PLL Macro Block */ | ||
613 | /* We need to delay 100u before enabling PLL. */ | ||
614 | udelay(200); | ||
615 | tmpu1b = rtl_read_byte(rtlpriv, AFE_PLL_CTRL); | ||
616 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, (tmpu1b | BIT(0) | BIT(4))); | ||
617 | |||
618 | /* for divider reset */ | ||
619 | udelay(100); | ||
620 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, (tmpu1b | BIT(0) | | ||
621 | BIT(4) | BIT(6))); | ||
622 | udelay(10); | ||
623 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, (tmpu1b | BIT(0) | BIT(4))); | ||
624 | udelay(10); | ||
625 | |||
626 | /* Enable MAC 80MHZ clock */ | ||
627 | tmpu1b = rtl_read_byte(rtlpriv, AFE_PLL_CTRL + 1); | ||
628 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL + 1, (tmpu1b | BIT(0))); | ||
629 | mdelay(1); | ||
630 | |||
631 | /* Release isolation AFE PLL & MD */ | ||
632 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL, 0xA6); | ||
633 | |||
634 | /* Enable MAC clock */ | ||
635 | tmpu2b = rtl_read_word(rtlpriv, SYS_CLKR); | ||
636 | rtl_write_word(rtlpriv, SYS_CLKR, (tmpu2b | BIT(12) | BIT(11))); | ||
637 | |||
638 | /* Enable Core digital and enable IOREG R/W */ | ||
639 | tmpu2b = rtl_read_word(rtlpriv, REG_SYS_FUNC_EN); | ||
640 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (tmpu2b | BIT(11))); | ||
641 | |||
642 | tmpu1b = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN + 1); | ||
643 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, tmpu1b & ~(BIT(7))); | ||
644 | |||
645 | /* enable REG_EN */ | ||
646 | rtl_write_word(rtlpriv, REG_SYS_FUNC_EN, (tmpu2b | BIT(11) | BIT(15))); | ||
647 | |||
648 | /* Switch the control path. */ | ||
649 | tmpu2b = rtl_read_word(rtlpriv, SYS_CLKR); | ||
650 | rtl_write_word(rtlpriv, SYS_CLKR, (tmpu2b & (~BIT(2)))); | ||
651 | |||
652 | tmpu1b = rtl_read_byte(rtlpriv, (SYS_CLKR + 1)); | ||
653 | tmpu1b = ((tmpu1b | BIT(7)) & (~BIT(6))); | ||
654 | if (!_rtl92ce_halset_sysclk(hw, tmpu1b)) | ||
655 | return; /* Set failed, return to prevent hang. */ | ||
656 | |||
657 | rtl_write_word(rtlpriv, CMDR, 0x07FC); | ||
658 | |||
659 | /* MH We must enable the section of code to prevent load IMEM fail. */ | ||
660 | /* Load MAC register from WMAc temporarily We simulate macreg. */ | ||
661 | /* txt HW will provide MAC txt later */ | ||
662 | rtl_write_byte(rtlpriv, 0x6, 0x30); | ||
663 | rtl_write_byte(rtlpriv, 0x49, 0xf0); | ||
664 | |||
665 | rtl_write_byte(rtlpriv, 0x4b, 0x81); | ||
666 | |||
667 | rtl_write_byte(rtlpriv, 0xb5, 0x21); | ||
668 | |||
669 | rtl_write_byte(rtlpriv, 0xdc, 0xff); | ||
670 | rtl_write_byte(rtlpriv, 0xdd, 0xff); | ||
671 | rtl_write_byte(rtlpriv, 0xde, 0xff); | ||
672 | rtl_write_byte(rtlpriv, 0xdf, 0xff); | ||
673 | |||
674 | rtl_write_byte(rtlpriv, 0x11a, 0x00); | ||
675 | rtl_write_byte(rtlpriv, 0x11b, 0x00); | ||
676 | |||
677 | for (i = 0; i < 32; i++) | ||
678 | rtl_write_byte(rtlpriv, INIMCS_SEL + i, 0x1b); | ||
679 | |||
680 | rtl_write_byte(rtlpriv, 0x236, 0xff); | ||
681 | |||
682 | rtl_write_byte(rtlpriv, 0x503, 0x22); | ||
683 | |||
684 | if (ppsc->support_aspm && !ppsc->support_backdoor) | ||
685 | rtl_write_byte(rtlpriv, 0x560, 0x40); | ||
686 | else | ||
687 | rtl_write_byte(rtlpriv, 0x560, 0x00); | ||
688 | |||
689 | rtl_write_byte(rtlpriv, DBG_PORT, 0x91); | ||
690 | |||
691 | /* Set RX Desc Address */ | ||
692 | rtl_write_dword(rtlpriv, RDQDA, rtlpci->rx_ring[RX_MPDU_QUEUE].dma); | ||
693 | rtl_write_dword(rtlpriv, RCDA, rtlpci->rx_ring[RX_CMD_QUEUE].dma); | ||
694 | |||
695 | /* Set TX Desc Address */ | ||
696 | rtl_write_dword(rtlpriv, TBKDA, rtlpci->tx_ring[BK_QUEUE].dma); | ||
697 | rtl_write_dword(rtlpriv, TBEDA, rtlpci->tx_ring[BE_QUEUE].dma); | ||
698 | rtl_write_dword(rtlpriv, TVIDA, rtlpci->tx_ring[VI_QUEUE].dma); | ||
699 | rtl_write_dword(rtlpriv, TVODA, rtlpci->tx_ring[VO_QUEUE].dma); | ||
700 | rtl_write_dword(rtlpriv, TBDA, rtlpci->tx_ring[BEACON_QUEUE].dma); | ||
701 | rtl_write_dword(rtlpriv, TCDA, rtlpci->tx_ring[TXCMD_QUEUE].dma); | ||
702 | rtl_write_dword(rtlpriv, TMDA, rtlpci->tx_ring[MGNT_QUEUE].dma); | ||
703 | rtl_write_dword(rtlpriv, THPDA, rtlpci->tx_ring[HIGH_QUEUE].dma); | ||
704 | rtl_write_dword(rtlpriv, HDA, rtlpci->tx_ring[HCCA_QUEUE].dma); | ||
705 | |||
706 | rtl_write_word(rtlpriv, CMDR, 0x37FC); | ||
707 | |||
708 | /* To make sure that TxDMA can ready to download FW. */ | ||
709 | /* We should reset TxDMA if IMEM RPT was not ready. */ | ||
710 | do { | ||
711 | tmpu1b = rtl_read_byte(rtlpriv, TCR); | ||
712 | if ((tmpu1b & TXDMA_INIT_VALUE) == TXDMA_INIT_VALUE) | ||
713 | break; | ||
714 | |||
715 | udelay(5); | ||
716 | } while (pollingcnt--); | ||
717 | |||
718 | if (pollingcnt <= 0) { | ||
719 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
720 | ("Polling TXDMA_INIT_VALUE " | ||
721 | "timeout!! Current TCR(%#x)\n", tmpu1b)); | ||
722 | tmpu1b = rtl_read_byte(rtlpriv, CMDR); | ||
723 | rtl_write_byte(rtlpriv, CMDR, tmpu1b & (~TXDMA_EN)); | ||
724 | udelay(2); | ||
725 | /* Reset TxDMA */ | ||
726 | rtl_write_byte(rtlpriv, CMDR, tmpu1b | TXDMA_EN); | ||
727 | } | ||
728 | |||
729 | /* After MACIO reset,we must refresh LED state. */ | ||
730 | if ((ppsc->rfoff_reason == RF_CHANGE_BY_IPS) || | ||
731 | (ppsc->rfoff_reason == 0)) { | ||
732 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
733 | struct rtl_led *pLed0 = &(pcipriv->ledctl.sw_led0); | ||
734 | enum rf_pwrstate rfpwr_state_toset; | ||
735 | rfpwr_state_toset = _rtl92se_rf_onoff_detect(hw); | ||
736 | |||
737 | if (rfpwr_state_toset == ERFON) | ||
738 | rtl92se_sw_led_on(hw, pLed0); | ||
739 | } | ||
740 | } | ||
741 | |||
742 | static void _rtl92se_macconfig_after_fwdownload(struct ieee80211_hw *hw) | ||
743 | { | ||
744 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
745 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
746 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
747 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
748 | u8 i; | ||
749 | u16 tmpu2b; | ||
750 | |||
751 | /* 1. System Configure Register (Offset: 0x0000 - 0x003F) */ | ||
752 | |||
753 | /* 2. Command Control Register (Offset: 0x0040 - 0x004F) */ | ||
754 | /* Turn on 0x40 Command register */ | ||
755 | rtl_write_word(rtlpriv, CMDR, (BBRSTN | BB_GLB_RSTN | | ||
756 | SCHEDULE_EN | MACRXEN | MACTXEN | DDMA_EN | FW2HW_EN | | ||
757 | RXDMA_EN | TXDMA_EN | HCI_RXDMA_EN | HCI_TXDMA_EN)); | ||
758 | |||
759 | /* Set TCR TX DMA pre 2 FULL enable bit */ | ||
760 | rtl_write_dword(rtlpriv, TCR, rtl_read_dword(rtlpriv, TCR) | | ||
761 | TXDMAPRE2FULL); | ||
762 | |||
763 | /* Set RCR */ | ||
764 | rtl_write_dword(rtlpriv, RCR, rtlpci->receive_config); | ||
765 | |||
766 | /* 3. MACID Setting Register (Offset: 0x0050 - 0x007F) */ | ||
767 | |||
768 | /* 4. Timing Control Register (Offset: 0x0080 - 0x009F) */ | ||
769 | /* Set CCK/OFDM SIFS */ | ||
770 | /* CCK SIFS shall always be 10us. */ | ||
771 | rtl_write_word(rtlpriv, SIFS_CCK, 0x0a0a); | ||
772 | rtl_write_word(rtlpriv, SIFS_OFDM, 0x1010); | ||
773 | |||
774 | /* Set AckTimeout */ | ||
775 | rtl_write_byte(rtlpriv, ACK_TIMEOUT, 0x40); | ||
776 | |||
777 | /* Beacon related */ | ||
778 | rtl_write_word(rtlpriv, BCN_INTERVAL, 100); | ||
779 | rtl_write_word(rtlpriv, ATIMWND, 2); | ||
780 | |||
781 | /* 5. FIFO Control Register (Offset: 0x00A0 - 0x015F) */ | ||
782 | /* 5.1 Initialize Number of Reserved Pages in Firmware Queue */ | ||
783 | /* Firmware allocate now, associate with FW internal setting.!!! */ | ||
784 | |||
785 | /* 5.2 Setting TX/RX page size 0/1/2/3/4=64/128/256/512/1024 */ | ||
786 | /* 5.3 Set driver info, we only accept PHY status now. */ | ||
787 | /* 5.4 Set RXDMA arbitration to control RXDMA/MAC/FW R/W for RXFIFO */ | ||
788 | rtl_write_byte(rtlpriv, RXDMA, rtl_read_byte(rtlpriv, RXDMA) | BIT(6)); | ||
789 | |||
790 | /* 6. Adaptive Control Register (Offset: 0x0160 - 0x01CF) */ | ||
791 | /* Set RRSR to all legacy rate and HT rate | ||
792 | * CCK rate is supported by default. | ||
793 | * CCK rate will be filtered out only when associated | ||
794 | * AP does not support it. | ||
795 | * Only enable ACK rate to OFDM 24M | ||
796 | * Disable RRSR for CCK rate in A-Cut */ | ||
797 | |||
798 | if (rtlhal->version == VERSION_8192S_ACUT) | ||
799 | rtl_write_byte(rtlpriv, RRSR, 0xf0); | ||
800 | else if (rtlhal->version == VERSION_8192S_BCUT) | ||
801 | rtl_write_byte(rtlpriv, RRSR, 0xff); | ||
802 | rtl_write_byte(rtlpriv, RRSR + 1, 0x01); | ||
803 | rtl_write_byte(rtlpriv, RRSR + 2, 0x00); | ||
804 | |||
805 | /* A-Cut IC do not support CCK rate. We forbid ARFR to */ | ||
806 | /* fallback to CCK rate */ | ||
807 | for (i = 0; i < 8; i++) { | ||
808 | /*Disable RRSR for CCK rate in A-Cut */ | ||
809 | if (rtlhal->version == VERSION_8192S_ACUT) | ||
810 | rtl_write_dword(rtlpriv, ARFR0 + i * 4, 0x1f0ff0f0); | ||
811 | } | ||
812 | |||
813 | /* Different rate use different AMPDU size */ | ||
814 | /* MCS32/ MCS15_SG use max AMPDU size 15*2=30K */ | ||
815 | rtl_write_byte(rtlpriv, AGGLEN_LMT_H, 0x0f); | ||
816 | /* MCS0/1/2/3 use max AMPDU size 4*2=8K */ | ||
817 | rtl_write_word(rtlpriv, AGGLEN_LMT_L, 0x7442); | ||
818 | /* MCS4/5 use max AMPDU size 8*2=16K 6/7 use 10*2=20K */ | ||
819 | rtl_write_word(rtlpriv, AGGLEN_LMT_L + 2, 0xddd7); | ||
820 | /* MCS8/9 use max AMPDU size 8*2=16K 10/11 use 10*2=20K */ | ||
821 | rtl_write_word(rtlpriv, AGGLEN_LMT_L + 4, 0xd772); | ||
822 | /* MCS12/13/14/15 use max AMPDU size 15*2=30K */ | ||
823 | rtl_write_word(rtlpriv, AGGLEN_LMT_L + 6, 0xfffd); | ||
824 | |||
825 | /* Set Data / Response auto rate fallack retry count */ | ||
826 | rtl_write_dword(rtlpriv, DARFRC, 0x04010000); | ||
827 | rtl_write_dword(rtlpriv, DARFRC + 4, 0x09070605); | ||
828 | rtl_write_dword(rtlpriv, RARFRC, 0x04010000); | ||
829 | rtl_write_dword(rtlpriv, RARFRC + 4, 0x09070605); | ||
830 | |||
831 | /* 7. EDCA Setting Register (Offset: 0x01D0 - 0x01FF) */ | ||
832 | /* Set all rate to support SG */ | ||
833 | rtl_write_word(rtlpriv, SG_RATE, 0xFFFF); | ||
834 | |||
835 | /* 8. WMAC, BA, and CCX related Register (Offset: 0x0200 - 0x023F) */ | ||
836 | /* Set NAV protection length */ | ||
837 | rtl_write_word(rtlpriv, NAV_PROT_LEN, 0x0080); | ||
838 | /* CF-END Threshold */ | ||
839 | rtl_write_byte(rtlpriv, CFEND_TH, 0xFF); | ||
840 | /* Set AMPDU minimum space */ | ||
841 | rtl_write_byte(rtlpriv, AMPDU_MIN_SPACE, 0x07); | ||
842 | /* Set TXOP stall control for several queue/HI/BCN/MGT/ */ | ||
843 | rtl_write_byte(rtlpriv, TXOP_STALL_CTRL, 0x00); | ||
844 | |||
845 | /* 9. Security Control Register (Offset: 0x0240 - 0x025F) */ | ||
846 | /* 10. Power Save Control Register (Offset: 0x0260 - 0x02DF) */ | ||
847 | /* 11. General Purpose Register (Offset: 0x02E0 - 0x02FF) */ | ||
848 | /* 12. Host Interrupt Status Register (Offset: 0x0300 - 0x030F) */ | ||
849 | /* 13. Test Mode and Debug Control Register (Offset: 0x0310 - 0x034F) */ | ||
850 | |||
851 | /* 14. Set driver info, we only accept PHY status now. */ | ||
852 | rtl_write_byte(rtlpriv, RXDRVINFO_SZ, 4); | ||
853 | |||
854 | /* 15. For EEPROM R/W Workaround */ | ||
855 | /* 16. For EFUSE to share REG_SYS_FUNC_EN with EEPROM!!! */ | ||
856 | tmpu2b = rtl_read_byte(rtlpriv, REG_SYS_FUNC_EN); | ||
857 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN, tmpu2b | BIT(13)); | ||
858 | tmpu2b = rtl_read_byte(rtlpriv, REG_SYS_ISO_CTRL); | ||
859 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL, tmpu2b & (~BIT(8))); | ||
860 | |||
861 | /* 17. For EFUSE */ | ||
862 | /* We may R/W EFUSE in EEPROM mode */ | ||
863 | if (rtlefuse->epromtype == EEPROM_BOOT_EFUSE) { | ||
864 | u8 tempval; | ||
865 | |||
866 | tempval = rtl_read_byte(rtlpriv, REG_SYS_ISO_CTRL + 1); | ||
867 | tempval &= 0xFE; | ||
868 | rtl_write_byte(rtlpriv, REG_SYS_ISO_CTRL + 1, tempval); | ||
869 | |||
870 | /* Change Program timing */ | ||
871 | rtl_write_byte(rtlpriv, REG_EFUSE_CTRL + 3, 0x72); | ||
872 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("EFUSE CONFIG OK\n")); | ||
873 | } | ||
874 | |||
875 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("OK\n")); | ||
876 | |||
877 | } | ||
878 | |||
879 | static void _rtl92se_hw_configure(struct ieee80211_hw *hw) | ||
880 | { | ||
881 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
882 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
883 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
884 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
885 | |||
886 | u8 reg_bw_opmode = 0; | ||
887 | u32 reg_ratr = 0, reg_rrsr = 0; | ||
888 | u8 regtmp = 0; | ||
889 | |||
890 | reg_bw_opmode = BW_OPMODE_20MHZ; | ||
891 | reg_ratr = RATE_ALL_CCK | RATE_ALL_OFDM_AG | RATE_ALL_OFDM_1SS | | ||
892 | RATE_ALL_OFDM_2SS; | ||
893 | reg_rrsr = RATE_ALL_CCK | RATE_ALL_OFDM_AG; | ||
894 | |||
895 | regtmp = rtl_read_byte(rtlpriv, INIRTSMCS_SEL); | ||
896 | reg_rrsr = ((reg_rrsr & 0x000fffff) << 8) | regtmp; | ||
897 | rtl_write_dword(rtlpriv, INIRTSMCS_SEL, reg_rrsr); | ||
898 | rtl_write_byte(rtlpriv, BW_OPMODE, reg_bw_opmode); | ||
899 | |||
900 | /* Set Retry Limit here */ | ||
901 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RETRY_LIMIT, | ||
902 | (u8 *)(&rtlpci->shortretry_limit)); | ||
903 | |||
904 | rtl_write_byte(rtlpriv, MLT, 0x8f); | ||
905 | |||
906 | /* For Min Spacing configuration. */ | ||
907 | switch (rtlphy->rf_type) { | ||
908 | case RF_1T2R: | ||
909 | case RF_1T1R: | ||
910 | rtlhal->minspace_cfg = (MAX_MSS_DENSITY_1T << 3); | ||
911 | break; | ||
912 | case RF_2T2R: | ||
913 | case RF_2T2R_GREEN: | ||
914 | rtlhal->minspace_cfg = (MAX_MSS_DENSITY_2T << 3); | ||
915 | break; | ||
916 | } | ||
917 | rtl_write_byte(rtlpriv, AMPDU_MIN_SPACE, rtlhal->minspace_cfg); | ||
918 | } | ||
919 | |||
920 | int rtl92se_hw_init(struct ieee80211_hw *hw) | ||
921 | { | ||
922 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
923 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
924 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
925 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
926 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
927 | u8 tmp_byte = 0; | ||
928 | |||
929 | bool rtstatus = true; | ||
930 | u8 tmp_u1b; | ||
931 | int err = false; | ||
932 | u8 i; | ||
933 | int wdcapra_add[] = { | ||
934 | EDCAPARA_BE, EDCAPARA_BK, | ||
935 | EDCAPARA_VI, EDCAPARA_VO}; | ||
936 | u8 secr_value = 0x0; | ||
937 | |||
938 | rtlpci->being_init_adapter = true; | ||
939 | |||
940 | rtlpriv->intf_ops->disable_aspm(hw); | ||
941 | |||
942 | /* 1. MAC Initialize */ | ||
943 | /* Before FW download, we have to set some MAC register */ | ||
944 | _rtl92se_macconfig_before_fwdownload(hw); | ||
945 | |||
946 | rtlhal->version = (enum version_8192s)((rtl_read_dword(rtlpriv, | ||
947 | PMC_FSM) >> 16) & 0xF); | ||
948 | |||
949 | rtl8192se_gpiobit3_cfg_inputmode(hw); | ||
950 | |||
951 | /* 2. download firmware */ | ||
952 | rtstatus = rtl92s_download_fw(hw); | ||
953 | if (!rtstatus) { | ||
954 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
955 | ("Failed to download FW. " | ||
956 | "Init HW without FW now.., Please copy FW into" | ||
957 | "/lib/firmware/rtlwifi\n")); | ||
958 | rtlhal->fw_ready = false; | ||
959 | } else { | ||
960 | rtlhal->fw_ready = true; | ||
961 | } | ||
962 | |||
963 | /* After FW download, we have to reset MAC register */ | ||
964 | _rtl92se_macconfig_after_fwdownload(hw); | ||
965 | |||
966 | /*Retrieve default FW Cmd IO map. */ | ||
967 | rtlhal->fwcmd_iomap = rtl_read_word(rtlpriv, LBUS_MON_ADDR); | ||
968 | rtlhal->fwcmd_ioparam = rtl_read_dword(rtlpriv, LBUS_ADDR_MASK); | ||
969 | |||
970 | /* 3. Initialize MAC/PHY Config by MACPHY_reg.txt */ | ||
971 | if (rtl92s_phy_mac_config(hw) != true) { | ||
972 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("MAC Config failed\n")); | ||
973 | return rtstatus; | ||
974 | } | ||
975 | |||
976 | /* Make sure BB/RF write OK. We should prevent enter IPS. radio off. */ | ||
977 | /* We must set flag avoid BB/RF config period later!! */ | ||
978 | rtl_write_dword(rtlpriv, CMDR, 0x37FC); | ||
979 | |||
980 | /* 4. Initialize BB After MAC Config PHY_reg.txt, AGC_Tab.txt */ | ||
981 | if (rtl92s_phy_bb_config(hw) != true) { | ||
982 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, ("BB Config failed\n")); | ||
983 | return rtstatus; | ||
984 | } | ||
985 | |||
986 | /* 5. Initiailze RF RAIO_A.txt RF RAIO_B.txt */ | ||
987 | /* Before initalizing RF. We can not use FW to do RF-R/W. */ | ||
988 | |||
989 | rtlphy->rf_mode = RF_OP_BY_SW_3WIRE; | ||
990 | |||
991 | /* RF Power Save */ | ||
992 | #if 0 | ||
993 | /* H/W or S/W RF OFF before sleep. */ | ||
994 | if (rtlpriv->psc.rfoff_reason > RF_CHANGE_BY_PS) { | ||
995 | u32 rfoffreason = rtlpriv->psc.rfoff_reason; | ||
996 | |||
997 | rtlpriv->psc.rfoff_reason = RF_CHANGE_BY_INIT; | ||
998 | rtlpriv->psc.rfpwr_state = ERFON; | ||
999 | rtl_ps_set_rf_state(hw, ERFOFF, rfoffreason, true); | ||
1000 | } else { | ||
1001 | /* gpio radio on/off is out of adapter start */ | ||
1002 | if (rtlpriv->psc.hwradiooff == false) { | ||
1003 | rtlpriv->psc.rfpwr_state = ERFON; | ||
1004 | rtlpriv->psc.rfoff_reason = 0; | ||
1005 | } | ||
1006 | } | ||
1007 | #endif | ||
1008 | |||
1009 | /* Before RF-R/W we must execute the IO from Scott's suggestion. */ | ||
1010 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL + 1, 0xDB); | ||
1011 | if (rtlhal->version == VERSION_8192S_ACUT) | ||
1012 | rtl_write_byte(rtlpriv, SPS1_CTRL + 3, 0x07); | ||
1013 | else | ||
1014 | rtl_write_byte(rtlpriv, RF_CTRL, 0x07); | ||
1015 | |||
1016 | if (rtl92s_phy_rf_config(hw) != true) { | ||
1017 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("RF Config failed\n")); | ||
1018 | return rtstatus; | ||
1019 | } | ||
1020 | |||
1021 | /* After read predefined TXT, we must set BB/MAC/RF | ||
1022 | * register as our requirement */ | ||
1023 | |||
1024 | rtlphy->rfreg_chnlval[0] = rtl92s_phy_query_rf_reg(hw, | ||
1025 | (enum radio_path)0, | ||
1026 | RF_CHNLBW, | ||
1027 | RFREG_OFFSET_MASK); | ||
1028 | rtlphy->rfreg_chnlval[1] = rtl92s_phy_query_rf_reg(hw, | ||
1029 | (enum radio_path)1, | ||
1030 | RF_CHNLBW, | ||
1031 | RFREG_OFFSET_MASK); | ||
1032 | |||
1033 | /*---- Set CCK and OFDM Block "ON"----*/ | ||
1034 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BCCKEN, 0x1); | ||
1035 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BOFDMEN, 0x1); | ||
1036 | |||
1037 | /*3 Set Hardware(Do nothing now) */ | ||
1038 | _rtl92se_hw_configure(hw); | ||
1039 | |||
1040 | /* Read EEPROM TX power index and PHY_REG_PG.txt to capture correct */ | ||
1041 | /* TX power index for different rate set. */ | ||
1042 | /* Get original hw reg values */ | ||
1043 | rtl92s_phy_get_hw_reg_originalvalue(hw); | ||
1044 | /* Write correct tx power index */ | ||
1045 | rtl92s_phy_set_txpower(hw, rtlphy->current_channel); | ||
1046 | |||
1047 | /* We must set MAC address after firmware download. */ | ||
1048 | for (i = 0; i < 6; i++) | ||
1049 | rtl_write_byte(rtlpriv, MACIDR0 + i, rtlefuse->dev_addr[i]); | ||
1050 | |||
1051 | /* EEPROM R/W workaround */ | ||
1052 | tmp_u1b = rtl_read_byte(rtlpriv, MAC_PINMUX_CFG); | ||
1053 | rtl_write_byte(rtlpriv, MAC_PINMUX_CFG, tmp_u1b & (~BIT(3))); | ||
1054 | |||
1055 | rtl_write_byte(rtlpriv, 0x4d, 0x0); | ||
1056 | |||
1057 | if (hal_get_firmwareversion(rtlpriv) >= 0x49) { | ||
1058 | tmp_byte = rtl_read_byte(rtlpriv, FW_RSVD_PG_CRTL) & (~BIT(4)); | ||
1059 | tmp_byte = tmp_byte | BIT(5); | ||
1060 | rtl_write_byte(rtlpriv, FW_RSVD_PG_CRTL, tmp_byte); | ||
1061 | rtl_write_dword(rtlpriv, TXDESC_MSK, 0xFFFFCFFF); | ||
1062 | } | ||
1063 | |||
1064 | /* We enable high power and RA related mechanism after NIC | ||
1065 | * initialized. */ | ||
1066 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_RA_INIT); | ||
1067 | |||
1068 | /* Add to prevent ASPM bug. */ | ||
1069 | /* Always enable hst and NIC clock request. */ | ||
1070 | rtl92s_phy_switch_ephy_parameter(hw); | ||
1071 | |||
1072 | /* Security related | ||
1073 | * 1. Clear all H/W keys. | ||
1074 | * 2. Enable H/W encryption/decryption. */ | ||
1075 | rtl_cam_reset_all_entry(hw); | ||
1076 | secr_value |= SCR_TXENCENABLE; | ||
1077 | secr_value |= SCR_RXENCENABLE; | ||
1078 | secr_value |= SCR_NOSKMC; | ||
1079 | rtl_write_byte(rtlpriv, REG_SECR, secr_value); | ||
1080 | |||
1081 | for (i = 0; i < 4; i++) | ||
1082 | rtl_write_dword(rtlpriv, wdcapra_add[i], 0x5e4322); | ||
1083 | |||
1084 | if (rtlphy->rf_type == RF_1T2R) { | ||
1085 | bool mrc2set = true; | ||
1086 | /* Turn on B-Path */ | ||
1087 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_MRC, (u8 *)&mrc2set); | ||
1088 | } | ||
1089 | |||
1090 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_POWER_ON); | ||
1091 | rtl92s_dm_init(hw); | ||
1092 | rtlpci->being_init_adapter = false; | ||
1093 | |||
1094 | return err; | ||
1095 | } | ||
1096 | |||
1097 | void rtl92se_set_mac_addr(struct rtl_io *io, const u8 * addr) | ||
1098 | { | ||
1099 | } | ||
1100 | |||
1101 | void rtl92se_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid) | ||
1102 | { | ||
1103 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1104 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1105 | u32 reg_rcr = rtlpci->receive_config; | ||
1106 | |||
1107 | if (rtlpriv->psc.rfpwr_state != ERFON) | ||
1108 | return; | ||
1109 | |||
1110 | if (check_bssid == true) { | ||
1111 | reg_rcr |= (RCR_CBSSID); | ||
1112 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, (u8 *)(®_rcr)); | ||
1113 | } else if (check_bssid == false) { | ||
1114 | reg_rcr &= (~RCR_CBSSID); | ||
1115 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_RCR, (u8 *)(®_rcr)); | ||
1116 | } | ||
1117 | |||
1118 | } | ||
1119 | |||
1120 | static int _rtl92se_set_media_status(struct ieee80211_hw *hw, | ||
1121 | enum nl80211_iftype type) | ||
1122 | { | ||
1123 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1124 | u8 bt_msr = rtl_read_byte(rtlpriv, MSR); | ||
1125 | enum led_ctl_mode ledaction = LED_CTL_NO_LINK; | ||
1126 | u32 temp; | ||
1127 | bt_msr &= ~MSR_LINK_MASK; | ||
1128 | |||
1129 | switch (type) { | ||
1130 | case NL80211_IFTYPE_UNSPECIFIED: | ||
1131 | bt_msr |= (MSR_LINK_NONE << MSR_LINK_SHIFT); | ||
1132 | ledaction = LED_CTL_LINK; | ||
1133 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1134 | ("Set Network type to NO LINK!\n")); | ||
1135 | break; | ||
1136 | case NL80211_IFTYPE_ADHOC: | ||
1137 | bt_msr |= (MSR_LINK_ADHOC << MSR_LINK_SHIFT); | ||
1138 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1139 | ("Set Network type to Ad Hoc!\n")); | ||
1140 | break; | ||
1141 | case NL80211_IFTYPE_STATION: | ||
1142 | bt_msr |= (MSR_LINK_MANAGED << MSR_LINK_SHIFT); | ||
1143 | ledaction = LED_CTL_LINK; | ||
1144 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1145 | ("Set Network type to STA!\n")); | ||
1146 | break; | ||
1147 | case NL80211_IFTYPE_AP: | ||
1148 | bt_msr |= (MSR_LINK_MASTER << MSR_LINK_SHIFT); | ||
1149 | RT_TRACE(rtlpriv, COMP_INIT, DBG_TRACE, | ||
1150 | ("Set Network type to AP!\n")); | ||
1151 | break; | ||
1152 | default: | ||
1153 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1154 | ("Network type %d not support!\n", type)); | ||
1155 | return 1; | ||
1156 | break; | ||
1157 | |||
1158 | } | ||
1159 | |||
1160 | rtl_write_byte(rtlpriv, (MSR), bt_msr); | ||
1161 | |||
1162 | temp = rtl_read_dword(rtlpriv, TCR); | ||
1163 | rtl_write_dword(rtlpriv, TCR, temp & (~BIT(8))); | ||
1164 | rtl_write_dword(rtlpriv, TCR, temp | BIT(8)); | ||
1165 | |||
1166 | |||
1167 | return 0; | ||
1168 | } | ||
1169 | |||
1170 | /* HW_VAR_MEDIA_STATUS & HW_VAR_CECHK_BSSID */ | ||
1171 | int rtl92se_set_network_type(struct ieee80211_hw *hw, enum nl80211_iftype type) | ||
1172 | { | ||
1173 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1174 | |||
1175 | if (_rtl92se_set_media_status(hw, type)) | ||
1176 | return -EOPNOTSUPP; | ||
1177 | |||
1178 | if (rtlpriv->mac80211.link_state == MAC80211_LINKED) { | ||
1179 | if (type != NL80211_IFTYPE_AP) | ||
1180 | rtl92se_set_check_bssid(hw, true); | ||
1181 | } else { | ||
1182 | rtl92se_set_check_bssid(hw, false); | ||
1183 | } | ||
1184 | |||
1185 | return 0; | ||
1186 | } | ||
1187 | |||
1188 | /* don't set REG_EDCA_BE_PARAM here because mac80211 will send pkt when scan */ | ||
1189 | void rtl92se_set_qos(struct ieee80211_hw *hw, int aci) | ||
1190 | { | ||
1191 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1192 | rtl92s_dm_init_edca_turbo(hw); | ||
1193 | |||
1194 | switch (aci) { | ||
1195 | case AC1_BK: | ||
1196 | rtl_write_dword(rtlpriv, EDCAPARA_BK, 0xa44f); | ||
1197 | break; | ||
1198 | case AC0_BE: | ||
1199 | /* rtl_write_dword(rtlpriv, EDCAPARA_BE, u4b_ac_param); */ | ||
1200 | break; | ||
1201 | case AC2_VI: | ||
1202 | rtl_write_dword(rtlpriv, EDCAPARA_VI, 0x5e4322); | ||
1203 | break; | ||
1204 | case AC3_VO: | ||
1205 | rtl_write_dword(rtlpriv, EDCAPARA_VO, 0x2f3222); | ||
1206 | break; | ||
1207 | default: | ||
1208 | RT_ASSERT(false, ("invalid aci: %d !\n", aci)); | ||
1209 | break; | ||
1210 | } | ||
1211 | } | ||
1212 | |||
1213 | void rtl92se_enable_interrupt(struct ieee80211_hw *hw) | ||
1214 | { | ||
1215 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1216 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1217 | |||
1218 | rtl_write_dword(rtlpriv, INTA_MASK, rtlpci->irq_mask[0]); | ||
1219 | /* Support Bit 32-37(Assign as Bit 0-5) interrupt setting now */ | ||
1220 | rtl_write_dword(rtlpriv, INTA_MASK + 4, rtlpci->irq_mask[1] & 0x3F); | ||
1221 | |||
1222 | rtlpci->irq_enabled = true; | ||
1223 | } | ||
1224 | |||
1225 | void rtl92se_disable_interrupt(struct ieee80211_hw *hw) | ||
1226 | { | ||
1227 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1228 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1229 | |||
1230 | rtl_write_dword(rtlpriv, INTA_MASK, 0); | ||
1231 | rtl_write_dword(rtlpriv, INTA_MASK + 4, 0); | ||
1232 | |||
1233 | rtlpci->irq_enabled = false; | ||
1234 | } | ||
1235 | |||
1236 | |||
1237 | static u8 _rtl92s_set_sysclk(struct ieee80211_hw *hw, u8 data) | ||
1238 | { | ||
1239 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1240 | u8 waitcnt = 100; | ||
1241 | bool result = false; | ||
1242 | u8 tmp; | ||
1243 | |||
1244 | rtl_write_byte(rtlpriv, SYS_CLKR + 1, data); | ||
1245 | |||
1246 | /* Wait the MAC synchronized. */ | ||
1247 | udelay(400); | ||
1248 | |||
1249 | /* Check if it is set ready. */ | ||
1250 | tmp = rtl_read_byte(rtlpriv, SYS_CLKR + 1); | ||
1251 | result = ((tmp & BIT(7)) == (data & BIT(7))); | ||
1252 | |||
1253 | if ((data & (BIT(6) | BIT(7))) == false) { | ||
1254 | waitcnt = 100; | ||
1255 | tmp = 0; | ||
1256 | |||
1257 | while (1) { | ||
1258 | waitcnt--; | ||
1259 | tmp = rtl_read_byte(rtlpriv, SYS_CLKR + 1); | ||
1260 | |||
1261 | if ((tmp & BIT(6))) | ||
1262 | break; | ||
1263 | |||
1264 | printk(KERN_ERR "wait for BIT(6) return value %x\n", | ||
1265 | tmp); | ||
1266 | |||
1267 | if (waitcnt == 0) | ||
1268 | break; | ||
1269 | udelay(10); | ||
1270 | } | ||
1271 | |||
1272 | if (waitcnt == 0) | ||
1273 | result = false; | ||
1274 | else | ||
1275 | result = true; | ||
1276 | } | ||
1277 | |||
1278 | return result; | ||
1279 | } | ||
1280 | |||
1281 | static void _rtl92s_phy_set_rfhalt(struct ieee80211_hw *hw) | ||
1282 | { | ||
1283 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1284 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1285 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1286 | u8 u1btmp; | ||
1287 | |||
1288 | if (rtlhal->driver_going2unload) | ||
1289 | rtl_write_byte(rtlpriv, 0x560, 0x0); | ||
1290 | |||
1291 | /* Power save for BB/RF */ | ||
1292 | u1btmp = rtl_read_byte(rtlpriv, LDOV12D_CTRL); | ||
1293 | u1btmp |= BIT(0); | ||
1294 | rtl_write_byte(rtlpriv, LDOV12D_CTRL, u1btmp); | ||
1295 | rtl_write_byte(rtlpriv, SPS1_CTRL, 0x0); | ||
1296 | rtl_write_byte(rtlpriv, TXPAUSE, 0xFF); | ||
1297 | rtl_write_word(rtlpriv, CMDR, 0x57FC); | ||
1298 | udelay(100); | ||
1299 | rtl_write_word(rtlpriv, CMDR, 0x77FC); | ||
1300 | rtl_write_byte(rtlpriv, PHY_CCA, 0x0); | ||
1301 | udelay(10); | ||
1302 | rtl_write_word(rtlpriv, CMDR, 0x37FC); | ||
1303 | udelay(10); | ||
1304 | rtl_write_word(rtlpriv, CMDR, 0x77FC); | ||
1305 | udelay(10); | ||
1306 | rtl_write_word(rtlpriv, CMDR, 0x57FC); | ||
1307 | rtl_write_word(rtlpriv, CMDR, 0x0000); | ||
1308 | |||
1309 | if (rtlhal->driver_going2unload) { | ||
1310 | u1btmp = rtl_read_byte(rtlpriv, (REG_SYS_FUNC_EN + 1)); | ||
1311 | u1btmp &= ~(BIT(0)); | ||
1312 | rtl_write_byte(rtlpriv, REG_SYS_FUNC_EN + 1, u1btmp); | ||
1313 | } | ||
1314 | |||
1315 | u1btmp = rtl_read_byte(rtlpriv, (SYS_CLKR + 1)); | ||
1316 | |||
1317 | /* Add description. After switch control path. register | ||
1318 | * after page1 will be invisible. We can not do any IO | ||
1319 | * for register>0x40. After resume&MACIO reset, we need | ||
1320 | * to remember previous reg content. */ | ||
1321 | if (u1btmp & BIT(7)) { | ||
1322 | u1btmp &= ~(BIT(6) | BIT(7)); | ||
1323 | if (!_rtl92s_set_sysclk(hw, u1btmp)) { | ||
1324 | printk(KERN_ERR "Switch ctrl path fail\n"); | ||
1325 | return; | ||
1326 | } | ||
1327 | } | ||
1328 | |||
1329 | /* Power save for MAC */ | ||
1330 | if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS && | ||
1331 | !rtlhal->driver_going2unload) { | ||
1332 | /* enable LED function */ | ||
1333 | rtl_write_byte(rtlpriv, 0x03, 0xF9); | ||
1334 | /* SW/HW radio off or halt adapter!! For example S3/S4 */ | ||
1335 | } else { | ||
1336 | /* LED function disable. Power range is about 8mA now. */ | ||
1337 | /* if write 0xF1 disconnet_pci power | ||
1338 | * ifconfig wlan0 down power are both high 35:70 */ | ||
1339 | /* if write oxF9 disconnet_pci power | ||
1340 | * ifconfig wlan0 down power are both low 12:45*/ | ||
1341 | rtl_write_byte(rtlpriv, 0x03, 0xF9); | ||
1342 | } | ||
1343 | |||
1344 | rtl_write_byte(rtlpriv, SYS_CLKR + 1, 0x70); | ||
1345 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL + 1, 0x68); | ||
1346 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, 0x00); | ||
1347 | rtl_write_byte(rtlpriv, LDOA15_CTRL, 0x34); | ||
1348 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL, 0x0E); | ||
1349 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
1350 | |||
1351 | } | ||
1352 | |||
1353 | static void _rtl92se_gen_refreshledstate(struct ieee80211_hw *hw) | ||
1354 | { | ||
1355 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1356 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1357 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
1358 | struct rtl_led *pLed0 = &(pcipriv->ledctl.sw_led0); | ||
1359 | |||
1360 | if (rtlpci->up_first_time == 1) | ||
1361 | return; | ||
1362 | |||
1363 | if (rtlpriv->psc.rfoff_reason == RF_CHANGE_BY_IPS) | ||
1364 | rtl92se_sw_led_on(hw, pLed0); | ||
1365 | else | ||
1366 | rtl92se_sw_led_off(hw, pLed0); | ||
1367 | } | ||
1368 | |||
1369 | |||
1370 | static void _rtl92se_power_domain_init(struct ieee80211_hw *hw) | ||
1371 | { | ||
1372 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1373 | u16 tmpu2b; | ||
1374 | u8 tmpu1b; | ||
1375 | |||
1376 | rtlpriv->psc.pwrdomain_protect = true; | ||
1377 | |||
1378 | tmpu1b = rtl_read_byte(rtlpriv, (SYS_CLKR + 1)); | ||
1379 | if (tmpu1b & BIT(7)) { | ||
1380 | tmpu1b &= ~(BIT(6) | BIT(7)); | ||
1381 | if (!_rtl92s_set_sysclk(hw, tmpu1b)) { | ||
1382 | rtlpriv->psc.pwrdomain_protect = false; | ||
1383 | return; | ||
1384 | } | ||
1385 | } | ||
1386 | |||
1387 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, 0x0); | ||
1388 | rtl_write_byte(rtlpriv, LDOA15_CTRL, 0x34); | ||
1389 | |||
1390 | /* Reset MAC-IO and CPU and Core Digital BIT10/11/15 */ | ||
1391 | tmpu1b = rtl_read_byte(rtlpriv, SYS_FUNC_EN + 1); | ||
1392 | |||
1393 | /* If IPS we need to turn LED on. So we not | ||
1394 | * not disable BIT 3/7 of reg3. */ | ||
1395 | if (rtlpriv->psc.rfoff_reason & (RF_CHANGE_BY_IPS | RF_CHANGE_BY_HW)) | ||
1396 | tmpu1b &= 0xFB; | ||
1397 | else | ||
1398 | tmpu1b &= 0x73; | ||
1399 | |||
1400 | rtl_write_byte(rtlpriv, SYS_FUNC_EN + 1, tmpu1b); | ||
1401 | /* wait for BIT 10/11/15 to pull high automatically!! */ | ||
1402 | mdelay(1); | ||
1403 | |||
1404 | rtl_write_byte(rtlpriv, CMDR, 0); | ||
1405 | rtl_write_byte(rtlpriv, TCR, 0); | ||
1406 | |||
1407 | /* Data sheet not define 0x562!!! Copy from WMAC!!!!! */ | ||
1408 | tmpu1b = rtl_read_byte(rtlpriv, 0x562); | ||
1409 | tmpu1b |= 0x08; | ||
1410 | rtl_write_byte(rtlpriv, 0x562, tmpu1b); | ||
1411 | tmpu1b &= ~(BIT(3)); | ||
1412 | rtl_write_byte(rtlpriv, 0x562, tmpu1b); | ||
1413 | |||
1414 | /* Enable AFE clock source */ | ||
1415 | tmpu1b = rtl_read_byte(rtlpriv, AFE_XTAL_CTRL); | ||
1416 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL, (tmpu1b | 0x01)); | ||
1417 | /* Delay 1.5ms */ | ||
1418 | udelay(1500); | ||
1419 | tmpu1b = rtl_read_byte(rtlpriv, AFE_XTAL_CTRL + 1); | ||
1420 | rtl_write_byte(rtlpriv, AFE_XTAL_CTRL + 1, (tmpu1b & 0xfb)); | ||
1421 | |||
1422 | /* Enable AFE Macro Block's Bandgap */ | ||
1423 | tmpu1b = rtl_read_byte(rtlpriv, AFE_MISC); | ||
1424 | rtl_write_byte(rtlpriv, AFE_MISC, (tmpu1b | BIT(0))); | ||
1425 | mdelay(1); | ||
1426 | |||
1427 | /* Enable AFE Mbias */ | ||
1428 | tmpu1b = rtl_read_byte(rtlpriv, AFE_MISC); | ||
1429 | rtl_write_byte(rtlpriv, AFE_MISC, (tmpu1b | 0x02)); | ||
1430 | mdelay(1); | ||
1431 | |||
1432 | /* Enable LDOA15 block */ | ||
1433 | tmpu1b = rtl_read_byte(rtlpriv, LDOA15_CTRL); | ||
1434 | rtl_write_byte(rtlpriv, LDOA15_CTRL, (tmpu1b | BIT(0))); | ||
1435 | |||
1436 | /* Set Digital Vdd to Retention isolation Path. */ | ||
1437 | tmpu2b = rtl_read_word(rtlpriv, SYS_ISO_CTRL); | ||
1438 | rtl_write_word(rtlpriv, SYS_ISO_CTRL, (tmpu2b | BIT(11))); | ||
1439 | |||
1440 | |||
1441 | /* For warm reboot NIC disappera bug. */ | ||
1442 | tmpu2b = rtl_read_word(rtlpriv, SYS_FUNC_EN); | ||
1443 | rtl_write_word(rtlpriv, SYS_FUNC_EN, (tmpu2b | BIT(13))); | ||
1444 | |||
1445 | rtl_write_byte(rtlpriv, SYS_ISO_CTRL + 1, 0x68); | ||
1446 | |||
1447 | /* Enable AFE PLL Macro Block */ | ||
1448 | tmpu1b = rtl_read_byte(rtlpriv, AFE_PLL_CTRL); | ||
1449 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL, (tmpu1b | BIT(0) | BIT(4))); | ||
1450 | /* Enable MAC 80MHZ clock */ | ||
1451 | tmpu1b = rtl_read_byte(rtlpriv, AFE_PLL_CTRL + 1); | ||
1452 | rtl_write_byte(rtlpriv, AFE_PLL_CTRL + 1, (tmpu1b | BIT(0))); | ||
1453 | mdelay(1); | ||
1454 | |||
1455 | /* Release isolation AFE PLL & MD */ | ||
1456 | rtl_write_byte(rtlpriv, SYS_ISO_CTRL, 0xA6); | ||
1457 | |||
1458 | /* Enable MAC clock */ | ||
1459 | tmpu2b = rtl_read_word(rtlpriv, SYS_CLKR); | ||
1460 | rtl_write_word(rtlpriv, SYS_CLKR, (tmpu2b | BIT(12) | BIT(11))); | ||
1461 | |||
1462 | /* Enable Core digital and enable IOREG R/W */ | ||
1463 | tmpu2b = rtl_read_word(rtlpriv, SYS_FUNC_EN); | ||
1464 | rtl_write_word(rtlpriv, SYS_FUNC_EN, (tmpu2b | BIT(11))); | ||
1465 | /* enable REG_EN */ | ||
1466 | rtl_write_word(rtlpriv, SYS_FUNC_EN, (tmpu2b | BIT(11) | BIT(15))); | ||
1467 | |||
1468 | /* Switch the control path. */ | ||
1469 | tmpu2b = rtl_read_word(rtlpriv, SYS_CLKR); | ||
1470 | rtl_write_word(rtlpriv, SYS_CLKR, (tmpu2b & (~BIT(2)))); | ||
1471 | |||
1472 | tmpu1b = rtl_read_byte(rtlpriv, (SYS_CLKR + 1)); | ||
1473 | tmpu1b = ((tmpu1b | BIT(7)) & (~BIT(6))); | ||
1474 | if (!_rtl92s_set_sysclk(hw, tmpu1b)) { | ||
1475 | rtlpriv->psc.pwrdomain_protect = false; | ||
1476 | return; | ||
1477 | } | ||
1478 | |||
1479 | rtl_write_word(rtlpriv, CMDR, 0x37FC); | ||
1480 | |||
1481 | /* After MACIO reset,we must refresh LED state. */ | ||
1482 | _rtl92se_gen_refreshledstate(hw); | ||
1483 | |||
1484 | rtlpriv->psc.pwrdomain_protect = false; | ||
1485 | } | ||
1486 | |||
1487 | void rtl92se_card_disable(struct ieee80211_hw *hw) | ||
1488 | { | ||
1489 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1490 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1491 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1492 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1493 | enum nl80211_iftype opmode; | ||
1494 | u8 wait = 30; | ||
1495 | |||
1496 | rtlpriv->intf_ops->enable_aspm(hw); | ||
1497 | |||
1498 | if (rtlpci->driver_is_goingto_unload || | ||
1499 | ppsc->rfoff_reason > RF_CHANGE_BY_PS) | ||
1500 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_POWER_OFF); | ||
1501 | |||
1502 | /* we should chnge GPIO to input mode | ||
1503 | * this will drop away current about 25mA*/ | ||
1504 | rtl8192se_gpiobit3_cfg_inputmode(hw); | ||
1505 | |||
1506 | /* this is very important for ips power save */ | ||
1507 | while (wait-- >= 10 && rtlpriv->psc.pwrdomain_protect) { | ||
1508 | if (rtlpriv->psc.pwrdomain_protect) | ||
1509 | mdelay(20); | ||
1510 | else | ||
1511 | break; | ||
1512 | } | ||
1513 | |||
1514 | mac->link_state = MAC80211_NOLINK; | ||
1515 | opmode = NL80211_IFTYPE_UNSPECIFIED; | ||
1516 | _rtl92se_set_media_status(hw, opmode); | ||
1517 | |||
1518 | _rtl92s_phy_set_rfhalt(hw); | ||
1519 | udelay(100); | ||
1520 | } | ||
1521 | |||
1522 | void rtl92se_interrupt_recognized(struct ieee80211_hw *hw, u32 *p_inta, | ||
1523 | u32 *p_intb) | ||
1524 | { | ||
1525 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1526 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1527 | |||
1528 | *p_inta = rtl_read_dword(rtlpriv, ISR) & rtlpci->irq_mask[0]; | ||
1529 | rtl_write_dword(rtlpriv, ISR, *p_inta); | ||
1530 | |||
1531 | *p_intb = rtl_read_dword(rtlpriv, ISR + 4) & rtlpci->irq_mask[1]; | ||
1532 | rtl_write_dword(rtlpriv, ISR + 4, *p_intb); | ||
1533 | } | ||
1534 | |||
1535 | void rtl92se_set_beacon_related_registers(struct ieee80211_hw *hw) | ||
1536 | { | ||
1537 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1538 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1539 | u16 bcntime_cfg = 0; | ||
1540 | u16 bcn_cw = 6, bcn_ifs = 0xf; | ||
1541 | u16 atim_window = 2; | ||
1542 | |||
1543 | /* ATIM Window (in unit of TU). */ | ||
1544 | rtl_write_word(rtlpriv, ATIMWND, atim_window); | ||
1545 | |||
1546 | /* Beacon interval (in unit of TU). */ | ||
1547 | rtl_write_word(rtlpriv, BCN_INTERVAL, mac->beacon_interval); | ||
1548 | |||
1549 | /* DrvErlyInt (in unit of TU). (Time to send | ||
1550 | * interrupt to notify driver to change | ||
1551 | * beacon content) */ | ||
1552 | rtl_write_word(rtlpriv, BCN_DRV_EARLY_INT, 10 << 4); | ||
1553 | |||
1554 | /* BcnDMATIM(in unit of us). Indicates the | ||
1555 | * time before TBTT to perform beacon queue DMA */ | ||
1556 | rtl_write_word(rtlpriv, BCN_DMATIME, 256); | ||
1557 | |||
1558 | /* Force beacon frame transmission even | ||
1559 | * after receiving beacon frame from | ||
1560 | * other ad hoc STA */ | ||
1561 | rtl_write_byte(rtlpriv, BCN_ERR_THRESH, 100); | ||
1562 | |||
1563 | /* Beacon Time Configuration */ | ||
1564 | if (mac->opmode == NL80211_IFTYPE_ADHOC) | ||
1565 | bcntime_cfg |= (bcn_cw << BCN_TCFG_CW_SHIFT); | ||
1566 | |||
1567 | /* TODO: bcn_ifs may required to be changed on ASIC */ | ||
1568 | bcntime_cfg |= bcn_ifs << BCN_TCFG_IFS; | ||
1569 | |||
1570 | /*for beacon changed */ | ||
1571 | rtl92s_phy_set_beacon_hwreg(hw, mac->beacon_interval); | ||
1572 | } | ||
1573 | |||
1574 | void rtl92se_set_beacon_interval(struct ieee80211_hw *hw) | ||
1575 | { | ||
1576 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1577 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1578 | u16 bcn_interval = mac->beacon_interval; | ||
1579 | |||
1580 | /* Beacon interval (in unit of TU). */ | ||
1581 | rtl_write_word(rtlpriv, BCN_INTERVAL, bcn_interval); | ||
1582 | /* 2008.10.24 added by tynli for beacon changed. */ | ||
1583 | rtl92s_phy_set_beacon_hwreg(hw, bcn_interval); | ||
1584 | } | ||
1585 | |||
1586 | void rtl92se_update_interrupt_mask(struct ieee80211_hw *hw, | ||
1587 | u32 add_msr, u32 rm_msr) | ||
1588 | { | ||
1589 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1590 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
1591 | |||
1592 | RT_TRACE(rtlpriv, COMP_INTR, DBG_LOUD, | ||
1593 | ("add_msr:%x, rm_msr:%x\n", add_msr, rm_msr)); | ||
1594 | |||
1595 | if (add_msr) | ||
1596 | rtlpci->irq_mask[0] |= add_msr; | ||
1597 | |||
1598 | if (rm_msr) | ||
1599 | rtlpci->irq_mask[0] &= (~rm_msr); | ||
1600 | |||
1601 | rtl92se_disable_interrupt(hw); | ||
1602 | rtl92se_enable_interrupt(hw); | ||
1603 | } | ||
1604 | |||
1605 | static void _rtl8192se_get_IC_Inferiority(struct ieee80211_hw *hw) | ||
1606 | { | ||
1607 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1608 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1609 | u8 efuse_id; | ||
1610 | |||
1611 | rtlhal->ic_class = IC_INFERIORITY_A; | ||
1612 | |||
1613 | /* Only retrieving while using EFUSE. */ | ||
1614 | if ((rtlefuse->epromtype == EEPROM_BOOT_EFUSE) && | ||
1615 | !rtlefuse->autoload_failflag) { | ||
1616 | efuse_id = efuse_read_1byte(hw, EFUSE_IC_ID_OFFSET); | ||
1617 | |||
1618 | if (efuse_id == 0xfe) | ||
1619 | rtlhal->ic_class = IC_INFERIORITY_B; | ||
1620 | } | ||
1621 | } | ||
1622 | |||
1623 | static void _rtl92se_read_adapter_info(struct ieee80211_hw *hw) | ||
1624 | { | ||
1625 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1626 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1627 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1628 | u16 i, usvalue; | ||
1629 | u16 eeprom_id; | ||
1630 | u8 tempval; | ||
1631 | u8 hwinfo[HWSET_MAX_SIZE_92S]; | ||
1632 | u8 rf_path, index; | ||
1633 | |||
1634 | if (rtlefuse->epromtype == EEPROM_93C46) { | ||
1635 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
1636 | ("RTL819X Not boot from eeprom, check it !!")); | ||
1637 | } else if (rtlefuse->epromtype == EEPROM_BOOT_EFUSE) { | ||
1638 | rtl_efuse_shadow_map_update(hw); | ||
1639 | |||
1640 | memcpy((void *)hwinfo, (void *) | ||
1641 | &rtlefuse->efuse_map[EFUSE_INIT_MAP][0], | ||
1642 | HWSET_MAX_SIZE_92S); | ||
1643 | } | ||
1644 | |||
1645 | RT_PRINT_DATA(rtlpriv, COMP_INIT, DBG_DMESG, ("MAP\n"), | ||
1646 | hwinfo, HWSET_MAX_SIZE_92S); | ||
1647 | |||
1648 | eeprom_id = *((u16 *)&hwinfo[0]); | ||
1649 | if (eeprom_id != RTL8190_EEPROM_ID) { | ||
1650 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
1651 | ("EEPROM ID(%#x) is invalid!!\n", eeprom_id)); | ||
1652 | rtlefuse->autoload_failflag = true; | ||
1653 | } else { | ||
1654 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload OK\n")); | ||
1655 | rtlefuse->autoload_failflag = false; | ||
1656 | } | ||
1657 | |||
1658 | if (rtlefuse->autoload_failflag == true) | ||
1659 | return; | ||
1660 | |||
1661 | _rtl8192se_get_IC_Inferiority(hw); | ||
1662 | |||
1663 | /* Read IC Version && Channel Plan */ | ||
1664 | /* VID, DID SE 0xA-D */ | ||
1665 | rtlefuse->eeprom_vid = *(u16 *)&hwinfo[EEPROM_VID]; | ||
1666 | rtlefuse->eeprom_did = *(u16 *)&hwinfo[EEPROM_DID]; | ||
1667 | rtlefuse->eeprom_svid = *(u16 *)&hwinfo[EEPROM_SVID]; | ||
1668 | rtlefuse->eeprom_smid = *(u16 *)&hwinfo[EEPROM_SMID]; | ||
1669 | rtlefuse->eeprom_version = *(u16 *)&hwinfo[EEPROM_VERSION]; | ||
1670 | |||
1671 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1672 | ("EEPROMId = 0x%4x\n", eeprom_id)); | ||
1673 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1674 | ("EEPROM VID = 0x%4x\n", rtlefuse->eeprom_vid)); | ||
1675 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1676 | ("EEPROM DID = 0x%4x\n", rtlefuse->eeprom_did)); | ||
1677 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1678 | ("EEPROM SVID = 0x%4x\n", rtlefuse->eeprom_svid)); | ||
1679 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1680 | ("EEPROM SMID = 0x%4x\n", rtlefuse->eeprom_smid)); | ||
1681 | |||
1682 | for (i = 0; i < 6; i += 2) { | ||
1683 | usvalue = *(u16 *)&hwinfo[EEPROM_MAC_ADDR + i]; | ||
1684 | *((u16 *) (&rtlefuse->dev_addr[i])) = usvalue; | ||
1685 | } | ||
1686 | |||
1687 | for (i = 0; i < 6; i++) | ||
1688 | rtl_write_byte(rtlpriv, MACIDR0 + i, rtlefuse->dev_addr[i]); | ||
1689 | |||
1690 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
1691 | (MAC_FMT "\n", MAC_ARG(rtlefuse->dev_addr))); | ||
1692 | |||
1693 | /* Get Tx Power Level by Channel */ | ||
1694 | /* Read Tx power of Channel 1 ~ 14 from EEPROM. */ | ||
1695 | /* 92S suupport RF A & B */ | ||
1696 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1697 | for (i = 0; i < 3; i++) { | ||
1698 | /* Read CCK RF A & B Tx power */ | ||
1699 | rtlefuse->eeprom_chnlarea_txpwr_cck[rf_path][i] = | ||
1700 | hwinfo[EEPROM_TXPOWERBASE + rf_path * 3 + i]; | ||
1701 | |||
1702 | /* Read OFDM RF A & B Tx power for 1T */ | ||
1703 | rtlefuse->eeprom_chnlarea_txpwr_ht40_1s[rf_path][i] = | ||
1704 | hwinfo[EEPROM_TXPOWERBASE + 6 + rf_path * 3 + i]; | ||
1705 | |||
1706 | /* Read OFDM RF A & B Tx power for 2T */ | ||
1707 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif[rf_path][i] | ||
1708 | = hwinfo[EEPROM_TXPOWERBASE + 12 + | ||
1709 | rf_path * 3 + i]; | ||
1710 | } | ||
1711 | } | ||
1712 | |||
1713 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1714 | for (i = 0; i < 3; i++) | ||
1715 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1716 | ("RF(%d) EEPROM CCK Area(%d) = 0x%x\n", rf_path, | ||
1717 | i, rtlefuse->eeprom_chnlarea_txpwr_cck | ||
1718 | [rf_path][i])); | ||
1719 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1720 | for (i = 0; i < 3; i++) | ||
1721 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1722 | ("RF(%d) EEPROM HT40 1S Area(%d) = 0x%x\n", | ||
1723 | rf_path, i, | ||
1724 | rtlefuse->eeprom_chnlarea_txpwr_ht40_1s | ||
1725 | [rf_path][i])); | ||
1726 | for (rf_path = 0; rf_path < 2; rf_path++) | ||
1727 | for (i = 0; i < 3; i++) | ||
1728 | RTPRINT(rtlpriv, FINIT, INIT_EEPROM, | ||
1729 | ("RF(%d) EEPROM HT40 2S Diff Area(%d) = 0x%x\n", | ||
1730 | rf_path, i, | ||
1731 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif | ||
1732 | [rf_path][i])); | ||
1733 | |||
1734 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1735 | |||
1736 | /* Assign dedicated channel tx power */ | ||
1737 | for (i = 0; i < 14; i++) { | ||
1738 | /* channel 1~3 use the same Tx Power Level. */ | ||
1739 | if (i < 3) | ||
1740 | index = 0; | ||
1741 | /* Channel 4-8 */ | ||
1742 | else if (i < 8) | ||
1743 | index = 1; | ||
1744 | /* Channel 9-14 */ | ||
1745 | else | ||
1746 | index = 2; | ||
1747 | |||
1748 | /* Record A & B CCK /OFDM - 1T/2T Channel area | ||
1749 | * tx power */ | ||
1750 | rtlefuse->txpwrlevel_cck[rf_path][i] = | ||
1751 | rtlefuse->eeprom_chnlarea_txpwr_cck | ||
1752 | [rf_path][index]; | ||
1753 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i] = | ||
1754 | rtlefuse->eeprom_chnlarea_txpwr_ht40_1s | ||
1755 | [rf_path][index]; | ||
1756 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i] = | ||
1757 | rtlefuse->eeprom_chnlarea_txpwr_ht40_2sdiif | ||
1758 | [rf_path][index]; | ||
1759 | } | ||
1760 | |||
1761 | for (i = 0; i < 14; i++) { | ||
1762 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1763 | ("RF(%d)-Ch(%d) [CCK / HT40_1S / HT40_2S] = " | ||
1764 | "[0x%x / 0x%x / 0x%x]\n", rf_path, i, | ||
1765 | rtlefuse->txpwrlevel_cck[rf_path][i], | ||
1766 | rtlefuse->txpwrlevel_ht40_1s[rf_path][i], | ||
1767 | rtlefuse->txpwrlevel_ht40_2s[rf_path][i])); | ||
1768 | } | ||
1769 | } | ||
1770 | |||
1771 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1772 | for (i = 0; i < 3; i++) { | ||
1773 | /* Read Power diff limit. */ | ||
1774 | rtlefuse->eeprom_pwrgroup[rf_path][i] = | ||
1775 | hwinfo[EEPROM_TXPWRGROUP + rf_path * 3 + i]; | ||
1776 | } | ||
1777 | } | ||
1778 | |||
1779 | for (rf_path = 0; rf_path < 2; rf_path++) { | ||
1780 | /* Fill Pwr group */ | ||
1781 | for (i = 0; i < 14; i++) { | ||
1782 | /* Chanel 1-3 */ | ||
1783 | if (i < 3) | ||
1784 | index = 0; | ||
1785 | /* Channel 4-8 */ | ||
1786 | else if (i < 8) | ||
1787 | index = 1; | ||
1788 | /* Channel 9-13 */ | ||
1789 | else | ||
1790 | index = 2; | ||
1791 | |||
1792 | rtlefuse->pwrgroup_ht20[rf_path][i] = | ||
1793 | (rtlefuse->eeprom_pwrgroup[rf_path][index] & | ||
1794 | 0xf); | ||
1795 | rtlefuse->pwrgroup_ht40[rf_path][i] = | ||
1796 | ((rtlefuse->eeprom_pwrgroup[rf_path][index] & | ||
1797 | 0xf0) >> 4); | ||
1798 | |||
1799 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1800 | ("RF-%d pwrgroup_ht20[%d] = 0x%x\n", | ||
1801 | rf_path, i, | ||
1802 | rtlefuse->pwrgroup_ht20[rf_path][i])); | ||
1803 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1804 | ("RF-%d pwrgroup_ht40[%d] = 0x%x\n", | ||
1805 | rf_path, i, | ||
1806 | rtlefuse->pwrgroup_ht40[rf_path][i])); | ||
1807 | } | ||
1808 | } | ||
1809 | |||
1810 | for (i = 0; i < 14; i++) { | ||
1811 | /* Read tx power difference between HT OFDM 20/40 MHZ */ | ||
1812 | /* channel 1-3 */ | ||
1813 | if (i < 3) | ||
1814 | index = 0; | ||
1815 | /* Channel 4-8 */ | ||
1816 | else if (i < 8) | ||
1817 | index = 1; | ||
1818 | /* Channel 9-14 */ | ||
1819 | else | ||
1820 | index = 2; | ||
1821 | |||
1822 | tempval = (*(u8 *)&hwinfo[EEPROM_TX_PWR_HT20_DIFF + | ||
1823 | index]) & 0xff; | ||
1824 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i] = (tempval & 0xF); | ||
1825 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i] = | ||
1826 | ((tempval >> 4) & 0xF); | ||
1827 | |||
1828 | /* Read OFDM<->HT tx power diff */ | ||
1829 | /* Channel 1-3 */ | ||
1830 | if (i < 3) | ||
1831 | index = 0; | ||
1832 | /* Channel 4-8 */ | ||
1833 | else if (i < 8) | ||
1834 | index = 0x11; | ||
1835 | /* Channel 9-14 */ | ||
1836 | else | ||
1837 | index = 1; | ||
1838 | |||
1839 | tempval = (*(u8 *)&hwinfo[EEPROM_TX_PWR_OFDM_DIFF + index]) | ||
1840 | & 0xff; | ||
1841 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i] = | ||
1842 | (tempval & 0xF); | ||
1843 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i] = | ||
1844 | ((tempval >> 4) & 0xF); | ||
1845 | |||
1846 | tempval = (*(u8 *)&hwinfo[TX_PWR_SAFETY_CHK]); | ||
1847 | rtlefuse->txpwr_safetyflag = (tempval & 0x01); | ||
1848 | } | ||
1849 | |||
1850 | rtlefuse->eeprom_regulatory = 0; | ||
1851 | if (rtlefuse->eeprom_version >= 2) { | ||
1852 | /* BIT(0)~2 */ | ||
1853 | if (rtlefuse->eeprom_version >= 4) | ||
1854 | rtlefuse->eeprom_regulatory = | ||
1855 | (hwinfo[EEPROM_REGULATORY] & 0x7); | ||
1856 | else /* BIT(0) */ | ||
1857 | rtlefuse->eeprom_regulatory = | ||
1858 | (hwinfo[EEPROM_REGULATORY] & 0x1); | ||
1859 | } | ||
1860 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1861 | ("eeprom_regulatory = 0x%x\n", rtlefuse->eeprom_regulatory)); | ||
1862 | |||
1863 | for (i = 0; i < 14; i++) | ||
1864 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1865 | ("RF-A Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
1866 | rtlefuse->txpwr_ht20diff[RF90_PATH_A][i])); | ||
1867 | for (i = 0; i < 14; i++) | ||
1868 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1869 | ("RF-A Legacy to Ht40 Diff[%d] = 0x%x\n", i, | ||
1870 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][i])); | ||
1871 | for (i = 0; i < 14; i++) | ||
1872 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1873 | ("RF-B Ht20 to HT40 Diff[%d] = 0x%x\n", i, | ||
1874 | rtlefuse->txpwr_ht20diff[RF90_PATH_B][i])); | ||
1875 | for (i = 0; i < 14; i++) | ||
1876 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, | ||
1877 | ("RF-B Legacy to HT40 Diff[%d] = 0x%x\n", i, | ||
1878 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_B][i])); | ||
1879 | |||
1880 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, ("TxPwrSafetyFlag = %d\n", | ||
1881 | rtlefuse->txpwr_safetyflag)); | ||
1882 | |||
1883 | /* Read RF-indication and Tx Power gain | ||
1884 | * index diff of legacy to HT OFDM rate. */ | ||
1885 | tempval = (*(u8 *)&hwinfo[EEPROM_RFIND_POWERDIFF]) & 0xff; | ||
1886 | rtlefuse->eeprom_txpowerdiff = tempval; | ||
1887 | rtlefuse->legacy_httxpowerdiff = | ||
1888 | rtlefuse->txpwr_legacyhtdiff[RF90_PATH_A][0]; | ||
1889 | |||
1890 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, ("TxPowerDiff = %#x\n", | ||
1891 | rtlefuse->eeprom_txpowerdiff)); | ||
1892 | |||
1893 | /* Get TSSI value for each path. */ | ||
1894 | usvalue = *(u16 *)&hwinfo[EEPROM_TSSI_A]; | ||
1895 | rtlefuse->eeprom_tssi[RF90_PATH_A] = (u8)((usvalue & 0xff00) >> 8); | ||
1896 | usvalue = *(u8 *)&hwinfo[EEPROM_TSSI_B]; | ||
1897 | rtlefuse->eeprom_tssi[RF90_PATH_B] = (u8)(usvalue & 0xff); | ||
1898 | |||
1899 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, ("TSSI_A = 0x%x, TSSI_B = 0x%x\n", | ||
1900 | rtlefuse->eeprom_tssi[RF90_PATH_A], | ||
1901 | rtlefuse->eeprom_tssi[RF90_PATH_B])); | ||
1902 | |||
1903 | /* Read antenna tx power offset of B/C/D to A from EEPROM */ | ||
1904 | /* and read ThermalMeter from EEPROM */ | ||
1905 | tempval = *(u8 *)&hwinfo[EEPROM_THERMALMETER]; | ||
1906 | rtlefuse->eeprom_thermalmeter = tempval; | ||
1907 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, ("thermalmeter = 0x%x\n", | ||
1908 | rtlefuse->eeprom_thermalmeter)); | ||
1909 | |||
1910 | /* ThermalMeter, BIT(0)~3 for RFIC1, BIT(4)~7 for RFIC2 */ | ||
1911 | rtlefuse->thermalmeter[0] = (rtlefuse->eeprom_thermalmeter & 0x1f); | ||
1912 | rtlefuse->tssi_13dbm = rtlefuse->eeprom_thermalmeter * 100; | ||
1913 | |||
1914 | /* Read CrystalCap from EEPROM */ | ||
1915 | tempval = (*(u8 *)&hwinfo[EEPROM_CRYSTALCAP]) >> 4; | ||
1916 | rtlefuse->eeprom_crystalcap = tempval; | ||
1917 | /* CrystalCap, BIT(12)~15 */ | ||
1918 | rtlefuse->crystalcap = rtlefuse->eeprom_crystalcap; | ||
1919 | |||
1920 | /* Read IC Version && Channel Plan */ | ||
1921 | /* Version ID, Channel plan */ | ||
1922 | rtlefuse->eeprom_channelplan = *(u8 *)&hwinfo[EEPROM_CHANNELPLAN]; | ||
1923 | rtlefuse->txpwr_fromeprom = true; | ||
1924 | RTPRINT(rtlpriv, FINIT, INIT_TxPower, ("EEPROM ChannelPlan = 0x%4x\n", | ||
1925 | rtlefuse->eeprom_channelplan)); | ||
1926 | |||
1927 | /* Read Customer ID or Board Type!!! */ | ||
1928 | tempval = *(u8 *)&hwinfo[EEPROM_BOARDTYPE]; | ||
1929 | /* Change RF type definition */ | ||
1930 | if (tempval == 0) | ||
1931 | rtlphy->rf_type = RF_2T2R; | ||
1932 | else if (tempval == 1) | ||
1933 | rtlphy->rf_type = RF_1T2R; | ||
1934 | else if (tempval == 2) | ||
1935 | rtlphy->rf_type = RF_1T2R; | ||
1936 | else if (tempval == 3) | ||
1937 | rtlphy->rf_type = RF_1T1R; | ||
1938 | |||
1939 | /* 1T2R but 1SS (1x1 receive combining) */ | ||
1940 | rtlefuse->b1x1_recvcombine = false; | ||
1941 | if (rtlphy->rf_type == RF_1T2R) { | ||
1942 | tempval = rtl_read_byte(rtlpriv, 0x07); | ||
1943 | if (!(tempval & BIT(0))) { | ||
1944 | rtlefuse->b1x1_recvcombine = true; | ||
1945 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1946 | ("RF_TYPE=1T2R but only 1SS\n")); | ||
1947 | } | ||
1948 | } | ||
1949 | rtlefuse->b1ss_support = rtlefuse->b1x1_recvcombine; | ||
1950 | rtlefuse->eeprom_oemid = *(u8 *)&hwinfo[EEPROM_CUSTOMID]; | ||
1951 | |||
1952 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("EEPROM Customer ID: 0x%2x", | ||
1953 | rtlefuse->eeprom_oemid)); | ||
1954 | |||
1955 | /* set channel paln to world wide 13 */ | ||
1956 | rtlefuse->channel_plan = COUNTRY_CODE_WORLD_WIDE_13; | ||
1957 | } | ||
1958 | |||
1959 | void rtl92se_read_eeprom_info(struct ieee80211_hw *hw) | ||
1960 | { | ||
1961 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1962 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1963 | u8 tmp_u1b = 0; | ||
1964 | |||
1965 | tmp_u1b = rtl_read_byte(rtlpriv, EPROM_CMD); | ||
1966 | |||
1967 | if (tmp_u1b & BIT(4)) { | ||
1968 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Boot from EEPROM\n")); | ||
1969 | rtlefuse->epromtype = EEPROM_93C46; | ||
1970 | } else { | ||
1971 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("Boot from EFUSE\n")); | ||
1972 | rtlefuse->epromtype = EEPROM_BOOT_EFUSE; | ||
1973 | } | ||
1974 | |||
1975 | if (tmp_u1b & BIT(5)) { | ||
1976 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Autoload OK\n")); | ||
1977 | rtlefuse->autoload_failflag = false; | ||
1978 | _rtl92se_read_adapter_info(hw); | ||
1979 | } else { | ||
1980 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("Autoload ERR!!\n")); | ||
1981 | rtlefuse->autoload_failflag = true; | ||
1982 | } | ||
1983 | } | ||
1984 | |||
1985 | static void rtl92se_update_hal_rate_table(struct ieee80211_hw *hw, | ||
1986 | struct ieee80211_sta *sta) | ||
1987 | { | ||
1988 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1989 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1990 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
1991 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1992 | u32 ratr_value; | ||
1993 | u8 ratr_index = 0; | ||
1994 | u8 nmode = mac->ht_enable; | ||
1995 | u8 mimo_ps = IEEE80211_SMPS_OFF; | ||
1996 | u16 shortgi_rate = 0; | ||
1997 | u32 tmp_ratr_value = 0; | ||
1998 | u8 curtxbw_40mhz = mac->bw_40; | ||
1999 | u8 curshortgi_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40) ? | ||
2000 | 1 : 0; | ||
2001 | u8 curshortgi_20mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20) ? | ||
2002 | 1 : 0; | ||
2003 | enum wireless_mode wirelessmode = mac->mode; | ||
2004 | |||
2005 | if (rtlhal->current_bandtype == BAND_ON_5G) | ||
2006 | ratr_value = sta->supp_rates[1] << 4; | ||
2007 | else | ||
2008 | ratr_value = sta->supp_rates[0]; | ||
2009 | ratr_value |= (sta->ht_cap.mcs.rx_mask[1] << 20 | | ||
2010 | sta->ht_cap.mcs.rx_mask[0] << 12); | ||
2011 | switch (wirelessmode) { | ||
2012 | case WIRELESS_MODE_B: | ||
2013 | ratr_value &= 0x0000000D; | ||
2014 | break; | ||
2015 | case WIRELESS_MODE_G: | ||
2016 | ratr_value &= 0x00000FF5; | ||
2017 | break; | ||
2018 | case WIRELESS_MODE_N_24G: | ||
2019 | case WIRELESS_MODE_N_5G: | ||
2020 | nmode = 1; | ||
2021 | if (mimo_ps == IEEE80211_SMPS_STATIC) { | ||
2022 | ratr_value &= 0x0007F005; | ||
2023 | } else { | ||
2024 | u32 ratr_mask; | ||
2025 | |||
2026 | if (get_rf_type(rtlphy) == RF_1T2R || | ||
2027 | get_rf_type(rtlphy) == RF_1T1R) { | ||
2028 | if (curtxbw_40mhz) | ||
2029 | ratr_mask = 0x000ff015; | ||
2030 | else | ||
2031 | ratr_mask = 0x000ff005; | ||
2032 | } else { | ||
2033 | if (curtxbw_40mhz) | ||
2034 | ratr_mask = 0x0f0ff015; | ||
2035 | else | ||
2036 | ratr_mask = 0x0f0ff005; | ||
2037 | } | ||
2038 | |||
2039 | ratr_value &= ratr_mask; | ||
2040 | } | ||
2041 | break; | ||
2042 | default: | ||
2043 | if (rtlphy->rf_type == RF_1T2R) | ||
2044 | ratr_value &= 0x000ff0ff; | ||
2045 | else | ||
2046 | ratr_value &= 0x0f0ff0ff; | ||
2047 | |||
2048 | break; | ||
2049 | } | ||
2050 | |||
2051 | if (rtlpriv->rtlhal.version >= VERSION_8192S_BCUT) | ||
2052 | ratr_value &= 0x0FFFFFFF; | ||
2053 | else if (rtlpriv->rtlhal.version == VERSION_8192S_ACUT) | ||
2054 | ratr_value &= 0x0FFFFFF0; | ||
2055 | |||
2056 | if (nmode && ((curtxbw_40mhz && | ||
2057 | curshortgi_40mhz) || (!curtxbw_40mhz && | ||
2058 | curshortgi_20mhz))) { | ||
2059 | |||
2060 | ratr_value |= 0x10000000; | ||
2061 | tmp_ratr_value = (ratr_value >> 12); | ||
2062 | |||
2063 | for (shortgi_rate = 15; shortgi_rate > 0; shortgi_rate--) { | ||
2064 | if ((1 << shortgi_rate) & tmp_ratr_value) | ||
2065 | break; | ||
2066 | } | ||
2067 | |||
2068 | shortgi_rate = (shortgi_rate << 12) | (shortgi_rate << 8) | | ||
2069 | (shortgi_rate << 4) | (shortgi_rate); | ||
2070 | |||
2071 | rtl_write_byte(rtlpriv, SG_RATE, shortgi_rate); | ||
2072 | } | ||
2073 | |||
2074 | rtl_write_dword(rtlpriv, ARFR0 + ratr_index * 4, ratr_value); | ||
2075 | if (ratr_value & 0xfffff000) | ||
2076 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_RA_REFRESH_N); | ||
2077 | else | ||
2078 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_RA_REFRESH_BG); | ||
2079 | |||
2080 | RT_TRACE(rtlpriv, COMP_RATR, DBG_DMESG, | ||
2081 | ("%x\n", rtl_read_dword(rtlpriv, ARFR0))); | ||
2082 | } | ||
2083 | |||
2084 | static void rtl92se_update_hal_rate_mask(struct ieee80211_hw *hw, | ||
2085 | struct ieee80211_sta *sta, | ||
2086 | u8 rssi_level) | ||
2087 | { | ||
2088 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2089 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
2090 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2091 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
2092 | struct rtl_sta_info *sta_entry = NULL; | ||
2093 | u32 ratr_bitmap; | ||
2094 | u8 ratr_index = 0; | ||
2095 | u8 curtxbw_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SUP_WIDTH_20_40) | ||
2096 | ? 1 : 0; | ||
2097 | u8 curshortgi_40mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_40) ? | ||
2098 | 1 : 0; | ||
2099 | u8 curshortgi_20mhz = (sta->ht_cap.cap & IEEE80211_HT_CAP_SGI_20) ? | ||
2100 | 1 : 0; | ||
2101 | enum wireless_mode wirelessmode = 0; | ||
2102 | bool shortgi = false; | ||
2103 | u32 ratr_value = 0; | ||
2104 | u8 shortgi_rate = 0; | ||
2105 | u32 mask = 0; | ||
2106 | u32 band = 0; | ||
2107 | bool bmulticast = false; | ||
2108 | u8 macid = 0; | ||
2109 | u8 mimo_ps = IEEE80211_SMPS_OFF; | ||
2110 | |||
2111 | sta_entry = (struct rtl_sta_info *) sta->drv_priv; | ||
2112 | wirelessmode = sta_entry->wireless_mode; | ||
2113 | if (mac->opmode == NL80211_IFTYPE_STATION) | ||
2114 | curtxbw_40mhz = mac->bw_40; | ||
2115 | else if (mac->opmode == NL80211_IFTYPE_AP || | ||
2116 | mac->opmode == NL80211_IFTYPE_ADHOC) | ||
2117 | macid = sta->aid + 1; | ||
2118 | |||
2119 | if (rtlhal->current_bandtype == BAND_ON_5G) | ||
2120 | ratr_bitmap = sta->supp_rates[1] << 4; | ||
2121 | else | ||
2122 | ratr_bitmap = sta->supp_rates[0]; | ||
2123 | ratr_bitmap |= (sta->ht_cap.mcs.rx_mask[1] << 20 | | ||
2124 | sta->ht_cap.mcs.rx_mask[0] << 12); | ||
2125 | switch (wirelessmode) { | ||
2126 | case WIRELESS_MODE_B: | ||
2127 | band |= WIRELESS_11B; | ||
2128 | ratr_index = RATR_INX_WIRELESS_B; | ||
2129 | if (ratr_bitmap & 0x0000000c) | ||
2130 | ratr_bitmap &= 0x0000000d; | ||
2131 | else | ||
2132 | ratr_bitmap &= 0x0000000f; | ||
2133 | break; | ||
2134 | case WIRELESS_MODE_G: | ||
2135 | band |= (WIRELESS_11G | WIRELESS_11B); | ||
2136 | ratr_index = RATR_INX_WIRELESS_GB; | ||
2137 | |||
2138 | if (rssi_level == 1) | ||
2139 | ratr_bitmap &= 0x00000f00; | ||
2140 | else if (rssi_level == 2) | ||
2141 | ratr_bitmap &= 0x00000ff0; | ||
2142 | else | ||
2143 | ratr_bitmap &= 0x00000ff5; | ||
2144 | break; | ||
2145 | case WIRELESS_MODE_A: | ||
2146 | band |= WIRELESS_11A; | ||
2147 | ratr_index = RATR_INX_WIRELESS_A; | ||
2148 | ratr_bitmap &= 0x00000ff0; | ||
2149 | break; | ||
2150 | case WIRELESS_MODE_N_24G: | ||
2151 | case WIRELESS_MODE_N_5G: | ||
2152 | band |= (WIRELESS_11N | WIRELESS_11G | WIRELESS_11B); | ||
2153 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
2154 | |||
2155 | if (mimo_ps == IEEE80211_SMPS_STATIC) { | ||
2156 | if (rssi_level == 1) | ||
2157 | ratr_bitmap &= 0x00070000; | ||
2158 | else if (rssi_level == 2) | ||
2159 | ratr_bitmap &= 0x0007f000; | ||
2160 | else | ||
2161 | ratr_bitmap &= 0x0007f005; | ||
2162 | } else { | ||
2163 | if (rtlphy->rf_type == RF_1T2R || | ||
2164 | rtlphy->rf_type == RF_1T1R) { | ||
2165 | if (rssi_level == 1) { | ||
2166 | ratr_bitmap &= 0x000f0000; | ||
2167 | } else if (rssi_level == 3) { | ||
2168 | ratr_bitmap &= 0x000fc000; | ||
2169 | } else if (rssi_level == 5) { | ||
2170 | ratr_bitmap &= 0x000ff000; | ||
2171 | } else { | ||
2172 | if (curtxbw_40mhz) | ||
2173 | ratr_bitmap &= 0x000ff015; | ||
2174 | else | ||
2175 | ratr_bitmap &= 0x000ff005; | ||
2176 | } | ||
2177 | } else { | ||
2178 | if (rssi_level == 1) { | ||
2179 | ratr_bitmap &= 0x0f8f0000; | ||
2180 | } else if (rssi_level == 3) { | ||
2181 | ratr_bitmap &= 0x0f8fc000; | ||
2182 | } else if (rssi_level == 5) { | ||
2183 | ratr_bitmap &= 0x0f8ff000; | ||
2184 | } else { | ||
2185 | if (curtxbw_40mhz) | ||
2186 | ratr_bitmap &= 0x0f8ff015; | ||
2187 | else | ||
2188 | ratr_bitmap &= 0x0f8ff005; | ||
2189 | } | ||
2190 | } | ||
2191 | } | ||
2192 | |||
2193 | if ((curtxbw_40mhz && curshortgi_40mhz) || | ||
2194 | (!curtxbw_40mhz && curshortgi_20mhz)) { | ||
2195 | if (macid == 0) | ||
2196 | shortgi = true; | ||
2197 | else if (macid == 1) | ||
2198 | shortgi = false; | ||
2199 | } | ||
2200 | break; | ||
2201 | default: | ||
2202 | band |= (WIRELESS_11N | WIRELESS_11G | WIRELESS_11B); | ||
2203 | ratr_index = RATR_INX_WIRELESS_NGB; | ||
2204 | |||
2205 | if (rtlphy->rf_type == RF_1T2R) | ||
2206 | ratr_bitmap &= 0x000ff0ff; | ||
2207 | else | ||
2208 | ratr_bitmap &= 0x0f8ff0ff; | ||
2209 | break; | ||
2210 | } | ||
2211 | |||
2212 | if (rtlpriv->rtlhal.version >= VERSION_8192S_BCUT) | ||
2213 | ratr_bitmap &= 0x0FFFFFFF; | ||
2214 | else if (rtlpriv->rtlhal.version == VERSION_8192S_ACUT) | ||
2215 | ratr_bitmap &= 0x0FFFFFF0; | ||
2216 | |||
2217 | if (shortgi) { | ||
2218 | ratr_bitmap |= 0x10000000; | ||
2219 | /* Get MAX MCS available. */ | ||
2220 | ratr_value = (ratr_bitmap >> 12); | ||
2221 | for (shortgi_rate = 15; shortgi_rate > 0; shortgi_rate--) { | ||
2222 | if ((1 << shortgi_rate) & ratr_value) | ||
2223 | break; | ||
2224 | } | ||
2225 | |||
2226 | shortgi_rate = (shortgi_rate << 12) | (shortgi_rate << 8) | | ||
2227 | (shortgi_rate << 4) | (shortgi_rate); | ||
2228 | rtl_write_byte(rtlpriv, SG_RATE, shortgi_rate); | ||
2229 | } | ||
2230 | |||
2231 | mask |= (bmulticast ? 1 : 0) << 9 | (macid & 0x1f) << 4 | (band & 0xf); | ||
2232 | |||
2233 | RT_TRACE(rtlpriv, COMP_RATR, DBG_TRACE, ("mask = %x, bitmap = %x\n", | ||
2234 | mask, ratr_bitmap)); | ||
2235 | rtl_write_dword(rtlpriv, 0x2c4, ratr_bitmap); | ||
2236 | rtl_write_dword(rtlpriv, WFM5, (FW_RA_UPDATE_MASK | (mask << 8))); | ||
2237 | |||
2238 | if (macid != 0) | ||
2239 | sta_entry->ratr_index = ratr_index; | ||
2240 | } | ||
2241 | |||
2242 | void rtl92se_update_hal_rate_tbl(struct ieee80211_hw *hw, | ||
2243 | struct ieee80211_sta *sta, u8 rssi_level) | ||
2244 | { | ||
2245 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2246 | |||
2247 | if (rtlpriv->dm.useramask) | ||
2248 | rtl92se_update_hal_rate_mask(hw, sta, rssi_level); | ||
2249 | else | ||
2250 | rtl92se_update_hal_rate_table(hw, sta); | ||
2251 | } | ||
2252 | |||
2253 | void rtl92se_update_channel_access_setting(struct ieee80211_hw *hw) | ||
2254 | { | ||
2255 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2256 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2257 | u16 sifs_timer; | ||
2258 | |||
2259 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SLOT_TIME, | ||
2260 | (u8 *)&mac->slot_time); | ||
2261 | sifs_timer = 0x0e0e; | ||
2262 | rtlpriv->cfg->ops->set_hw_reg(hw, HW_VAR_SIFS, (u8 *)&sifs_timer); | ||
2263 | |||
2264 | } | ||
2265 | |||
2266 | /* this ifunction is for RFKILL, it's different with windows, | ||
2267 | * because UI will disable wireless when GPIO Radio Off. | ||
2268 | * And here we not check or Disable/Enable ASPM like windows*/ | ||
2269 | bool rtl92se_gpio_radio_on_off_checking(struct ieee80211_hw *hw, u8 *valid) | ||
2270 | { | ||
2271 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2272 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
2273 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
2274 | enum rf_pwrstate rfpwr_toset, cur_rfstate; | ||
2275 | unsigned long flag = 0; | ||
2276 | bool actuallyset = false; | ||
2277 | bool turnonbypowerdomain = false; | ||
2278 | |||
2279 | /* just 8191se can check gpio before firstup, 92c/92d have fixed it */ | ||
2280 | if ((rtlpci->up_first_time == 1) || (rtlpci->being_init_adapter)) | ||
2281 | return false; | ||
2282 | |||
2283 | if (ppsc->swrf_processing) | ||
2284 | return false; | ||
2285 | |||
2286 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2287 | if (ppsc->rfchange_inprogress) { | ||
2288 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2289 | return false; | ||
2290 | } else { | ||
2291 | ppsc->rfchange_inprogress = true; | ||
2292 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2293 | } | ||
2294 | |||
2295 | cur_rfstate = ppsc->rfpwr_state; | ||
2296 | |||
2297 | /* because after _rtl92s_phy_set_rfhalt, all power | ||
2298 | * closed, so we must open some power for GPIO check, | ||
2299 | * or we will always check GPIO RFOFF here, | ||
2300 | * And we should close power after GPIO check */ | ||
2301 | if (RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC)) { | ||
2302 | _rtl92se_power_domain_init(hw); | ||
2303 | turnonbypowerdomain = true; | ||
2304 | } | ||
2305 | |||
2306 | rfpwr_toset = _rtl92se_rf_onoff_detect(hw); | ||
2307 | |||
2308 | if ((ppsc->hwradiooff == true) && (rfpwr_toset == ERFON)) { | ||
2309 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
2310 | ("RFKILL-HW Radio ON, RF ON\n")); | ||
2311 | |||
2312 | rfpwr_toset = ERFON; | ||
2313 | ppsc->hwradiooff = false; | ||
2314 | actuallyset = true; | ||
2315 | } else if ((ppsc->hwradiooff == false) && (rfpwr_toset == ERFOFF)) { | ||
2316 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
2317 | ("RFKILL-HW Radio OFF, RF OFF\n")); | ||
2318 | |||
2319 | rfpwr_toset = ERFOFF; | ||
2320 | ppsc->hwradiooff = true; | ||
2321 | actuallyset = true; | ||
2322 | } | ||
2323 | |||
2324 | if (actuallyset) { | ||
2325 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2326 | ppsc->rfchange_inprogress = false; | ||
2327 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2328 | |||
2329 | /* this not include ifconfig wlan0 down case */ | ||
2330 | /* } else if (rfpwr_toset == ERFOFF || cur_rfstate == ERFOFF) { */ | ||
2331 | } else { | ||
2332 | /* because power_domain_init may be happen when | ||
2333 | * _rtl92s_phy_set_rfhalt, this will open some powers | ||
2334 | * and cause current increasing about 40 mA for ips, | ||
2335 | * rfoff and ifconfig down, so we set | ||
2336 | * _rtl92s_phy_set_rfhalt again here */ | ||
2337 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC && | ||
2338 | turnonbypowerdomain) { | ||
2339 | _rtl92s_phy_set_rfhalt(hw); | ||
2340 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
2341 | } | ||
2342 | |||
2343 | spin_lock_irqsave(&rtlpriv->locks.rf_ps_lock, flag); | ||
2344 | ppsc->rfchange_inprogress = false; | ||
2345 | spin_unlock_irqrestore(&rtlpriv->locks.rf_ps_lock, flag); | ||
2346 | } | ||
2347 | |||
2348 | *valid = 1; | ||
2349 | return !ppsc->hwradiooff; | ||
2350 | |||
2351 | } | ||
2352 | |||
2353 | /* Is_wepkey just used for WEP used as group & pairwise key | ||
2354 | * if pairwise is AES ang group is WEP Is_wepkey == false.*/ | ||
2355 | void rtl92se_set_key(struct ieee80211_hw *hw, u32 key_index, u8 *p_macaddr, | ||
2356 | bool is_group, u8 enc_algo, bool is_wepkey, bool clear_all) | ||
2357 | { | ||
2358 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
2359 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
2360 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
2361 | u8 *macaddr = p_macaddr; | ||
2362 | |||
2363 | u32 entry_id = 0; | ||
2364 | bool is_pairwise = false; | ||
2365 | |||
2366 | static u8 cam_const_addr[4][6] = { | ||
2367 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, | ||
2368 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x01}, | ||
2369 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x02}, | ||
2370 | {0x00, 0x00, 0x00, 0x00, 0x00, 0x03} | ||
2371 | }; | ||
2372 | static u8 cam_const_broad[] = { | ||
2373 | 0xff, 0xff, 0xff, 0xff, 0xff, 0xff | ||
2374 | }; | ||
2375 | |||
2376 | if (clear_all) { | ||
2377 | u8 idx = 0; | ||
2378 | u8 cam_offset = 0; | ||
2379 | u8 clear_number = 5; | ||
2380 | |||
2381 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, ("clear_all\n")); | ||
2382 | |||
2383 | for (idx = 0; idx < clear_number; idx++) { | ||
2384 | rtl_cam_mark_invalid(hw, cam_offset + idx); | ||
2385 | rtl_cam_empty_entry(hw, cam_offset + idx); | ||
2386 | |||
2387 | if (idx < 5) { | ||
2388 | memset(rtlpriv->sec.key_buf[idx], 0, | ||
2389 | MAX_KEY_LEN); | ||
2390 | rtlpriv->sec.key_len[idx] = 0; | ||
2391 | } | ||
2392 | } | ||
2393 | |||
2394 | } else { | ||
2395 | switch (enc_algo) { | ||
2396 | case WEP40_ENCRYPTION: | ||
2397 | enc_algo = CAM_WEP40; | ||
2398 | break; | ||
2399 | case WEP104_ENCRYPTION: | ||
2400 | enc_algo = CAM_WEP104; | ||
2401 | break; | ||
2402 | case TKIP_ENCRYPTION: | ||
2403 | enc_algo = CAM_TKIP; | ||
2404 | break; | ||
2405 | case AESCCMP_ENCRYPTION: | ||
2406 | enc_algo = CAM_AES; | ||
2407 | break; | ||
2408 | default: | ||
2409 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
2410 | ("switch case not process\n")); | ||
2411 | enc_algo = CAM_TKIP; | ||
2412 | break; | ||
2413 | } | ||
2414 | |||
2415 | if (is_wepkey || rtlpriv->sec.use_defaultkey) { | ||
2416 | macaddr = cam_const_addr[key_index]; | ||
2417 | entry_id = key_index; | ||
2418 | } else { | ||
2419 | if (is_group) { | ||
2420 | macaddr = cam_const_broad; | ||
2421 | entry_id = key_index; | ||
2422 | } else { | ||
2423 | if (mac->opmode == NL80211_IFTYPE_AP) { | ||
2424 | entry_id = rtl_cam_get_free_entry(hw, | ||
2425 | p_macaddr); | ||
2426 | if (entry_id >= TOTAL_CAM_ENTRY) { | ||
2427 | RT_TRACE(rtlpriv, | ||
2428 | COMP_SEC, DBG_EMERG, | ||
2429 | ("Can not find free hw" | ||
2430 | " security cam entry\n")); | ||
2431 | return; | ||
2432 | } | ||
2433 | } else { | ||
2434 | entry_id = CAM_PAIRWISE_KEY_POSITION; | ||
2435 | } | ||
2436 | |||
2437 | key_index = PAIRWISE_KEYIDX; | ||
2438 | is_pairwise = true; | ||
2439 | } | ||
2440 | } | ||
2441 | |||
2442 | if (rtlpriv->sec.key_len[key_index] == 0) { | ||
2443 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2444 | ("delete one entry, entry_id is %d\n", | ||
2445 | entry_id)); | ||
2446 | if (mac->opmode == NL80211_IFTYPE_AP) | ||
2447 | rtl_cam_del_entry(hw, p_macaddr); | ||
2448 | rtl_cam_delete_one_entry(hw, p_macaddr, entry_id); | ||
2449 | } else { | ||
2450 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2451 | ("The insert KEY length is %d\n", | ||
2452 | rtlpriv->sec.key_len[PAIRWISE_KEYIDX])); | ||
2453 | RT_TRACE(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2454 | ("The insert KEY is %x %x\n", | ||
2455 | rtlpriv->sec.key_buf[0][0], | ||
2456 | rtlpriv->sec.key_buf[0][1])); | ||
2457 | |||
2458 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2459 | ("add one entry\n")); | ||
2460 | if (is_pairwise) { | ||
2461 | RT_PRINT_DATA(rtlpriv, COMP_SEC, DBG_LOUD, | ||
2462 | "Pairwiase Key content :", | ||
2463 | rtlpriv->sec.pairwise_key, | ||
2464 | rtlpriv->sec.key_len[PAIRWISE_KEYIDX]); | ||
2465 | |||
2466 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2467 | ("set Pairwiase key\n")); | ||
2468 | |||
2469 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
2470 | entry_id, enc_algo, | ||
2471 | CAM_CONFIG_NO_USEDK, | ||
2472 | rtlpriv->sec.key_buf[key_index]); | ||
2473 | } else { | ||
2474 | RT_TRACE(rtlpriv, COMP_SEC, DBG_DMESG, | ||
2475 | ("set group key\n")); | ||
2476 | |||
2477 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
2478 | rtl_cam_add_one_entry(hw, | ||
2479 | rtlefuse->dev_addr, | ||
2480 | PAIRWISE_KEYIDX, | ||
2481 | CAM_PAIRWISE_KEY_POSITION, | ||
2482 | enc_algo, CAM_CONFIG_NO_USEDK, | ||
2483 | rtlpriv->sec.key_buf[entry_id]); | ||
2484 | } | ||
2485 | |||
2486 | rtl_cam_add_one_entry(hw, macaddr, key_index, | ||
2487 | entry_id, enc_algo, | ||
2488 | CAM_CONFIG_NO_USEDK, | ||
2489 | rtlpriv->sec.key_buf[entry_id]); | ||
2490 | } | ||
2491 | |||
2492 | } | ||
2493 | } | ||
2494 | } | ||
2495 | |||
2496 | void rtl92se_suspend(struct ieee80211_hw *hw) | ||
2497 | { | ||
2498 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
2499 | |||
2500 | rtlpci->up_first_time = true; | ||
2501 | } | ||
2502 | |||
2503 | void rtl92se_resume(struct ieee80211_hw *hw) | ||
2504 | { | ||
2505 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
2506 | u32 val; | ||
2507 | |||
2508 | pci_read_config_dword(rtlpci->pdev, 0x40, &val); | ||
2509 | if ((val & 0x0000ff00) != 0) | ||
2510 | pci_write_config_dword(rtlpci->pdev, 0x40, | ||
2511 | val & 0xffff00ff); | ||
2512 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/hw.h b/drivers/net/wireless/rtlwifi/rtl8192se/hw.h new file mode 100644 index 000000000000..6160a9bfe98a --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/hw.h | |||
@@ -0,0 +1,79 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_PCI92SE_HW_H__ | ||
30 | #define __REALTEK_PCI92SE_HW_H__ | ||
31 | |||
32 | #define MSR_LINK_MANAGED 2 | ||
33 | #define MSR_LINK_NONE 0 | ||
34 | #define MSR_LINK_SHIFT 0 | ||
35 | #define MSR_LINK_ADHOC 1 | ||
36 | #define MSR_LINK_MASTER 3 | ||
37 | |||
38 | enum WIRELESS_NETWORK_TYPE { | ||
39 | WIRELESS_11B = 1, | ||
40 | WIRELESS_11G = 2, | ||
41 | WIRELESS_11A = 4, | ||
42 | WIRELESS_11N = 8 | ||
43 | }; | ||
44 | |||
45 | void rtl92se_get_hw_reg(struct ieee80211_hw *hw, | ||
46 | u8 variable, u8 *val); | ||
47 | void rtl92se_read_eeprom_info(struct ieee80211_hw *hw); | ||
48 | void rtl92se_interrupt_recognized(struct ieee80211_hw *hw, | ||
49 | u32 *inta, u32 *intb); | ||
50 | int rtl92se_hw_init(struct ieee80211_hw *hw); | ||
51 | void rtl92se_card_disable(struct ieee80211_hw *hw); | ||
52 | void rtl92se_enable_interrupt(struct ieee80211_hw *hw); | ||
53 | void rtl92se_disable_interrupt(struct ieee80211_hw *hw); | ||
54 | int rtl92se_set_network_type(struct ieee80211_hw *hw, | ||
55 | enum nl80211_iftype type); | ||
56 | void rtl92se_set_check_bssid(struct ieee80211_hw *hw, bool check_bssid); | ||
57 | void rtl92se_set_mac_addr(struct rtl_io *io, const u8 * addr); | ||
58 | void rtl92se_set_qos(struct ieee80211_hw *hw, int aci); | ||
59 | void rtl92se_set_beacon_related_registers(struct ieee80211_hw *hw); | ||
60 | void rtl92se_set_beacon_interval(struct ieee80211_hw *hw); | ||
61 | void rtl92se_update_interrupt_mask(struct ieee80211_hw *hw, | ||
62 | u32 add_msr, u32 rm_msr); | ||
63 | void rtl92se_set_hw_reg(struct ieee80211_hw *hw, u8 variable, | ||
64 | u8 *val); | ||
65 | void rtl92se_update_hal_rate_tbl(struct ieee80211_hw *hw, | ||
66 | struct ieee80211_sta *sta, u8 rssi_level); | ||
67 | void rtl92se_update_channel_access_setting(struct ieee80211_hw *hw); | ||
68 | bool rtl92se_gpio_radio_on_off_checking(struct ieee80211_hw *hw, | ||
69 | u8 *valid); | ||
70 | void rtl8192se_gpiobit3_cfg_inputmode(struct ieee80211_hw *hw); | ||
71 | void rtl92se_enable_hw_security_config(struct ieee80211_hw *hw); | ||
72 | void rtl92se_set_key(struct ieee80211_hw *hw, | ||
73 | u32 key_index, u8 *macaddr, bool is_group, | ||
74 | u8 enc_algo, bool is_wepkey, bool clear_all); | ||
75 | void rtl92se_suspend(struct ieee80211_hw *hw); | ||
76 | void rtl92se_resume(struct ieee80211_hw *hw); | ||
77 | |||
78 | #endif | ||
79 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/led.c b/drivers/net/wireless/rtlwifi/rtl8192se/led.c new file mode 100644 index 000000000000..6d4f66616680 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/led.c | |||
@@ -0,0 +1,149 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "reg.h" | ||
33 | #include "led.h" | ||
34 | |||
35 | static void _rtl92se_init_led(struct ieee80211_hw *hw, | ||
36 | struct rtl_led *pled, enum rtl_led_pin ledpin) | ||
37 | { | ||
38 | pled->hw = hw; | ||
39 | pled->ledpin = ledpin; | ||
40 | pled->ledon = false; | ||
41 | } | ||
42 | |||
43 | void rtl92se_init_sw_leds(struct ieee80211_hw *hw) | ||
44 | { | ||
45 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
46 | _rtl92se_init_led(hw, &(pcipriv->ledctl.sw_led0), LED_PIN_LED0); | ||
47 | _rtl92se_init_led(hw, &(pcipriv->ledctl.sw_led1), LED_PIN_LED1); | ||
48 | } | ||
49 | |||
50 | void rtl92se_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
51 | { | ||
52 | u8 ledcfg; | ||
53 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
54 | |||
55 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
56 | ("LedAddr:%X ledpin=%d\n", LEDCFG, pled->ledpin)); | ||
57 | |||
58 | ledcfg = rtl_read_byte(rtlpriv, LEDCFG); | ||
59 | |||
60 | switch (pled->ledpin) { | ||
61 | case LED_PIN_GPIO0: | ||
62 | break; | ||
63 | case LED_PIN_LED0: | ||
64 | rtl_write_byte(rtlpriv, LEDCFG, ledcfg & 0xf0); | ||
65 | break; | ||
66 | case LED_PIN_LED1: | ||
67 | rtl_write_byte(rtlpriv, LEDCFG, ledcfg & 0x0f); | ||
68 | break; | ||
69 | default: | ||
70 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
71 | ("switch case not process\n")); | ||
72 | break; | ||
73 | } | ||
74 | pled->ledon = true; | ||
75 | } | ||
76 | |||
77 | void rtl92se_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled) | ||
78 | { | ||
79 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
80 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
81 | u8 ledcfg; | ||
82 | |||
83 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, | ||
84 | ("LedAddr:%X ledpin=%d\n", LEDCFG, pled->ledpin)); | ||
85 | |||
86 | ledcfg = rtl_read_byte(rtlpriv, LEDCFG); | ||
87 | |||
88 | switch (pled->ledpin) { | ||
89 | case LED_PIN_GPIO0: | ||
90 | break; | ||
91 | case LED_PIN_LED0: | ||
92 | ledcfg &= 0xf0; | ||
93 | if (pcipriv->ledctl.led_opendrain == true) | ||
94 | rtl_write_byte(rtlpriv, LEDCFG, (ledcfg | BIT(1))); | ||
95 | else | ||
96 | rtl_write_byte(rtlpriv, LEDCFG, (ledcfg | BIT(3))); | ||
97 | break; | ||
98 | case LED_PIN_LED1: | ||
99 | ledcfg &= 0x0f; | ||
100 | rtl_write_byte(rtlpriv, LEDCFG, (ledcfg | BIT(3))); | ||
101 | break; | ||
102 | default: | ||
103 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
104 | ("switch case not process\n")); | ||
105 | break; | ||
106 | } | ||
107 | pled->ledon = false; | ||
108 | } | ||
109 | |||
110 | static void _rtl92se_sw_led_control(struct ieee80211_hw *hw, | ||
111 | enum led_ctl_mode ledaction) | ||
112 | { | ||
113 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
114 | struct rtl_led *pLed0 = &(pcipriv->ledctl.sw_led0); | ||
115 | switch (ledaction) { | ||
116 | case LED_CTL_POWER_ON: | ||
117 | case LED_CTL_LINK: | ||
118 | case LED_CTL_NO_LINK: | ||
119 | rtl92se_sw_led_on(hw, pLed0); | ||
120 | break; | ||
121 | case LED_CTL_POWER_OFF: | ||
122 | rtl92se_sw_led_off(hw, pLed0); | ||
123 | break; | ||
124 | default: | ||
125 | break; | ||
126 | } | ||
127 | } | ||
128 | |||
129 | void rtl92se_led_control(struct ieee80211_hw *hw, enum led_ctl_mode ledaction) | ||
130 | { | ||
131 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
132 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
133 | |||
134 | if ((ppsc->rfoff_reason > RF_CHANGE_BY_PS) && | ||
135 | (ledaction == LED_CTL_TX || | ||
136 | ledaction == LED_CTL_RX || | ||
137 | ledaction == LED_CTL_SITE_SURVEY || | ||
138 | ledaction == LED_CTL_LINK || | ||
139 | ledaction == LED_CTL_NO_LINK || | ||
140 | ledaction == LED_CTL_START_TO_LINK || | ||
141 | ledaction == LED_CTL_POWER_ON)) { | ||
142 | return; | ||
143 | } | ||
144 | RT_TRACE(rtlpriv, COMP_LED, DBG_LOUD, ("ledaction %d,\n", | ||
145 | ledaction)); | ||
146 | |||
147 | _rtl92se_sw_led_control(hw, ledaction); | ||
148 | } | ||
149 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/led.h b/drivers/net/wireless/rtlwifi/rtl8192se/led.h new file mode 100644 index 000000000000..8cce3870af3c --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/led.h | |||
@@ -0,0 +1,37 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_PCI92SE_LED_H__ | ||
30 | #define __REALTEK_PCI92SE_LED_H__ | ||
31 | |||
32 | void rtl92se_init_sw_leds(struct ieee80211_hw *hw); | ||
33 | void rtl92se_sw_led_on(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
34 | void rtl92se_sw_led_off(struct ieee80211_hw *hw, struct rtl_led *pled); | ||
35 | void rtl92se_led_control(struct ieee80211_hw *hw, enum led_ctl_mode ledaction); | ||
36 | |||
37 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/phy.c b/drivers/net/wireless/rtlwifi/rtl8192se/phy.c new file mode 100644 index 000000000000..63b45e60a95e --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/phy.c | |||
@@ -0,0 +1,1740 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../ps.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "phy.h" | ||
36 | #include "rf.h" | ||
37 | #include "dm.h" | ||
38 | #include "fw.h" | ||
39 | #include "hw.h" | ||
40 | #include "table.h" | ||
41 | |||
42 | static u32 _rtl92s_phy_calculate_bit_shift(u32 bitmask) | ||
43 | { | ||
44 | u32 i; | ||
45 | |||
46 | for (i = 0; i <= 31; i++) { | ||
47 | if (((bitmask >> i) & 0x1) == 1) | ||
48 | break; | ||
49 | } | ||
50 | |||
51 | return i; | ||
52 | } | ||
53 | |||
54 | u32 rtl92s_phy_query_bb_reg(struct ieee80211_hw *hw, u32 regaddr, u32 bitmask) | ||
55 | { | ||
56 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
57 | u32 returnvalue = 0, originalvalue, bitshift; | ||
58 | |||
59 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)\n", | ||
60 | regaddr, bitmask)); | ||
61 | |||
62 | originalvalue = rtl_read_dword(rtlpriv, regaddr); | ||
63 | bitshift = _rtl92s_phy_calculate_bit_shift(bitmask); | ||
64 | returnvalue = (originalvalue & bitmask) >> bitshift; | ||
65 | |||
66 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, | ||
67 | ("BBR MASK=0x%x Addr[0x%x]=0x%x\n", | ||
68 | bitmask, regaddr, originalvalue)); | ||
69 | |||
70 | return returnvalue; | ||
71 | |||
72 | } | ||
73 | |||
74 | void rtl92s_phy_set_bb_reg(struct ieee80211_hw *hw, u32 regaddr, u32 bitmask, | ||
75 | u32 data) | ||
76 | { | ||
77 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
78 | u32 originalvalue, bitshift; | ||
79 | |||
80 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)," | ||
81 | " data(%#x)\n", regaddr, bitmask, data)); | ||
82 | |||
83 | if (bitmask != MASKDWORD) { | ||
84 | originalvalue = rtl_read_dword(rtlpriv, regaddr); | ||
85 | bitshift = _rtl92s_phy_calculate_bit_shift(bitmask); | ||
86 | data = ((originalvalue & (~bitmask)) | (data << bitshift)); | ||
87 | } | ||
88 | |||
89 | rtl_write_dword(rtlpriv, regaddr, data); | ||
90 | |||
91 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)," | ||
92 | " data(%#x)\n", regaddr, bitmask, data)); | ||
93 | |||
94 | } | ||
95 | |||
96 | static u32 _rtl92s_phy_rf_serial_read(struct ieee80211_hw *hw, | ||
97 | enum radio_path rfpath, u32 offset) | ||
98 | { | ||
99 | |||
100 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
101 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
102 | struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
103 | u32 newoffset; | ||
104 | u32 tmplong, tmplong2; | ||
105 | u8 rfpi_enable = 0; | ||
106 | u32 retvalue = 0; | ||
107 | |||
108 | offset &= 0x3f; | ||
109 | newoffset = offset; | ||
110 | |||
111 | tmplong = rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD); | ||
112 | |||
113 | if (rfpath == RF90_PATH_A) | ||
114 | tmplong2 = tmplong; | ||
115 | else | ||
116 | tmplong2 = rtl_get_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD); | ||
117 | |||
118 | tmplong2 = (tmplong2 & (~BLSSI_READADDRESS)) | (newoffset << 23) | | ||
119 | BLSSI_READEDGE; | ||
120 | |||
121 | rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, | ||
122 | tmplong & (~BLSSI_READEDGE)); | ||
123 | |||
124 | mdelay(1); | ||
125 | |||
126 | rtl_set_bbreg(hw, pphyreg->rfhssi_para2, MASKDWORD, tmplong2); | ||
127 | mdelay(1); | ||
128 | |||
129 | rtl_set_bbreg(hw, RFPGA0_XA_HSSIPARAMETER2, MASKDWORD, tmplong | | ||
130 | BLSSI_READEDGE); | ||
131 | mdelay(1); | ||
132 | |||
133 | if (rfpath == RF90_PATH_A) | ||
134 | rfpi_enable = (u8)rtl_get_bbreg(hw, RFPGA0_XA_HSSIPARAMETER1, | ||
135 | BIT(8)); | ||
136 | else if (rfpath == RF90_PATH_B) | ||
137 | rfpi_enable = (u8)rtl_get_bbreg(hw, RFPGA0_XB_HSSIPARAMETER1, | ||
138 | BIT(8)); | ||
139 | |||
140 | if (rfpi_enable) | ||
141 | retvalue = rtl_get_bbreg(hw, pphyreg->rflssi_readbackpi, | ||
142 | BLSSI_READBACK_DATA); | ||
143 | else | ||
144 | retvalue = rtl_get_bbreg(hw, pphyreg->rflssi_readback, | ||
145 | BLSSI_READBACK_DATA); | ||
146 | |||
147 | retvalue = rtl_get_bbreg(hw, pphyreg->rflssi_readback, | ||
148 | BLSSI_READBACK_DATA); | ||
149 | |||
150 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("RFR-%d Addr[0x%x]=0x%x\n", | ||
151 | rfpath, pphyreg->rflssi_readback, retvalue)); | ||
152 | |||
153 | return retvalue; | ||
154 | |||
155 | } | ||
156 | |||
157 | static void _rtl92s_phy_rf_serial_write(struct ieee80211_hw *hw, | ||
158 | enum radio_path rfpath, u32 offset, | ||
159 | u32 data) | ||
160 | { | ||
161 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
162 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
163 | struct bb_reg_def *pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
164 | u32 data_and_addr = 0; | ||
165 | u32 newoffset; | ||
166 | |||
167 | offset &= 0x3f; | ||
168 | newoffset = offset; | ||
169 | |||
170 | data_and_addr = ((newoffset << 20) | (data & 0x000fffff)) & 0x0fffffff; | ||
171 | rtl_set_bbreg(hw, pphyreg->rf3wire_offset, MASKDWORD, data_and_addr); | ||
172 | |||
173 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("RFW-%d Addr[0x%x]=0x%x\n", | ||
174 | rfpath, pphyreg->rf3wire_offset, data_and_addr)); | ||
175 | } | ||
176 | |||
177 | |||
178 | u32 rtl92s_phy_query_rf_reg(struct ieee80211_hw *hw, enum radio_path rfpath, | ||
179 | u32 regaddr, u32 bitmask) | ||
180 | { | ||
181 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
182 | u32 original_value, readback_value, bitshift; | ||
183 | unsigned long flags; | ||
184 | |||
185 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), rfpath(%#x), " | ||
186 | "bitmask(%#x)\n", regaddr, rfpath, bitmask)); | ||
187 | |||
188 | spin_lock_irqsave(&rtlpriv->locks.rf_lock, flags); | ||
189 | |||
190 | original_value = _rtl92s_phy_rf_serial_read(hw, rfpath, regaddr); | ||
191 | |||
192 | bitshift = _rtl92s_phy_calculate_bit_shift(bitmask); | ||
193 | readback_value = (original_value & bitmask) >> bitshift; | ||
194 | |||
195 | spin_unlock_irqrestore(&rtlpriv->locks.rf_lock, flags); | ||
196 | |||
197 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), rfpath(%#x), " | ||
198 | "bitmask(%#x), original_value(%#x)\n", regaddr, rfpath, | ||
199 | bitmask, original_value)); | ||
200 | |||
201 | return readback_value; | ||
202 | } | ||
203 | |||
204 | void rtl92s_phy_set_rf_reg(struct ieee80211_hw *hw, enum radio_path rfpath, | ||
205 | u32 regaddr, u32 bitmask, u32 data) | ||
206 | { | ||
207 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
208 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
209 | u32 original_value, bitshift; | ||
210 | unsigned long flags; | ||
211 | |||
212 | if (!((rtlphy->rf_pathmap >> rfpath) & 0x1)) | ||
213 | return; | ||
214 | |||
215 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x)," | ||
216 | " data(%#x), rfpath(%#x)\n", regaddr, bitmask, data, rfpath)); | ||
217 | |||
218 | spin_lock_irqsave(&rtlpriv->locks.rf_lock, flags); | ||
219 | |||
220 | if (bitmask != RFREG_OFFSET_MASK) { | ||
221 | original_value = _rtl92s_phy_rf_serial_read(hw, rfpath, | ||
222 | regaddr); | ||
223 | bitshift = _rtl92s_phy_calculate_bit_shift(bitmask); | ||
224 | data = ((original_value & (~bitmask)) | (data << bitshift)); | ||
225 | } | ||
226 | |||
227 | _rtl92s_phy_rf_serial_write(hw, rfpath, regaddr, data); | ||
228 | |||
229 | spin_unlock_irqrestore(&rtlpriv->locks.rf_lock, flags); | ||
230 | |||
231 | RT_TRACE(rtlpriv, COMP_RF, DBG_TRACE, ("regaddr(%#x), bitmask(%#x), " | ||
232 | "data(%#x), rfpath(%#x)\n", regaddr, bitmask, data, rfpath)); | ||
233 | |||
234 | } | ||
235 | |||
236 | void rtl92s_phy_scan_operation_backup(struct ieee80211_hw *hw, | ||
237 | u8 operation) | ||
238 | { | ||
239 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
240 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
241 | |||
242 | if (!is_hal_stop(rtlhal)) { | ||
243 | switch (operation) { | ||
244 | case SCAN_OPT_BACKUP: | ||
245 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_PAUSE_DM_BY_SCAN); | ||
246 | break; | ||
247 | case SCAN_OPT_RESTORE: | ||
248 | rtl92s_phy_set_fw_cmd(hw, FW_CMD_RESUME_DM_BY_SCAN); | ||
249 | break; | ||
250 | default: | ||
251 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
252 | ("Unknown operation.\n")); | ||
253 | break; | ||
254 | } | ||
255 | } | ||
256 | } | ||
257 | |||
258 | void rtl92s_phy_set_bw_mode(struct ieee80211_hw *hw, | ||
259 | enum nl80211_channel_type ch_type) | ||
260 | { | ||
261 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
262 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
263 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
264 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
265 | u8 reg_bw_opmode; | ||
266 | u8 reg_prsr_rsc; | ||
267 | |||
268 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("Switch to %s bandwidth\n", | ||
269 | rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20 ? | ||
270 | "20MHz" : "40MHz")); | ||
271 | |||
272 | if (rtlphy->set_bwmode_inprogress) | ||
273 | return; | ||
274 | if (is_hal_stop(rtlhal)) | ||
275 | return; | ||
276 | |||
277 | rtlphy->set_bwmode_inprogress = true; | ||
278 | |||
279 | reg_bw_opmode = rtl_read_byte(rtlpriv, BW_OPMODE); | ||
280 | reg_prsr_rsc = rtl_read_byte(rtlpriv, RRSR + 2); | ||
281 | |||
282 | switch (rtlphy->current_chan_bw) { | ||
283 | case HT_CHANNEL_WIDTH_20: | ||
284 | reg_bw_opmode |= BW_OPMODE_20MHZ; | ||
285 | rtl_write_byte(rtlpriv, BW_OPMODE, reg_bw_opmode); | ||
286 | break; | ||
287 | case HT_CHANNEL_WIDTH_20_40: | ||
288 | reg_bw_opmode &= ~BW_OPMODE_20MHZ; | ||
289 | rtl_write_byte(rtlpriv, BW_OPMODE, reg_bw_opmode); | ||
290 | break; | ||
291 | default: | ||
292 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
293 | ("unknown bandwidth: %#X\n", | ||
294 | rtlphy->current_chan_bw)); | ||
295 | break; | ||
296 | } | ||
297 | |||
298 | switch (rtlphy->current_chan_bw) { | ||
299 | case HT_CHANNEL_WIDTH_20: | ||
300 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x0); | ||
301 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x0); | ||
302 | |||
303 | if (rtlhal->version >= VERSION_8192S_BCUT) | ||
304 | rtl_write_byte(rtlpriv, RFPGA0_ANALOGPARAMETER2, 0x58); | ||
305 | break; | ||
306 | case HT_CHANNEL_WIDTH_20_40: | ||
307 | rtl_set_bbreg(hw, RFPGA0_RFMOD, BRFMOD, 0x1); | ||
308 | rtl_set_bbreg(hw, RFPGA1_RFMOD, BRFMOD, 0x1); | ||
309 | |||
310 | rtl_set_bbreg(hw, RCCK0_SYSTEM, BCCK_SIDEBAND, | ||
311 | (mac->cur_40_prime_sc >> 1)); | ||
312 | rtl_set_bbreg(hw, ROFDM1_LSTF, 0xC00, mac->cur_40_prime_sc); | ||
313 | |||
314 | if (rtlhal->version >= VERSION_8192S_BCUT) | ||
315 | rtl_write_byte(rtlpriv, RFPGA0_ANALOGPARAMETER2, 0x18); | ||
316 | break; | ||
317 | default: | ||
318 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
319 | ("unknown bandwidth: %#X\n", rtlphy->current_chan_bw)); | ||
320 | break; | ||
321 | } | ||
322 | |||
323 | rtl92s_phy_rf6052_set_bandwidth(hw, rtlphy->current_chan_bw); | ||
324 | rtlphy->set_bwmode_inprogress = false; | ||
325 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n")); | ||
326 | } | ||
327 | |||
328 | static bool _rtl92s_phy_set_sw_chnl_cmdarray(struct swchnlcmd *cmdtable, | ||
329 | u32 cmdtableidx, u32 cmdtablesz, enum swchnlcmd_id cmdid, | ||
330 | u32 para1, u32 para2, u32 msdelay) | ||
331 | { | ||
332 | struct swchnlcmd *pcmd; | ||
333 | |||
334 | if (cmdtable == NULL) { | ||
335 | RT_ASSERT(false, ("cmdtable cannot be NULL.\n")); | ||
336 | return false; | ||
337 | } | ||
338 | |||
339 | if (cmdtableidx >= cmdtablesz) | ||
340 | return false; | ||
341 | |||
342 | pcmd = cmdtable + cmdtableidx; | ||
343 | pcmd->cmdid = cmdid; | ||
344 | pcmd->para1 = para1; | ||
345 | pcmd->para2 = para2; | ||
346 | pcmd->msdelay = msdelay; | ||
347 | |||
348 | return true; | ||
349 | } | ||
350 | |||
351 | static bool _rtl92s_phy_sw_chnl_step_by_step(struct ieee80211_hw *hw, | ||
352 | u8 channel, u8 *stage, u8 *step, u32 *delay) | ||
353 | { | ||
354 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
355 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
356 | struct swchnlcmd precommoncmd[MAX_PRECMD_CNT]; | ||
357 | u32 precommoncmdcnt; | ||
358 | struct swchnlcmd postcommoncmd[MAX_POSTCMD_CNT]; | ||
359 | u32 postcommoncmdcnt; | ||
360 | struct swchnlcmd rfdependcmd[MAX_RFDEPENDCMD_CNT]; | ||
361 | u32 rfdependcmdcnt; | ||
362 | struct swchnlcmd *currentcmd = NULL; | ||
363 | u8 rfpath; | ||
364 | u8 num_total_rfpath = rtlphy->num_total_rfpath; | ||
365 | |||
366 | precommoncmdcnt = 0; | ||
367 | _rtl92s_phy_set_sw_chnl_cmdarray(precommoncmd, precommoncmdcnt++, | ||
368 | MAX_PRECMD_CNT, CMDID_SET_TXPOWEROWER_LEVEL, 0, 0, 0); | ||
369 | _rtl92s_phy_set_sw_chnl_cmdarray(precommoncmd, precommoncmdcnt++, | ||
370 | MAX_PRECMD_CNT, CMDID_END, 0, 0, 0); | ||
371 | |||
372 | postcommoncmdcnt = 0; | ||
373 | |||
374 | _rtl92s_phy_set_sw_chnl_cmdarray(postcommoncmd, postcommoncmdcnt++, | ||
375 | MAX_POSTCMD_CNT, CMDID_END, 0, 0, 0); | ||
376 | |||
377 | rfdependcmdcnt = 0; | ||
378 | |||
379 | RT_ASSERT((channel >= 1 && channel <= 14), | ||
380 | ("illegal channel for Zebra: %d\n", channel)); | ||
381 | |||
382 | _rtl92s_phy_set_sw_chnl_cmdarray(rfdependcmd, rfdependcmdcnt++, | ||
383 | MAX_RFDEPENDCMD_CNT, CMDID_RF_WRITEREG, | ||
384 | RF_CHNLBW, channel, 10); | ||
385 | |||
386 | _rtl92s_phy_set_sw_chnl_cmdarray(rfdependcmd, rfdependcmdcnt++, | ||
387 | MAX_RFDEPENDCMD_CNT, CMDID_END, 0, 0, 0); | ||
388 | |||
389 | do { | ||
390 | switch (*stage) { | ||
391 | case 0: | ||
392 | currentcmd = &precommoncmd[*step]; | ||
393 | break; | ||
394 | case 1: | ||
395 | currentcmd = &rfdependcmd[*step]; | ||
396 | break; | ||
397 | case 2: | ||
398 | currentcmd = &postcommoncmd[*step]; | ||
399 | break; | ||
400 | } | ||
401 | |||
402 | if (currentcmd->cmdid == CMDID_END) { | ||
403 | if ((*stage) == 2) { | ||
404 | return true; | ||
405 | } else { | ||
406 | (*stage)++; | ||
407 | (*step) = 0; | ||
408 | continue; | ||
409 | } | ||
410 | } | ||
411 | |||
412 | switch (currentcmd->cmdid) { | ||
413 | case CMDID_SET_TXPOWEROWER_LEVEL: | ||
414 | rtl92s_phy_set_txpower(hw, channel); | ||
415 | break; | ||
416 | case CMDID_WRITEPORT_ULONG: | ||
417 | rtl_write_dword(rtlpriv, currentcmd->para1, | ||
418 | currentcmd->para2); | ||
419 | break; | ||
420 | case CMDID_WRITEPORT_USHORT: | ||
421 | rtl_write_word(rtlpriv, currentcmd->para1, | ||
422 | (u16)currentcmd->para2); | ||
423 | break; | ||
424 | case CMDID_WRITEPORT_UCHAR: | ||
425 | rtl_write_byte(rtlpriv, currentcmd->para1, | ||
426 | (u8)currentcmd->para2); | ||
427 | break; | ||
428 | case CMDID_RF_WRITEREG: | ||
429 | for (rfpath = 0; rfpath < num_total_rfpath; rfpath++) { | ||
430 | rtlphy->rfreg_chnlval[rfpath] = | ||
431 | ((rtlphy->rfreg_chnlval[rfpath] & | ||
432 | 0xfffffc00) | currentcmd->para2); | ||
433 | rtl_set_rfreg(hw, (enum radio_path)rfpath, | ||
434 | currentcmd->para1, | ||
435 | RFREG_OFFSET_MASK, | ||
436 | rtlphy->rfreg_chnlval[rfpath]); | ||
437 | } | ||
438 | break; | ||
439 | default: | ||
440 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
441 | ("switch case not process\n")); | ||
442 | break; | ||
443 | } | ||
444 | |||
445 | break; | ||
446 | } while (true); | ||
447 | |||
448 | (*delay) = currentcmd->msdelay; | ||
449 | (*step)++; | ||
450 | return false; | ||
451 | } | ||
452 | |||
453 | u8 rtl92s_phy_sw_chnl(struct ieee80211_hw *hw) | ||
454 | { | ||
455 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
456 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
457 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
458 | u32 delay; | ||
459 | bool ret; | ||
460 | |||
461 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, | ||
462 | ("switch to channel%d\n", | ||
463 | rtlphy->current_channel)); | ||
464 | |||
465 | if (rtlphy->sw_chnl_inprogress) | ||
466 | return 0; | ||
467 | |||
468 | if (rtlphy->set_bwmode_inprogress) | ||
469 | return 0; | ||
470 | |||
471 | if (is_hal_stop(rtlhal)) | ||
472 | return 0; | ||
473 | |||
474 | rtlphy->sw_chnl_inprogress = true; | ||
475 | rtlphy->sw_chnl_stage = 0; | ||
476 | rtlphy->sw_chnl_step = 0; | ||
477 | |||
478 | do { | ||
479 | if (!rtlphy->sw_chnl_inprogress) | ||
480 | break; | ||
481 | |||
482 | ret = _rtl92s_phy_sw_chnl_step_by_step(hw, | ||
483 | rtlphy->current_channel, | ||
484 | &rtlphy->sw_chnl_stage, | ||
485 | &rtlphy->sw_chnl_step, &delay); | ||
486 | if (!ret) { | ||
487 | if (delay > 0) | ||
488 | mdelay(delay); | ||
489 | else | ||
490 | continue; | ||
491 | } else { | ||
492 | rtlphy->sw_chnl_inprogress = false; | ||
493 | } | ||
494 | break; | ||
495 | } while (true); | ||
496 | |||
497 | rtlphy->sw_chnl_inprogress = false; | ||
498 | |||
499 | RT_TRACE(rtlpriv, COMP_SCAN, DBG_TRACE, ("<==\n")); | ||
500 | |||
501 | return 1; | ||
502 | } | ||
503 | |||
504 | static void _rtl92se_phy_set_rf_sleep(struct ieee80211_hw *hw) | ||
505 | { | ||
506 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
507 | u8 u1btmp; | ||
508 | |||
509 | u1btmp = rtl_read_byte(rtlpriv, LDOV12D_CTRL); | ||
510 | u1btmp |= BIT(0); | ||
511 | |||
512 | rtl_write_byte(rtlpriv, LDOV12D_CTRL, u1btmp); | ||
513 | rtl_write_byte(rtlpriv, SPS1_CTRL, 0x0); | ||
514 | rtl_write_byte(rtlpriv, TXPAUSE, 0xFF); | ||
515 | rtl_write_word(rtlpriv, CMDR, 0x57FC); | ||
516 | udelay(100); | ||
517 | |||
518 | rtl_write_word(rtlpriv, CMDR, 0x77FC); | ||
519 | rtl_write_byte(rtlpriv, PHY_CCA, 0x0); | ||
520 | udelay(10); | ||
521 | |||
522 | rtl_write_word(rtlpriv, CMDR, 0x37FC); | ||
523 | udelay(10); | ||
524 | |||
525 | rtl_write_word(rtlpriv, CMDR, 0x77FC); | ||
526 | udelay(10); | ||
527 | |||
528 | rtl_write_word(rtlpriv, CMDR, 0x57FC); | ||
529 | |||
530 | /* we should chnge GPIO to input mode | ||
531 | * this will drop away current about 25mA*/ | ||
532 | rtl8192se_gpiobit3_cfg_inputmode(hw); | ||
533 | } | ||
534 | |||
535 | bool rtl92s_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
536 | enum rf_pwrstate rfpwr_state) | ||
537 | { | ||
538 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
539 | struct rtl_pci_priv *pcipriv = rtl_pcipriv(hw); | ||
540 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
541 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
542 | bool bresult = true; | ||
543 | u8 i, queue_id; | ||
544 | struct rtl8192_tx_ring *ring = NULL; | ||
545 | |||
546 | if (rfpwr_state == ppsc->rfpwr_state) | ||
547 | return false; | ||
548 | |||
549 | ppsc->set_rfpowerstate_inprogress = true; | ||
550 | |||
551 | switch (rfpwr_state) { | ||
552 | case ERFON:{ | ||
553 | if ((ppsc->rfpwr_state == ERFOFF) && | ||
554 | RT_IN_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC)) { | ||
555 | |||
556 | bool rtstatus; | ||
557 | u32 InitializeCount = 0; | ||
558 | do { | ||
559 | InitializeCount++; | ||
560 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
561 | ("IPS Set eRf nic enable\n")); | ||
562 | rtstatus = rtl_ps_enable_nic(hw); | ||
563 | } while ((rtstatus != true) && | ||
564 | (InitializeCount < 10)); | ||
565 | |||
566 | RT_CLEAR_PS_LEVEL(ppsc, | ||
567 | RT_RF_OFF_LEVL_HALT_NIC); | ||
568 | } else { | ||
569 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
570 | ("awake, sleeped:%d ms " | ||
571 | "state_inap:%x\n", | ||
572 | jiffies_to_msecs(jiffies - | ||
573 | ppsc->last_sleep_jiffies), | ||
574 | rtlpriv->psc.state_inap)); | ||
575 | ppsc->last_awake_jiffies = jiffies; | ||
576 | rtl_write_word(rtlpriv, CMDR, 0x37FC); | ||
577 | rtl_write_byte(rtlpriv, TXPAUSE, 0x00); | ||
578 | rtl_write_byte(rtlpriv, PHY_CCA, 0x3); | ||
579 | } | ||
580 | |||
581 | if (mac->link_state == MAC80211_LINKED) | ||
582 | rtlpriv->cfg->ops->led_control(hw, | ||
583 | LED_CTL_LINK); | ||
584 | else | ||
585 | rtlpriv->cfg->ops->led_control(hw, | ||
586 | LED_CTL_NO_LINK); | ||
587 | break; | ||
588 | } | ||
589 | case ERFOFF:{ | ||
590 | if (ppsc->reg_rfps_level & RT_RF_OFF_LEVL_HALT_NIC) { | ||
591 | RT_TRACE(rtlpriv, COMP_RF, DBG_DMESG, | ||
592 | ("IPS Set eRf nic disable\n")); | ||
593 | rtl_ps_disable_nic(hw); | ||
594 | RT_SET_PS_LEVEL(ppsc, RT_RF_OFF_LEVL_HALT_NIC); | ||
595 | } else { | ||
596 | if (ppsc->rfoff_reason == RF_CHANGE_BY_IPS) | ||
597 | rtlpriv->cfg->ops->led_control(hw, | ||
598 | LED_CTL_NO_LINK); | ||
599 | else | ||
600 | rtlpriv->cfg->ops->led_control(hw, | ||
601 | LED_CTL_POWER_OFF); | ||
602 | } | ||
603 | break; | ||
604 | } | ||
605 | case ERFSLEEP: | ||
606 | if (ppsc->rfpwr_state == ERFOFF) | ||
607 | break; | ||
608 | |||
609 | for (queue_id = 0, i = 0; | ||
610 | queue_id < RTL_PCI_MAX_TX_QUEUE_COUNT;) { | ||
611 | ring = &pcipriv->dev.tx_ring[queue_id]; | ||
612 | if (skb_queue_len(&ring->queue) == 0 || | ||
613 | queue_id == BEACON_QUEUE) { | ||
614 | queue_id++; | ||
615 | continue; | ||
616 | } else { | ||
617 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
618 | ("eRf Off/Sleep: " | ||
619 | "%d times TcbBusyQueue[%d] = " | ||
620 | "%d before doze!\n", | ||
621 | (i + 1), queue_id, | ||
622 | skb_queue_len(&ring->queue))); | ||
623 | |||
624 | udelay(10); | ||
625 | i++; | ||
626 | } | ||
627 | |||
628 | if (i >= MAX_DOZE_WAITING_TIMES_9x) { | ||
629 | RT_TRACE(rtlpriv, COMP_ERR, DBG_WARNING, | ||
630 | ("\nERFOFF: %d times" | ||
631 | "TcbBusyQueue[%d] = %d !\n", | ||
632 | MAX_DOZE_WAITING_TIMES_9x, | ||
633 | queue_id, | ||
634 | skb_queue_len(&ring->queue))); | ||
635 | break; | ||
636 | } | ||
637 | } | ||
638 | |||
639 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
640 | ("Set ERFSLEEP awaked:%d ms\n", | ||
641 | jiffies_to_msecs(jiffies - | ||
642 | ppsc->last_awake_jiffies))); | ||
643 | |||
644 | RT_TRACE(rtlpriv, COMP_POWER, DBG_DMESG, | ||
645 | ("sleep awaked:%d ms " | ||
646 | "state_inap:%x\n", jiffies_to_msecs(jiffies - | ||
647 | ppsc->last_awake_jiffies), | ||
648 | rtlpriv->psc.state_inap)); | ||
649 | ppsc->last_sleep_jiffies = jiffies; | ||
650 | _rtl92se_phy_set_rf_sleep(hw); | ||
651 | break; | ||
652 | default: | ||
653 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
654 | ("switch case not process\n")); | ||
655 | bresult = false; | ||
656 | break; | ||
657 | } | ||
658 | |||
659 | if (bresult) | ||
660 | ppsc->rfpwr_state = rfpwr_state; | ||
661 | |||
662 | ppsc->set_rfpowerstate_inprogress = false; | ||
663 | |||
664 | return bresult; | ||
665 | } | ||
666 | |||
667 | static bool _rtl92s_phy_config_rfpa_bias_current(struct ieee80211_hw *hw, | ||
668 | enum radio_path rfpath) | ||
669 | { | ||
670 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
671 | bool rtstatus = true; | ||
672 | u32 tmpval = 0; | ||
673 | |||
674 | /* If inferiority IC, we have to increase the PA bias current */ | ||
675 | if (rtlhal->ic_class != IC_INFERIORITY_A) { | ||
676 | tmpval = rtl92s_phy_query_rf_reg(hw, rfpath, RF_IPA, 0xf); | ||
677 | rtl92s_phy_set_rf_reg(hw, rfpath, RF_IPA, 0xf, tmpval + 1); | ||
678 | } | ||
679 | |||
680 | return rtstatus; | ||
681 | } | ||
682 | |||
683 | static void _rtl92s_store_pwrindex_diffrate_offset(struct ieee80211_hw *hw, | ||
684 | u32 reg_addr, u32 bitmask, u32 data) | ||
685 | { | ||
686 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
687 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
688 | |||
689 | if (reg_addr == RTXAGC_RATE18_06) | ||
690 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][0] = | ||
691 | data; | ||
692 | if (reg_addr == RTXAGC_RATE54_24) | ||
693 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][1] = | ||
694 | data; | ||
695 | if (reg_addr == RTXAGC_CCK_MCS32) | ||
696 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][6] = | ||
697 | data; | ||
698 | if (reg_addr == RTXAGC_MCS03_MCS00) | ||
699 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][2] = | ||
700 | data; | ||
701 | if (reg_addr == RTXAGC_MCS07_MCS04) | ||
702 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][3] = | ||
703 | data; | ||
704 | if (reg_addr == RTXAGC_MCS11_MCS08) | ||
705 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][4] = | ||
706 | data; | ||
707 | if (reg_addr == RTXAGC_MCS15_MCS12) { | ||
708 | rtlphy->mcs_txpwrlevel_origoffset[rtlphy->pwrgroup_cnt][5] = | ||
709 | data; | ||
710 | rtlphy->pwrgroup_cnt++; | ||
711 | } | ||
712 | } | ||
713 | |||
714 | static void _rtl92s_phy_init_register_definition(struct ieee80211_hw *hw) | ||
715 | { | ||
716 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
717 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
718 | |||
719 | /*RF Interface Sowrtware Control */ | ||
720 | rtlphy->phyreg_def[RF90_PATH_A].rfintfs = RFPGA0_XAB_RFINTERFACESW; | ||
721 | rtlphy->phyreg_def[RF90_PATH_B].rfintfs = RFPGA0_XAB_RFINTERFACESW; | ||
722 | rtlphy->phyreg_def[RF90_PATH_C].rfintfs = RFPGA0_XCD_RFINTERFACESW; | ||
723 | rtlphy->phyreg_def[RF90_PATH_D].rfintfs = RFPGA0_XCD_RFINTERFACESW; | ||
724 | |||
725 | /* RF Interface Readback Value */ | ||
726 | rtlphy->phyreg_def[RF90_PATH_A].rfintfi = RFPGA0_XAB_RFINTERFACERB; | ||
727 | rtlphy->phyreg_def[RF90_PATH_B].rfintfi = RFPGA0_XAB_RFINTERFACERB; | ||
728 | rtlphy->phyreg_def[RF90_PATH_C].rfintfi = RFPGA0_XCD_RFINTERFACERB; | ||
729 | rtlphy->phyreg_def[RF90_PATH_D].rfintfi = RFPGA0_XCD_RFINTERFACERB; | ||
730 | |||
731 | /* RF Interface Output (and Enable) */ | ||
732 | rtlphy->phyreg_def[RF90_PATH_A].rfintfo = RFPGA0_XA_RFINTERFACEOE; | ||
733 | rtlphy->phyreg_def[RF90_PATH_B].rfintfo = RFPGA0_XB_RFINTERFACEOE; | ||
734 | rtlphy->phyreg_def[RF90_PATH_C].rfintfo = RFPGA0_XC_RFINTERFACEOE; | ||
735 | rtlphy->phyreg_def[RF90_PATH_D].rfintfo = RFPGA0_XD_RFINTERFACEOE; | ||
736 | |||
737 | /* RF Interface (Output and) Enable */ | ||
738 | rtlphy->phyreg_def[RF90_PATH_A].rfintfe = RFPGA0_XA_RFINTERFACEOE; | ||
739 | rtlphy->phyreg_def[RF90_PATH_B].rfintfe = RFPGA0_XB_RFINTERFACEOE; | ||
740 | rtlphy->phyreg_def[RF90_PATH_C].rfintfe = RFPGA0_XC_RFINTERFACEOE; | ||
741 | rtlphy->phyreg_def[RF90_PATH_D].rfintfe = RFPGA0_XD_RFINTERFACEOE; | ||
742 | |||
743 | /* Addr of LSSI. Wirte RF register by driver */ | ||
744 | rtlphy->phyreg_def[RF90_PATH_A].rf3wire_offset = | ||
745 | RFPGA0_XA_LSSIPARAMETER; | ||
746 | rtlphy->phyreg_def[RF90_PATH_B].rf3wire_offset = | ||
747 | RFPGA0_XB_LSSIPARAMETER; | ||
748 | rtlphy->phyreg_def[RF90_PATH_C].rf3wire_offset = | ||
749 | RFPGA0_XC_LSSIPARAMETER; | ||
750 | rtlphy->phyreg_def[RF90_PATH_D].rf3wire_offset = | ||
751 | RFPGA0_XD_LSSIPARAMETER; | ||
752 | |||
753 | /* RF parameter */ | ||
754 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_select = RFPGA0_XAB_RFPARAMETER; | ||
755 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_select = RFPGA0_XAB_RFPARAMETER; | ||
756 | rtlphy->phyreg_def[RF90_PATH_C].rflssi_select = RFPGA0_XCD_RFPARAMETER; | ||
757 | rtlphy->phyreg_def[RF90_PATH_D].rflssi_select = RFPGA0_XCD_RFPARAMETER; | ||
758 | |||
759 | /* Tx AGC Gain Stage (same for all path. Should we remove this?) */ | ||
760 | rtlphy->phyreg_def[RF90_PATH_A].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
761 | rtlphy->phyreg_def[RF90_PATH_B].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
762 | rtlphy->phyreg_def[RF90_PATH_C].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
763 | rtlphy->phyreg_def[RF90_PATH_D].rftxgain_stage = RFPGA0_TXGAINSTAGE; | ||
764 | |||
765 | /* Tranceiver A~D HSSI Parameter-1 */ | ||
766 | rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para1 = RFPGA0_XA_HSSIPARAMETER1; | ||
767 | rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para1 = RFPGA0_XB_HSSIPARAMETER1; | ||
768 | rtlphy->phyreg_def[RF90_PATH_C].rfhssi_para1 = RFPGA0_XC_HSSIPARAMETER1; | ||
769 | rtlphy->phyreg_def[RF90_PATH_D].rfhssi_para1 = RFPGA0_XD_HSSIPARAMETER1; | ||
770 | |||
771 | /* Tranceiver A~D HSSI Parameter-2 */ | ||
772 | rtlphy->phyreg_def[RF90_PATH_A].rfhssi_para2 = RFPGA0_XA_HSSIPARAMETER2; | ||
773 | rtlphy->phyreg_def[RF90_PATH_B].rfhssi_para2 = RFPGA0_XB_HSSIPARAMETER2; | ||
774 | rtlphy->phyreg_def[RF90_PATH_C].rfhssi_para2 = RFPGA0_XC_HSSIPARAMETER2; | ||
775 | rtlphy->phyreg_def[RF90_PATH_D].rfhssi_para2 = RFPGA0_XD_HSSIPARAMETER2; | ||
776 | |||
777 | /* RF switch Control */ | ||
778 | rtlphy->phyreg_def[RF90_PATH_A].rfswitch_control = | ||
779 | RFPGA0_XAB_SWITCHCONTROL; | ||
780 | rtlphy->phyreg_def[RF90_PATH_B].rfswitch_control = | ||
781 | RFPGA0_XAB_SWITCHCONTROL; | ||
782 | rtlphy->phyreg_def[RF90_PATH_C].rfswitch_control = | ||
783 | RFPGA0_XCD_SWITCHCONTROL; | ||
784 | rtlphy->phyreg_def[RF90_PATH_D].rfswitch_control = | ||
785 | RFPGA0_XCD_SWITCHCONTROL; | ||
786 | |||
787 | /* AGC control 1 */ | ||
788 | rtlphy->phyreg_def[RF90_PATH_A].rfagc_control1 = ROFDM0_XAAGCCORE1; | ||
789 | rtlphy->phyreg_def[RF90_PATH_B].rfagc_control1 = ROFDM0_XBAGCCORE1; | ||
790 | rtlphy->phyreg_def[RF90_PATH_C].rfagc_control1 = ROFDM0_XCAGCCORE1; | ||
791 | rtlphy->phyreg_def[RF90_PATH_D].rfagc_control1 = ROFDM0_XDAGCCORE1; | ||
792 | |||
793 | /* AGC control 2 */ | ||
794 | rtlphy->phyreg_def[RF90_PATH_A].rfagc_control2 = ROFDM0_XAAGCCORE2; | ||
795 | rtlphy->phyreg_def[RF90_PATH_B].rfagc_control2 = ROFDM0_XBAGCCORE2; | ||
796 | rtlphy->phyreg_def[RF90_PATH_C].rfagc_control2 = ROFDM0_XCAGCCORE2; | ||
797 | rtlphy->phyreg_def[RF90_PATH_D].rfagc_control2 = ROFDM0_XDAGCCORE2; | ||
798 | |||
799 | /* RX AFE control 1 */ | ||
800 | rtlphy->phyreg_def[RF90_PATH_A].rfrxiq_imbalance = | ||
801 | ROFDM0_XARXIQIMBALANCE; | ||
802 | rtlphy->phyreg_def[RF90_PATH_B].rfrxiq_imbalance = | ||
803 | ROFDM0_XBRXIQIMBALANCE; | ||
804 | rtlphy->phyreg_def[RF90_PATH_C].rfrxiq_imbalance = | ||
805 | ROFDM0_XCRXIQIMBALANCE; | ||
806 | rtlphy->phyreg_def[RF90_PATH_D].rfrxiq_imbalance = | ||
807 | ROFDM0_XDRXIQIMBALANCE; | ||
808 | |||
809 | /* RX AFE control 1 */ | ||
810 | rtlphy->phyreg_def[RF90_PATH_A].rfrx_afe = ROFDM0_XARXAFE; | ||
811 | rtlphy->phyreg_def[RF90_PATH_B].rfrx_afe = ROFDM0_XBRXAFE; | ||
812 | rtlphy->phyreg_def[RF90_PATH_C].rfrx_afe = ROFDM0_XCRXAFE; | ||
813 | rtlphy->phyreg_def[RF90_PATH_D].rfrx_afe = ROFDM0_XDRXAFE; | ||
814 | |||
815 | /* Tx AFE control 1 */ | ||
816 | rtlphy->phyreg_def[RF90_PATH_A].rftxiq_imbalance = | ||
817 | ROFDM0_XATXIQIMBALANCE; | ||
818 | rtlphy->phyreg_def[RF90_PATH_B].rftxiq_imbalance = | ||
819 | ROFDM0_XBTXIQIMBALANCE; | ||
820 | rtlphy->phyreg_def[RF90_PATH_C].rftxiq_imbalance = | ||
821 | ROFDM0_XCTXIQIMBALANCE; | ||
822 | rtlphy->phyreg_def[RF90_PATH_D].rftxiq_imbalance = | ||
823 | ROFDM0_XDTXIQIMBALANCE; | ||
824 | |||
825 | /* Tx AFE control 2 */ | ||
826 | rtlphy->phyreg_def[RF90_PATH_A].rftx_afe = ROFDM0_XATXAFE; | ||
827 | rtlphy->phyreg_def[RF90_PATH_B].rftx_afe = ROFDM0_XBTXAFE; | ||
828 | rtlphy->phyreg_def[RF90_PATH_C].rftx_afe = ROFDM0_XCTXAFE; | ||
829 | rtlphy->phyreg_def[RF90_PATH_D].rftx_afe = ROFDM0_XDTXAFE; | ||
830 | |||
831 | /* Tranceiver LSSI Readback */ | ||
832 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_readback = | ||
833 | RFPGA0_XA_LSSIREADBACK; | ||
834 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_readback = | ||
835 | RFPGA0_XB_LSSIREADBACK; | ||
836 | rtlphy->phyreg_def[RF90_PATH_C].rflssi_readback = | ||
837 | RFPGA0_XC_LSSIREADBACK; | ||
838 | rtlphy->phyreg_def[RF90_PATH_D].rflssi_readback = | ||
839 | RFPGA0_XD_LSSIREADBACK; | ||
840 | |||
841 | /* Tranceiver LSSI Readback PI mode */ | ||
842 | rtlphy->phyreg_def[RF90_PATH_A].rflssi_readbackpi = | ||
843 | TRANSCEIVERA_HSPI_READBACK; | ||
844 | rtlphy->phyreg_def[RF90_PATH_B].rflssi_readbackpi = | ||
845 | TRANSCEIVERB_HSPI_READBACK; | ||
846 | } | ||
847 | |||
848 | |||
849 | static bool _rtl92s_phy_config_bb(struct ieee80211_hw *hw, u8 configtype) | ||
850 | { | ||
851 | int i; | ||
852 | u32 *phy_reg_table; | ||
853 | u32 *agc_table; | ||
854 | u16 phy_reg_len, agc_len; | ||
855 | |||
856 | agc_len = AGCTAB_ARRAYLENGTH; | ||
857 | agc_table = rtl8192seagctab_array; | ||
858 | /* Default RF_type: 2T2R */ | ||
859 | phy_reg_len = PHY_REG_2T2RARRAYLENGTH; | ||
860 | phy_reg_table = rtl8192sephy_reg_2t2rarray; | ||
861 | |||
862 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
863 | for (i = 0; i < phy_reg_len; i = i + 2) { | ||
864 | if (phy_reg_table[i] == 0xfe) | ||
865 | mdelay(50); | ||
866 | else if (phy_reg_table[i] == 0xfd) | ||
867 | mdelay(5); | ||
868 | else if (phy_reg_table[i] == 0xfc) | ||
869 | mdelay(1); | ||
870 | else if (phy_reg_table[i] == 0xfb) | ||
871 | udelay(50); | ||
872 | else if (phy_reg_table[i] == 0xfa) | ||
873 | udelay(5); | ||
874 | else if (phy_reg_table[i] == 0xf9) | ||
875 | udelay(1); | ||
876 | |||
877 | /* Add delay for ECS T20 & LG malow platform, */ | ||
878 | udelay(1); | ||
879 | |||
880 | rtl92s_phy_set_bb_reg(hw, phy_reg_table[i], MASKDWORD, | ||
881 | phy_reg_table[i + 1]); | ||
882 | } | ||
883 | } else if (configtype == BASEBAND_CONFIG_AGC_TAB) { | ||
884 | for (i = 0; i < agc_len; i = i + 2) { | ||
885 | rtl92s_phy_set_bb_reg(hw, agc_table[i], MASKDWORD, | ||
886 | agc_table[i + 1]); | ||
887 | |||
888 | /* Add delay for ECS T20 & LG malow platform */ | ||
889 | udelay(1); | ||
890 | } | ||
891 | } | ||
892 | |||
893 | return true; | ||
894 | } | ||
895 | |||
896 | static bool _rtl92s_phy_set_bb_to_diff_rf(struct ieee80211_hw *hw, | ||
897 | u8 configtype) | ||
898 | { | ||
899 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
900 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
901 | u32 *phy_regarray2xtxr_table; | ||
902 | u16 phy_regarray2xtxr_len; | ||
903 | int i; | ||
904 | |||
905 | if (rtlphy->rf_type == RF_1T1R) { | ||
906 | phy_regarray2xtxr_table = rtl8192sephy_changeto_1t1rarray; | ||
907 | phy_regarray2xtxr_len = PHY_CHANGETO_1T1RARRAYLENGTH; | ||
908 | } else if (rtlphy->rf_type == RF_1T2R) { | ||
909 | phy_regarray2xtxr_table = rtl8192sephy_changeto_1t2rarray; | ||
910 | phy_regarray2xtxr_len = PHY_CHANGETO_1T2RARRAYLENGTH; | ||
911 | } else { | ||
912 | return false; | ||
913 | } | ||
914 | |||
915 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
916 | for (i = 0; i < phy_regarray2xtxr_len; i = i + 3) { | ||
917 | if (phy_regarray2xtxr_table[i] == 0xfe) | ||
918 | mdelay(50); | ||
919 | else if (phy_regarray2xtxr_table[i] == 0xfd) | ||
920 | mdelay(5); | ||
921 | else if (phy_regarray2xtxr_table[i] == 0xfc) | ||
922 | mdelay(1); | ||
923 | else if (phy_regarray2xtxr_table[i] == 0xfb) | ||
924 | udelay(50); | ||
925 | else if (phy_regarray2xtxr_table[i] == 0xfa) | ||
926 | udelay(5); | ||
927 | else if (phy_regarray2xtxr_table[i] == 0xf9) | ||
928 | udelay(1); | ||
929 | |||
930 | rtl92s_phy_set_bb_reg(hw, phy_regarray2xtxr_table[i], | ||
931 | phy_regarray2xtxr_table[i + 1], | ||
932 | phy_regarray2xtxr_table[i + 2]); | ||
933 | } | ||
934 | } | ||
935 | |||
936 | return true; | ||
937 | } | ||
938 | |||
939 | static bool _rtl92s_phy_config_bb_with_pg(struct ieee80211_hw *hw, | ||
940 | u8 configtype) | ||
941 | { | ||
942 | int i; | ||
943 | u32 *phy_table_pg; | ||
944 | u16 phy_pg_len; | ||
945 | |||
946 | phy_pg_len = PHY_REG_ARRAY_PGLENGTH; | ||
947 | phy_table_pg = rtl8192sephy_reg_array_pg; | ||
948 | |||
949 | if (configtype == BASEBAND_CONFIG_PHY_REG) { | ||
950 | for (i = 0; i < phy_pg_len; i = i + 3) { | ||
951 | if (phy_table_pg[i] == 0xfe) | ||
952 | mdelay(50); | ||
953 | else if (phy_table_pg[i] == 0xfd) | ||
954 | mdelay(5); | ||
955 | else if (phy_table_pg[i] == 0xfc) | ||
956 | mdelay(1); | ||
957 | else if (phy_table_pg[i] == 0xfb) | ||
958 | udelay(50); | ||
959 | else if (phy_table_pg[i] == 0xfa) | ||
960 | udelay(5); | ||
961 | else if (phy_table_pg[i] == 0xf9) | ||
962 | udelay(1); | ||
963 | |||
964 | _rtl92s_store_pwrindex_diffrate_offset(hw, | ||
965 | phy_table_pg[i], | ||
966 | phy_table_pg[i + 1], | ||
967 | phy_table_pg[i + 2]); | ||
968 | rtl92s_phy_set_bb_reg(hw, phy_table_pg[i], | ||
969 | phy_table_pg[i + 1], | ||
970 | phy_table_pg[i + 2]); | ||
971 | } | ||
972 | } | ||
973 | |||
974 | return true; | ||
975 | } | ||
976 | |||
977 | static bool _rtl92s_phy_bb_config_parafile(struct ieee80211_hw *hw) | ||
978 | { | ||
979 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
980 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
981 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
982 | bool rtstatus = true; | ||
983 | |||
984 | /* 1. Read PHY_REG.TXT BB INIT!! */ | ||
985 | /* We will separate as 1T1R/1T2R/1T2R_GREEN/2T2R */ | ||
986 | if (rtlphy->rf_type == RF_1T2R || rtlphy->rf_type == RF_2T2R || | ||
987 | rtlphy->rf_type == RF_1T1R || rtlphy->rf_type == RF_2T2R_GREEN) { | ||
988 | rtstatus = _rtl92s_phy_config_bb(hw, BASEBAND_CONFIG_PHY_REG); | ||
989 | |||
990 | if (rtlphy->rf_type != RF_2T2R && | ||
991 | rtlphy->rf_type != RF_2T2R_GREEN) | ||
992 | /* so we should reconfig BB reg with the right | ||
993 | * PHY parameters. */ | ||
994 | rtstatus = _rtl92s_phy_set_bb_to_diff_rf(hw, | ||
995 | BASEBAND_CONFIG_PHY_REG); | ||
996 | } else { | ||
997 | rtstatus = false; | ||
998 | } | ||
999 | |||
1000 | if (rtstatus != true) { | ||
1001 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
1002 | ("Write BB Reg Fail!!")); | ||
1003 | goto phy_BB8190_Config_ParaFile_Fail; | ||
1004 | } | ||
1005 | |||
1006 | /* 2. If EEPROM or EFUSE autoload OK, We must config by | ||
1007 | * PHY_REG_PG.txt */ | ||
1008 | if (rtlefuse->autoload_failflag == false) { | ||
1009 | rtlphy->pwrgroup_cnt = 0; | ||
1010 | |||
1011 | rtstatus = _rtl92s_phy_config_bb_with_pg(hw, | ||
1012 | BASEBAND_CONFIG_PHY_REG); | ||
1013 | } | ||
1014 | if (rtstatus != true) { | ||
1015 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
1016 | ("_rtl92s_phy_bb_config_parafile(): " | ||
1017 | "BB_PG Reg Fail!!")); | ||
1018 | goto phy_BB8190_Config_ParaFile_Fail; | ||
1019 | } | ||
1020 | |||
1021 | /* 3. BB AGC table Initialization */ | ||
1022 | rtstatus = _rtl92s_phy_config_bb(hw, BASEBAND_CONFIG_AGC_TAB); | ||
1023 | |||
1024 | if (rtstatus != true) { | ||
1025 | printk(KERN_ERR "_rtl92s_phy_bb_config_parafile(): " | ||
1026 | "AGC Table Fail\n"); | ||
1027 | goto phy_BB8190_Config_ParaFile_Fail; | ||
1028 | } | ||
1029 | |||
1030 | /* Check if the CCK HighPower is turned ON. */ | ||
1031 | /* This is used to calculate PWDB. */ | ||
1032 | rtlphy->cck_high_power = (bool)(rtl92s_phy_query_bb_reg(hw, | ||
1033 | RFPGA0_XA_HSSIPARAMETER2, 0x200)); | ||
1034 | |||
1035 | phy_BB8190_Config_ParaFile_Fail: | ||
1036 | return rtstatus; | ||
1037 | } | ||
1038 | |||
1039 | u8 rtl92s_phy_config_rf(struct ieee80211_hw *hw, enum radio_path rfpath) | ||
1040 | { | ||
1041 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1042 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1043 | int i; | ||
1044 | bool rtstatus = true; | ||
1045 | u32 *radio_a_table; | ||
1046 | u32 *radio_b_table; | ||
1047 | u16 radio_a_tblen, radio_b_tblen; | ||
1048 | |||
1049 | radio_a_tblen = RADIOA_1T_ARRAYLENGTH; | ||
1050 | radio_a_table = rtl8192seradioa_1t_array; | ||
1051 | |||
1052 | /* Using Green mode array table for RF_2T2R_GREEN */ | ||
1053 | if (rtlphy->rf_type == RF_2T2R_GREEN) { | ||
1054 | radio_b_table = rtl8192seradiob_gm_array; | ||
1055 | radio_b_tblen = RADIOB_GM_ARRAYLENGTH; | ||
1056 | } else { | ||
1057 | radio_b_table = rtl8192seradiob_array; | ||
1058 | radio_b_tblen = RADIOB_ARRAYLENGTH; | ||
1059 | } | ||
1060 | |||
1061 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Radio No %x\n", rfpath)); | ||
1062 | rtstatus = true; | ||
1063 | |||
1064 | switch (rfpath) { | ||
1065 | case RF90_PATH_A: | ||
1066 | for (i = 0; i < radio_a_tblen; i = i + 2) { | ||
1067 | if (radio_a_table[i] == 0xfe) | ||
1068 | /* Delay specific ms. Only RF configuration | ||
1069 | * requires delay. */ | ||
1070 | mdelay(50); | ||
1071 | else if (radio_a_table[i] == 0xfd) | ||
1072 | mdelay(5); | ||
1073 | else if (radio_a_table[i] == 0xfc) | ||
1074 | mdelay(1); | ||
1075 | else if (radio_a_table[i] == 0xfb) | ||
1076 | udelay(50); | ||
1077 | else if (radio_a_table[i] == 0xfa) | ||
1078 | udelay(5); | ||
1079 | else if (radio_a_table[i] == 0xf9) | ||
1080 | udelay(1); | ||
1081 | else | ||
1082 | rtl92s_phy_set_rf_reg(hw, rfpath, | ||
1083 | radio_a_table[i], | ||
1084 | MASK20BITS, | ||
1085 | radio_a_table[i + 1]); | ||
1086 | |||
1087 | /* Add delay for ECS T20 & LG malow platform */ | ||
1088 | udelay(1); | ||
1089 | } | ||
1090 | |||
1091 | /* PA Bias current for inferiority IC */ | ||
1092 | _rtl92s_phy_config_rfpa_bias_current(hw, rfpath); | ||
1093 | break; | ||
1094 | case RF90_PATH_B: | ||
1095 | for (i = 0; i < radio_b_tblen; i = i + 2) { | ||
1096 | if (radio_b_table[i] == 0xfe) | ||
1097 | /* Delay specific ms. Only RF configuration | ||
1098 | * requires delay.*/ | ||
1099 | mdelay(50); | ||
1100 | else if (radio_b_table[i] == 0xfd) | ||
1101 | mdelay(5); | ||
1102 | else if (radio_b_table[i] == 0xfc) | ||
1103 | mdelay(1); | ||
1104 | else if (radio_b_table[i] == 0xfb) | ||
1105 | udelay(50); | ||
1106 | else if (radio_b_table[i] == 0xfa) | ||
1107 | udelay(5); | ||
1108 | else if (radio_b_table[i] == 0xf9) | ||
1109 | udelay(1); | ||
1110 | else | ||
1111 | rtl92s_phy_set_rf_reg(hw, rfpath, | ||
1112 | radio_b_table[i], | ||
1113 | MASK20BITS, | ||
1114 | radio_b_table[i + 1]); | ||
1115 | |||
1116 | /* Add delay for ECS T20 & LG malow platform */ | ||
1117 | udelay(1); | ||
1118 | } | ||
1119 | break; | ||
1120 | case RF90_PATH_C: | ||
1121 | ; | ||
1122 | break; | ||
1123 | case RF90_PATH_D: | ||
1124 | ; | ||
1125 | break; | ||
1126 | default: | ||
1127 | break; | ||
1128 | } | ||
1129 | |||
1130 | return rtstatus; | ||
1131 | } | ||
1132 | |||
1133 | |||
1134 | bool rtl92s_phy_mac_config(struct ieee80211_hw *hw) | ||
1135 | { | ||
1136 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1137 | u32 i; | ||
1138 | u32 arraylength; | ||
1139 | u32 *ptraArray; | ||
1140 | |||
1141 | arraylength = MAC_2T_ARRAYLENGTH; | ||
1142 | ptraArray = rtl8192semac_2t_array; | ||
1143 | |||
1144 | for (i = 0; i < arraylength; i = i + 2) | ||
1145 | rtl_write_byte(rtlpriv, ptraArray[i], (u8)ptraArray[i + 1]); | ||
1146 | |||
1147 | return true; | ||
1148 | } | ||
1149 | |||
1150 | |||
1151 | bool rtl92s_phy_bb_config(struct ieee80211_hw *hw) | ||
1152 | { | ||
1153 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1154 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1155 | bool rtstatus = true; | ||
1156 | u8 pathmap, index, rf_num = 0; | ||
1157 | u8 path1, path2; | ||
1158 | |||
1159 | _rtl92s_phy_init_register_definition(hw); | ||
1160 | |||
1161 | /* Config BB and AGC */ | ||
1162 | rtstatus = _rtl92s_phy_bb_config_parafile(hw); | ||
1163 | |||
1164 | |||
1165 | /* Check BB/RF confiuration setting. */ | ||
1166 | /* We only need to configure RF which is turned on. */ | ||
1167 | path1 = (u8)(rtl92s_phy_query_bb_reg(hw, RFPGA0_TXINFO, 0xf)); | ||
1168 | mdelay(10); | ||
1169 | path2 = (u8)(rtl92s_phy_query_bb_reg(hw, ROFDM0_TRXPATHENABLE, 0xf)); | ||
1170 | pathmap = path1 | path2; | ||
1171 | |||
1172 | rtlphy->rf_pathmap = pathmap; | ||
1173 | for (index = 0; index < 4; index++) { | ||
1174 | if ((pathmap >> index) & 0x1) | ||
1175 | rf_num++; | ||
1176 | } | ||
1177 | |||
1178 | if ((rtlphy->rf_type == RF_1T1R && rf_num != 1) || | ||
1179 | (rtlphy->rf_type == RF_1T2R && rf_num != 2) || | ||
1180 | (rtlphy->rf_type == RF_2T2R && rf_num != 2) || | ||
1181 | (rtlphy->rf_type == RF_2T2R_GREEN && rf_num != 2)) { | ||
1182 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
1183 | ("RF_Type(%x) does not match " | ||
1184 | "RF_Num(%x)!!\n", rtlphy->rf_type, rf_num)); | ||
1185 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
1186 | ("path1 0x%x, path2 0x%x, pathmap " | ||
1187 | "0x%x\n", path1, path2, pathmap)); | ||
1188 | } | ||
1189 | |||
1190 | return rtstatus; | ||
1191 | } | ||
1192 | |||
1193 | bool rtl92s_phy_rf_config(struct ieee80211_hw *hw) | ||
1194 | { | ||
1195 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1196 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1197 | |||
1198 | /* Initialize general global value */ | ||
1199 | if (rtlphy->rf_type == RF_1T1R) | ||
1200 | rtlphy->num_total_rfpath = 1; | ||
1201 | else | ||
1202 | rtlphy->num_total_rfpath = 2; | ||
1203 | |||
1204 | /* Config BB and RF */ | ||
1205 | return rtl92s_phy_rf6052_config(hw); | ||
1206 | } | ||
1207 | |||
1208 | void rtl92s_phy_get_hw_reg_originalvalue(struct ieee80211_hw *hw) | ||
1209 | { | ||
1210 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1211 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1212 | |||
1213 | /* read rx initial gain */ | ||
1214 | rtlphy->default_initialgain[0] = rtl_get_bbreg(hw, | ||
1215 | ROFDM0_XAAGCCORE1, MASKBYTE0); | ||
1216 | rtlphy->default_initialgain[1] = rtl_get_bbreg(hw, | ||
1217 | ROFDM0_XBAGCCORE1, MASKBYTE0); | ||
1218 | rtlphy->default_initialgain[2] = rtl_get_bbreg(hw, | ||
1219 | ROFDM0_XCAGCCORE1, MASKBYTE0); | ||
1220 | rtlphy->default_initialgain[3] = rtl_get_bbreg(hw, | ||
1221 | ROFDM0_XDAGCCORE1, MASKBYTE0); | ||
1222 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, ("Default initial gain " | ||
1223 | "(c50=0x%x, c58=0x%x, c60=0x%x, c68=0x%x)\n", | ||
1224 | rtlphy->default_initialgain[0], | ||
1225 | rtlphy->default_initialgain[1], | ||
1226 | rtlphy->default_initialgain[2], | ||
1227 | rtlphy->default_initialgain[3])); | ||
1228 | |||
1229 | /* read framesync */ | ||
1230 | rtlphy->framesync = rtl_get_bbreg(hw, ROFDM0_RXDETECTOR3, MASKBYTE0); | ||
1231 | rtlphy->framesync_c34 = rtl_get_bbreg(hw, ROFDM0_RXDETECTOR2, | ||
1232 | MASKDWORD); | ||
1233 | RT_TRACE(rtlpriv, COMP_INIT, DBG_LOUD, | ||
1234 | ("Default framesync (0x%x) = 0x%x\n", | ||
1235 | ROFDM0_RXDETECTOR3, rtlphy->framesync)); | ||
1236 | |||
1237 | } | ||
1238 | |||
1239 | static void _rtl92s_phy_get_txpower_index(struct ieee80211_hw *hw, u8 channel, | ||
1240 | u8 *cckpowerlevel, u8 *ofdmpowerLevel) | ||
1241 | { | ||
1242 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1243 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1244 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1245 | u8 index = (channel - 1); | ||
1246 | |||
1247 | /* 1. CCK */ | ||
1248 | /* RF-A */ | ||
1249 | cckpowerlevel[0] = rtlefuse->txpwrlevel_cck[0][index]; | ||
1250 | /* RF-B */ | ||
1251 | cckpowerlevel[1] = rtlefuse->txpwrlevel_cck[1][index]; | ||
1252 | |||
1253 | /* 2. OFDM for 1T or 2T */ | ||
1254 | if (rtlphy->rf_type == RF_1T2R || rtlphy->rf_type == RF_1T1R) { | ||
1255 | /* Read HT 40 OFDM TX power */ | ||
1256 | ofdmpowerLevel[0] = rtlefuse->txpwrlevel_ht40_1s[0][index]; | ||
1257 | ofdmpowerLevel[1] = rtlefuse->txpwrlevel_ht40_1s[1][index]; | ||
1258 | } else if (rtlphy->rf_type == RF_2T2R) { | ||
1259 | /* Read HT 40 OFDM TX power */ | ||
1260 | ofdmpowerLevel[0] = rtlefuse->txpwrlevel_ht40_2s[0][index]; | ||
1261 | ofdmpowerLevel[1] = rtlefuse->txpwrlevel_ht40_2s[1][index]; | ||
1262 | } | ||
1263 | } | ||
1264 | |||
1265 | static void _rtl92s_phy_ccxpower_indexcheck(struct ieee80211_hw *hw, | ||
1266 | u8 channel, u8 *cckpowerlevel, u8 *ofdmpowerlevel) | ||
1267 | { | ||
1268 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1269 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1270 | |||
1271 | rtlphy->cur_cck_txpwridx = cckpowerlevel[0]; | ||
1272 | rtlphy->cur_ofdm24g_txpwridx = ofdmpowerlevel[0]; | ||
1273 | } | ||
1274 | |||
1275 | void rtl92s_phy_set_txpower(struct ieee80211_hw *hw, u8 channel) | ||
1276 | { | ||
1277 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1278 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1279 | /* [0]:RF-A, [1]:RF-B */ | ||
1280 | u8 cckpowerlevel[2], ofdmpowerLevel[2]; | ||
1281 | |||
1282 | if (rtlefuse->txpwr_fromeprom == false) | ||
1283 | return; | ||
1284 | |||
1285 | /* Mainly we use RF-A Tx Power to write the Tx Power registers, | ||
1286 | * but the RF-B Tx Power must be calculated by the antenna diff. | ||
1287 | * So we have to rewrite Antenna gain offset register here. | ||
1288 | * Please refer to BB register 0x80c | ||
1289 | * 1. For CCK. | ||
1290 | * 2. For OFDM 1T or 2T */ | ||
1291 | _rtl92s_phy_get_txpower_index(hw, channel, &cckpowerlevel[0], | ||
1292 | &ofdmpowerLevel[0]); | ||
1293 | |||
1294 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
1295 | ("Channel-%d, cckPowerLevel (A / B) = " | ||
1296 | "0x%x / 0x%x, ofdmPowerLevel (A / B) = 0x%x / 0x%x\n", | ||
1297 | channel, cckpowerlevel[0], cckpowerlevel[1], | ||
1298 | ofdmpowerLevel[0], ofdmpowerLevel[1])); | ||
1299 | |||
1300 | _rtl92s_phy_ccxpower_indexcheck(hw, channel, &cckpowerlevel[0], | ||
1301 | &ofdmpowerLevel[0]); | ||
1302 | |||
1303 | rtl92s_phy_rf6052_set_ccktxpower(hw, cckpowerlevel[0]); | ||
1304 | rtl92s_phy_rf6052_set_ofdmtxpower(hw, &ofdmpowerLevel[0], channel); | ||
1305 | |||
1306 | } | ||
1307 | |||
1308 | void rtl92s_phy_chk_fwcmd_iodone(struct ieee80211_hw *hw) | ||
1309 | { | ||
1310 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1311 | u16 pollingcnt = 10000; | ||
1312 | u32 tmpvalue; | ||
1313 | |||
1314 | /* Make sure that CMD IO has be accepted by FW. */ | ||
1315 | do { | ||
1316 | udelay(10); | ||
1317 | |||
1318 | tmpvalue = rtl_read_dword(rtlpriv, WFM5); | ||
1319 | if (tmpvalue == 0) | ||
1320 | break; | ||
1321 | } while (--pollingcnt); | ||
1322 | |||
1323 | if (pollingcnt == 0) | ||
1324 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, ("Set FW Cmd fail!!\n")); | ||
1325 | } | ||
1326 | |||
1327 | |||
1328 | static void _rtl92s_phy_set_fwcmd_io(struct ieee80211_hw *hw) | ||
1329 | { | ||
1330 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1331 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1332 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
1333 | u32 input, current_aid = 0; | ||
1334 | |||
1335 | if (is_hal_stop(rtlhal)) | ||
1336 | return; | ||
1337 | |||
1338 | /* We re-map RA related CMD IO to combinational ones */ | ||
1339 | /* if FW version is v.52 or later. */ | ||
1340 | switch (rtlhal->current_fwcmd_io) { | ||
1341 | case FW_CMD_RA_REFRESH_N: | ||
1342 | rtlhal->current_fwcmd_io = FW_CMD_RA_REFRESH_N_COMB; | ||
1343 | break; | ||
1344 | case FW_CMD_RA_REFRESH_BG: | ||
1345 | rtlhal->current_fwcmd_io = FW_CMD_RA_REFRESH_BG_COMB; | ||
1346 | break; | ||
1347 | default: | ||
1348 | break; | ||
1349 | } | ||
1350 | |||
1351 | switch (rtlhal->current_fwcmd_io) { | ||
1352 | case FW_CMD_RA_RESET: | ||
1353 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1354 | ("FW_CMD_RA_RESET\n")); | ||
1355 | rtl_write_dword(rtlpriv, WFM5, FW_RA_RESET); | ||
1356 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1357 | break; | ||
1358 | case FW_CMD_RA_ACTIVE: | ||
1359 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1360 | ("FW_CMD_RA_ACTIVE\n")); | ||
1361 | rtl_write_dword(rtlpriv, WFM5, FW_RA_ACTIVE); | ||
1362 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1363 | break; | ||
1364 | case FW_CMD_RA_REFRESH_N: | ||
1365 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1366 | ("FW_CMD_RA_REFRESH_N\n")); | ||
1367 | input = FW_RA_REFRESH; | ||
1368 | rtl_write_dword(rtlpriv, WFM5, input); | ||
1369 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1370 | rtl_write_dword(rtlpriv, WFM5, FW_RA_ENABLE_RSSI_MASK); | ||
1371 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1372 | break; | ||
1373 | case FW_CMD_RA_REFRESH_BG: | ||
1374 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1375 | ("FW_CMD_RA_REFRESH_BG\n")); | ||
1376 | rtl_write_dword(rtlpriv, WFM5, FW_RA_REFRESH); | ||
1377 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1378 | rtl_write_dword(rtlpriv, WFM5, FW_RA_DISABLE_RSSI_MASK); | ||
1379 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1380 | break; | ||
1381 | case FW_CMD_RA_REFRESH_N_COMB: | ||
1382 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1383 | ("FW_CMD_RA_REFRESH_N_COMB\n")); | ||
1384 | input = FW_RA_IOT_N_COMB; | ||
1385 | rtl_write_dword(rtlpriv, WFM5, input); | ||
1386 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1387 | break; | ||
1388 | case FW_CMD_RA_REFRESH_BG_COMB: | ||
1389 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1390 | ("FW_CMD_RA_REFRESH_BG_COMB\n")); | ||
1391 | input = FW_RA_IOT_BG_COMB; | ||
1392 | rtl_write_dword(rtlpriv, WFM5, input); | ||
1393 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1394 | break; | ||
1395 | case FW_CMD_IQK_ENABLE: | ||
1396 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1397 | ("FW_CMD_IQK_ENABLE\n")); | ||
1398 | rtl_write_dword(rtlpriv, WFM5, FW_IQK_ENABLE); | ||
1399 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1400 | break; | ||
1401 | case FW_CMD_PAUSE_DM_BY_SCAN: | ||
1402 | /* Lower initial gain */ | ||
1403 | rtl_set_bbreg(hw, ROFDM0_XAAGCCORE1, MASKBYTE0, 0x17); | ||
1404 | rtl_set_bbreg(hw, ROFDM0_XBAGCCORE1, MASKBYTE0, 0x17); | ||
1405 | /* CCA threshold */ | ||
1406 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, 0x40); | ||
1407 | break; | ||
1408 | case FW_CMD_RESUME_DM_BY_SCAN: | ||
1409 | /* CCA threshold */ | ||
1410 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, 0xcd); | ||
1411 | rtl92s_phy_set_txpower(hw, rtlphy->current_channel); | ||
1412 | break; | ||
1413 | case FW_CMD_HIGH_PWR_DISABLE: | ||
1414 | if (rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) | ||
1415 | break; | ||
1416 | |||
1417 | /* Lower initial gain */ | ||
1418 | rtl_set_bbreg(hw, ROFDM0_XAAGCCORE1, MASKBYTE0, 0x17); | ||
1419 | rtl_set_bbreg(hw, ROFDM0_XBAGCCORE1, MASKBYTE0, 0x17); | ||
1420 | /* CCA threshold */ | ||
1421 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, 0x40); | ||
1422 | break; | ||
1423 | case FW_CMD_HIGH_PWR_ENABLE: | ||
1424 | if ((rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) || | ||
1425 | (rtlpriv->dm.dynamic_txpower_enable == true)) | ||
1426 | break; | ||
1427 | |||
1428 | /* CCA threshold */ | ||
1429 | rtl_set_bbreg(hw, RCCK0_CCA, MASKBYTE2, 0xcd); | ||
1430 | break; | ||
1431 | case FW_CMD_LPS_ENTER: | ||
1432 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1433 | ("FW_CMD_LPS_ENTER\n")); | ||
1434 | current_aid = rtlpriv->mac80211.assoc_id; | ||
1435 | rtl_write_dword(rtlpriv, WFM5, (FW_LPS_ENTER | | ||
1436 | ((current_aid | 0xc000) << 8))); | ||
1437 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1438 | /* FW set TXOP disable here, so disable EDCA | ||
1439 | * turbo mode until driver leave LPS */ | ||
1440 | break; | ||
1441 | case FW_CMD_LPS_LEAVE: | ||
1442 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1443 | ("FW_CMD_LPS_LEAVE\n")); | ||
1444 | rtl_write_dword(rtlpriv, WFM5, FW_LPS_LEAVE); | ||
1445 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1446 | break; | ||
1447 | case FW_CMD_ADD_A2_ENTRY: | ||
1448 | RT_TRACE(rtlpriv, COMP_CMD, DBG_DMESG, | ||
1449 | ("FW_CMD_ADD_A2_ENTRY\n")); | ||
1450 | rtl_write_dword(rtlpriv, WFM5, FW_ADD_A2_ENTRY); | ||
1451 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1452 | break; | ||
1453 | case FW_CMD_CTRL_DM_BY_DRIVER: | ||
1454 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1455 | ("FW_CMD_CTRL_DM_BY_DRIVER\n")); | ||
1456 | rtl_write_dword(rtlpriv, WFM5, FW_CTRL_DM_BY_DRIVER); | ||
1457 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1458 | break; | ||
1459 | |||
1460 | default: | ||
1461 | break; | ||
1462 | } | ||
1463 | |||
1464 | rtl92s_phy_chk_fwcmd_iodone(hw); | ||
1465 | |||
1466 | /* Clear FW CMD operation flag. */ | ||
1467 | rtlhal->set_fwcmd_inprogress = false; | ||
1468 | } | ||
1469 | |||
1470 | bool rtl92s_phy_set_fw_cmd(struct ieee80211_hw *hw, enum fwcmd_iotype fw_cmdio) | ||
1471 | { | ||
1472 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1473 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
1474 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
1475 | u32 fw_param = FW_CMD_IO_PARA_QUERY(rtlpriv); | ||
1476 | u16 fw_cmdmap = FW_CMD_IO_QUERY(rtlpriv); | ||
1477 | bool bPostProcessing = false; | ||
1478 | |||
1479 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1480 | ("Set FW Cmd(%#x), set_fwcmd_inprogress(%d)\n", | ||
1481 | fw_cmdio, rtlhal->set_fwcmd_inprogress)); | ||
1482 | |||
1483 | do { | ||
1484 | /* We re-map to combined FW CMD ones if firmware version */ | ||
1485 | /* is v.53 or later. */ | ||
1486 | switch (fw_cmdio) { | ||
1487 | case FW_CMD_RA_REFRESH_N: | ||
1488 | fw_cmdio = FW_CMD_RA_REFRESH_N_COMB; | ||
1489 | break; | ||
1490 | case FW_CMD_RA_REFRESH_BG: | ||
1491 | fw_cmdio = FW_CMD_RA_REFRESH_BG_COMB; | ||
1492 | break; | ||
1493 | default: | ||
1494 | break; | ||
1495 | } | ||
1496 | |||
1497 | /* If firmware version is v.62 or later, | ||
1498 | * use FW_CMD_IO_SET for FW_CMD_CTRL_DM_BY_DRIVER */ | ||
1499 | if (hal_get_firmwareversion(rtlpriv) >= 0x3E) { | ||
1500 | if (fw_cmdio == FW_CMD_CTRL_DM_BY_DRIVER) | ||
1501 | fw_cmdio = FW_CMD_CTRL_DM_BY_DRIVER_NEW; | ||
1502 | } | ||
1503 | |||
1504 | |||
1505 | /* We shall revise all FW Cmd IO into Reg0x364 | ||
1506 | * DM map table in the future. */ | ||
1507 | switch (fw_cmdio) { | ||
1508 | case FW_CMD_RA_INIT: | ||
1509 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, ("RA init!!\n")); | ||
1510 | fw_cmdmap |= FW_RA_INIT_CTL; | ||
1511 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1512 | /* Clear control flag to sync with FW. */ | ||
1513 | FW_CMD_IO_CLR(rtlpriv, FW_RA_INIT_CTL); | ||
1514 | break; | ||
1515 | case FW_CMD_DIG_DISABLE: | ||
1516 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1517 | ("Set DIG disable!!\n")); | ||
1518 | fw_cmdmap &= ~FW_DIG_ENABLE_CTL; | ||
1519 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1520 | break; | ||
1521 | case FW_CMD_DIG_ENABLE: | ||
1522 | case FW_CMD_DIG_RESUME: | ||
1523 | if (!(rtlpriv->dm.dm_flag & HAL_DM_DIG_DISABLE)) { | ||
1524 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1525 | ("Set DIG enable or resume!!\n")); | ||
1526 | fw_cmdmap |= (FW_DIG_ENABLE_CTL | FW_SS_CTL); | ||
1527 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1528 | } | ||
1529 | break; | ||
1530 | case FW_CMD_DIG_HALT: | ||
1531 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1532 | ("Set DIG halt!!\n")); | ||
1533 | fw_cmdmap &= ~(FW_DIG_ENABLE_CTL | FW_SS_CTL); | ||
1534 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1535 | break; | ||
1536 | case FW_CMD_TXPWR_TRACK_THERMAL: { | ||
1537 | u8 thermalval = 0; | ||
1538 | fw_cmdmap |= FW_PWR_TRK_CTL; | ||
1539 | |||
1540 | /* Clear FW parameter in terms of thermal parts. */ | ||
1541 | fw_param &= FW_PWR_TRK_PARAM_CLR; | ||
1542 | |||
1543 | thermalval = rtlpriv->dm.thermalvalue; | ||
1544 | fw_param |= ((thermalval << 24) | | ||
1545 | (rtlefuse->thermalmeter[0] << 16)); | ||
1546 | |||
1547 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1548 | ("Set TxPwr tracking!! " | ||
1549 | "FwCmdMap(%#x), FwParam(%#x)\n", | ||
1550 | fw_cmdmap, fw_param)); | ||
1551 | |||
1552 | FW_CMD_PARA_SET(rtlpriv, fw_param); | ||
1553 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1554 | |||
1555 | /* Clear control flag to sync with FW. */ | ||
1556 | FW_CMD_IO_CLR(rtlpriv, FW_PWR_TRK_CTL); | ||
1557 | } | ||
1558 | break; | ||
1559 | /* The following FW CMDs are only compatible to | ||
1560 | * v.53 or later. */ | ||
1561 | case FW_CMD_RA_REFRESH_N_COMB: | ||
1562 | fw_cmdmap |= FW_RA_N_CTL; | ||
1563 | |||
1564 | /* Clear RA BG mode control. */ | ||
1565 | fw_cmdmap &= ~(FW_RA_BG_CTL | FW_RA_INIT_CTL); | ||
1566 | |||
1567 | /* Clear FW parameter in terms of RA parts. */ | ||
1568 | fw_param &= FW_RA_PARAM_CLR; | ||
1569 | |||
1570 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1571 | ("[FW CMD] [New Version] " | ||
1572 | "Set RA/IOT Comb in n mode!! FwCmdMap(%#x), " | ||
1573 | "FwParam(%#x)\n", fw_cmdmap, fw_param)); | ||
1574 | |||
1575 | FW_CMD_PARA_SET(rtlpriv, fw_param); | ||
1576 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1577 | |||
1578 | /* Clear control flag to sync with FW. */ | ||
1579 | FW_CMD_IO_CLR(rtlpriv, FW_RA_N_CTL); | ||
1580 | break; | ||
1581 | case FW_CMD_RA_REFRESH_BG_COMB: | ||
1582 | fw_cmdmap |= FW_RA_BG_CTL; | ||
1583 | |||
1584 | /* Clear RA n-mode control. */ | ||
1585 | fw_cmdmap &= ~(FW_RA_N_CTL | FW_RA_INIT_CTL); | ||
1586 | /* Clear FW parameter in terms of RA parts. */ | ||
1587 | fw_param &= FW_RA_PARAM_CLR; | ||
1588 | |||
1589 | FW_CMD_PARA_SET(rtlpriv, fw_param); | ||
1590 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1591 | |||
1592 | /* Clear control flag to sync with FW. */ | ||
1593 | FW_CMD_IO_CLR(rtlpriv, FW_RA_BG_CTL); | ||
1594 | break; | ||
1595 | case FW_CMD_IQK_ENABLE: | ||
1596 | fw_cmdmap |= FW_IQK_CTL; | ||
1597 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1598 | /* Clear control flag to sync with FW. */ | ||
1599 | FW_CMD_IO_CLR(rtlpriv, FW_IQK_CTL); | ||
1600 | break; | ||
1601 | /* The following FW CMD is compatible to v.62 or later. */ | ||
1602 | case FW_CMD_CTRL_DM_BY_DRIVER_NEW: | ||
1603 | fw_cmdmap |= FW_DRIVER_CTRL_DM_CTL; | ||
1604 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1605 | break; | ||
1606 | /* The followed FW Cmds needs post-processing later. */ | ||
1607 | case FW_CMD_RESUME_DM_BY_SCAN: | ||
1608 | fw_cmdmap |= (FW_DIG_ENABLE_CTL | | ||
1609 | FW_HIGH_PWR_ENABLE_CTL | | ||
1610 | FW_SS_CTL); | ||
1611 | |||
1612 | if (rtlpriv->dm.dm_flag & HAL_DM_DIG_DISABLE || | ||
1613 | !digtable.dig_enable_flag) | ||
1614 | fw_cmdmap &= ~FW_DIG_ENABLE_CTL; | ||
1615 | |||
1616 | if ((rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) || | ||
1617 | (rtlpriv->dm.dynamic_txpower_enable == true)) | ||
1618 | fw_cmdmap &= ~FW_HIGH_PWR_ENABLE_CTL; | ||
1619 | |||
1620 | if ((digtable.dig_ext_port_stage == | ||
1621 | DIG_EXT_PORT_STAGE_0) || | ||
1622 | (digtable.dig_ext_port_stage == | ||
1623 | DIG_EXT_PORT_STAGE_1)) | ||
1624 | fw_cmdmap &= ~FW_DIG_ENABLE_CTL; | ||
1625 | |||
1626 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1627 | bPostProcessing = true; | ||
1628 | break; | ||
1629 | case FW_CMD_PAUSE_DM_BY_SCAN: | ||
1630 | fw_cmdmap &= ~(FW_DIG_ENABLE_CTL | | ||
1631 | FW_HIGH_PWR_ENABLE_CTL | | ||
1632 | FW_SS_CTL); | ||
1633 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1634 | bPostProcessing = true; | ||
1635 | break; | ||
1636 | case FW_CMD_HIGH_PWR_DISABLE: | ||
1637 | fw_cmdmap &= ~FW_HIGH_PWR_ENABLE_CTL; | ||
1638 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1639 | bPostProcessing = true; | ||
1640 | break; | ||
1641 | case FW_CMD_HIGH_PWR_ENABLE: | ||
1642 | if (!(rtlpriv->dm.dm_flag & HAL_DM_HIPWR_DISABLE) && | ||
1643 | (rtlpriv->dm.dynamic_txpower_enable != true)) { | ||
1644 | fw_cmdmap |= (FW_HIGH_PWR_ENABLE_CTL | | ||
1645 | FW_SS_CTL); | ||
1646 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1647 | bPostProcessing = true; | ||
1648 | } | ||
1649 | break; | ||
1650 | case FW_CMD_DIG_MODE_FA: | ||
1651 | fw_cmdmap |= FW_FA_CTL; | ||
1652 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1653 | break; | ||
1654 | case FW_CMD_DIG_MODE_SS: | ||
1655 | fw_cmdmap &= ~FW_FA_CTL; | ||
1656 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1657 | break; | ||
1658 | case FW_CMD_PAPE_CONTROL: | ||
1659 | RT_TRACE(rtlpriv, COMP_CMD, DBG_LOUD, | ||
1660 | ("[FW CMD] Set PAPE Control\n")); | ||
1661 | fw_cmdmap &= ~FW_PAPE_CTL_BY_SW_HW; | ||
1662 | |||
1663 | FW_CMD_IO_SET(rtlpriv, fw_cmdmap); | ||
1664 | break; | ||
1665 | default: | ||
1666 | /* Pass to original FW CMD processing callback | ||
1667 | * routine. */ | ||
1668 | bPostProcessing = true; | ||
1669 | break; | ||
1670 | } | ||
1671 | } while (false); | ||
1672 | |||
1673 | /* We shall post processing these FW CMD if | ||
1674 | * variable bPostProcessing is set. */ | ||
1675 | if (bPostProcessing && !rtlhal->set_fwcmd_inprogress) { | ||
1676 | rtlhal->set_fwcmd_inprogress = true; | ||
1677 | /* Update current FW Cmd for callback use. */ | ||
1678 | rtlhal->current_fwcmd_io = fw_cmdio; | ||
1679 | } else { | ||
1680 | return false; | ||
1681 | } | ||
1682 | |||
1683 | _rtl92s_phy_set_fwcmd_io(hw); | ||
1684 | return true; | ||
1685 | } | ||
1686 | |||
1687 | static void _rtl92s_phy_check_ephy_switchready(struct ieee80211_hw *hw) | ||
1688 | { | ||
1689 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1690 | u32 delay = 100; | ||
1691 | u8 regu1; | ||
1692 | |||
1693 | regu1 = rtl_read_byte(rtlpriv, 0x554); | ||
1694 | while ((regu1 & BIT(5)) && (delay > 0)) { | ||
1695 | regu1 = rtl_read_byte(rtlpriv, 0x554); | ||
1696 | delay--; | ||
1697 | /* We delay only 50us to prevent | ||
1698 | * being scheduled out. */ | ||
1699 | udelay(50); | ||
1700 | } | ||
1701 | } | ||
1702 | |||
1703 | void rtl92s_phy_switch_ephy_parameter(struct ieee80211_hw *hw) | ||
1704 | { | ||
1705 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1706 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
1707 | |||
1708 | /* The way to be capable to switch clock request | ||
1709 | * when the PG setting does not support clock request. | ||
1710 | * This is the backdoor solution to switch clock | ||
1711 | * request before ASPM or D3. */ | ||
1712 | rtl_write_dword(rtlpriv, 0x540, 0x73c11); | ||
1713 | rtl_write_dword(rtlpriv, 0x548, 0x2407c); | ||
1714 | |||
1715 | /* Switch EPHY parameter!!!! */ | ||
1716 | rtl_write_word(rtlpriv, 0x550, 0x1000); | ||
1717 | rtl_write_byte(rtlpriv, 0x554, 0x20); | ||
1718 | _rtl92s_phy_check_ephy_switchready(hw); | ||
1719 | |||
1720 | rtl_write_word(rtlpriv, 0x550, 0xa0eb); | ||
1721 | rtl_write_byte(rtlpriv, 0x554, 0x3e); | ||
1722 | _rtl92s_phy_check_ephy_switchready(hw); | ||
1723 | |||
1724 | rtl_write_word(rtlpriv, 0x550, 0xff80); | ||
1725 | rtl_write_byte(rtlpriv, 0x554, 0x39); | ||
1726 | _rtl92s_phy_check_ephy_switchready(hw); | ||
1727 | |||
1728 | /* Delay L1 enter time */ | ||
1729 | if (ppsc->support_aspm && !ppsc->support_backdoor) | ||
1730 | rtl_write_byte(rtlpriv, 0x560, 0x40); | ||
1731 | else | ||
1732 | rtl_write_byte(rtlpriv, 0x560, 0x00); | ||
1733 | |||
1734 | } | ||
1735 | |||
1736 | void rtl92s_phy_set_beacon_hwreg(struct ieee80211_hw *hw, u16 BeaconInterval) | ||
1737 | { | ||
1738 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1739 | rtl_write_dword(rtlpriv, WFM5, 0xF1000000 | (BeaconInterval << 8)); | ||
1740 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/phy.h b/drivers/net/wireless/rtlwifi/rtl8192se/phy.h new file mode 100644 index 000000000000..37e504af6446 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/phy.h | |||
@@ -0,0 +1,101 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __RTL92S_PHY_H__ | ||
30 | #define __RTL92S_PHY_H__ | ||
31 | |||
32 | #define MAX_TXPWR_IDX_NMODE_92S 63 | ||
33 | #define MAX_DOZE_WAITING_TIMES_9x 64 | ||
34 | |||
35 | /* Channel switch:The size of | ||
36 | * command tables for switch channel */ | ||
37 | #define MAX_PRECMD_CNT 16 | ||
38 | #define MAX_RFDEPENDCMD_CNT 16 | ||
39 | #define MAX_POSTCMD_CNT 16 | ||
40 | |||
41 | #define RF90_PATH_MAX 4 | ||
42 | |||
43 | enum version_8192s { | ||
44 | VERSION_8192S_ACUT, | ||
45 | VERSION_8192S_BCUT, | ||
46 | VERSION_8192S_CCUT | ||
47 | }; | ||
48 | |||
49 | enum swchnlcmd_id { | ||
50 | CMDID_END, | ||
51 | CMDID_SET_TXPOWEROWER_LEVEL, | ||
52 | CMDID_BBREGWRITE10, | ||
53 | CMDID_WRITEPORT_ULONG, | ||
54 | CMDID_WRITEPORT_USHORT, | ||
55 | CMDID_WRITEPORT_UCHAR, | ||
56 | CMDID_RF_WRITEREG, | ||
57 | }; | ||
58 | |||
59 | struct swchnlcmd { | ||
60 | enum swchnlcmd_id cmdid; | ||
61 | u32 para1; | ||
62 | u32 para2; | ||
63 | u32 msdelay; | ||
64 | }; | ||
65 | |||
66 | enum baseband_config_type { | ||
67 | /* Radio Path A */ | ||
68 | BASEBAND_CONFIG_PHY_REG = 0, | ||
69 | /* Radio Path B */ | ||
70 | BASEBAND_CONFIG_AGC_TAB = 1, | ||
71 | }; | ||
72 | |||
73 | #define hal_get_firmwareversion(rtlpriv) \ | ||
74 | (((struct rt_firmware *)(rtlpriv->rtlhal.pfirmware))->firmwareversion) | ||
75 | |||
76 | u32 rtl92s_phy_query_bb_reg(struct ieee80211_hw *hw, u32 regaddr, u32 bitmask); | ||
77 | void rtl92s_phy_set_bb_reg(struct ieee80211_hw *hw, u32 regaddr, u32 bitmask, | ||
78 | u32 data); | ||
79 | void rtl92s_phy_scan_operation_backup(struct ieee80211_hw *hw, u8 operation); | ||
80 | u32 rtl92s_phy_query_rf_reg(struct ieee80211_hw *hw, enum radio_path rfpath, | ||
81 | u32 regaddr, u32 bitmask); | ||
82 | void rtl92s_phy_set_rf_reg(struct ieee80211_hw *hw, enum radio_path rfpath, | ||
83 | u32 regaddr, u32 bitmask, u32 data); | ||
84 | void rtl92s_phy_set_bw_mode(struct ieee80211_hw *hw, | ||
85 | enum nl80211_channel_type ch_type); | ||
86 | u8 rtl92s_phy_sw_chnl(struct ieee80211_hw *hw); | ||
87 | bool rtl92s_phy_set_rf_power_state(struct ieee80211_hw *hw, | ||
88 | enum rf_pwrstate rfpower_state); | ||
89 | bool rtl92s_phy_mac_config(struct ieee80211_hw *hw); | ||
90 | void rtl92s_phy_switch_ephy_parameter(struct ieee80211_hw *hw); | ||
91 | bool rtl92s_phy_bb_config(struct ieee80211_hw *hw); | ||
92 | bool rtl92s_phy_rf_config(struct ieee80211_hw *hw); | ||
93 | void rtl92s_phy_get_hw_reg_originalvalue(struct ieee80211_hw *hw); | ||
94 | void rtl92s_phy_set_txpower(struct ieee80211_hw *hw, u8 channel); | ||
95 | bool rtl92s_phy_set_fw_cmd(struct ieee80211_hw *hw, enum fwcmd_iotype fwcmd_io); | ||
96 | void rtl92s_phy_chk_fwcmd_iodone(struct ieee80211_hw *hw); | ||
97 | void rtl92s_phy_set_beacon_hwreg(struct ieee80211_hw *hw, u16 beaconinterval); | ||
98 | u8 rtl92s_phy_config_rf(struct ieee80211_hw *hw, enum radio_path rfpath) ; | ||
99 | |||
100 | #endif | ||
101 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/reg.h b/drivers/net/wireless/rtlwifi/rtl8192se/reg.h new file mode 100644 index 000000000000..0116eaddbfac --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/reg.h | |||
@@ -0,0 +1,1188 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_92S_REG_H__ | ||
30 | #define __REALTEK_92S_REG_H__ | ||
31 | |||
32 | /* 1. System Configuration Registers */ | ||
33 | #define REG_SYS_ISO_CTRL 0x0000 | ||
34 | #define REG_SYS_FUNC_EN 0x0002 | ||
35 | #define PMC_FSM 0x0004 | ||
36 | #define SYS_CLKR 0x0008 | ||
37 | #define EPROM_CMD 0x000A | ||
38 | #define EE_VPD 0x000C | ||
39 | #define AFE_MISC 0x0010 | ||
40 | #define SPS0_CTRL 0x0011 | ||
41 | #define SPS1_CTRL 0x0018 | ||
42 | #define RF_CTRL 0x001F | ||
43 | #define LDOA15_CTRL 0x0020 | ||
44 | #define LDOV12D_CTRL 0x0021 | ||
45 | #define LDOHCI12_CTRL 0x0022 | ||
46 | #define LDO_USB_SDIO 0x0023 | ||
47 | #define LPLDO_CTRL 0x0024 | ||
48 | #define AFE_XTAL_CTRL 0x0026 | ||
49 | #define AFE_PLL_CTRL 0x0028 | ||
50 | #define REG_EFUSE_CTRL 0x0030 | ||
51 | #define REG_EFUSE_TEST 0x0034 | ||
52 | #define PWR_DATA 0x0038 | ||
53 | #define DBG_PORT 0x003A | ||
54 | #define DPS_TIMER 0x003C | ||
55 | #define RCLK_MON 0x003E | ||
56 | |||
57 | /* 2. Command Control Registers */ | ||
58 | #define CMDR 0x0040 | ||
59 | #define TXPAUSE 0x0042 | ||
60 | #define LBKMD_SEL 0x0043 | ||
61 | #define TCR 0x0044 | ||
62 | #define RCR 0x0048 | ||
63 | #define MSR 0x004C | ||
64 | #define SYSF_CFG 0x004D | ||
65 | #define RX_PKY_LIMIT 0x004E | ||
66 | #define MBIDCTRL 0x004F | ||
67 | |||
68 | /* 3. MACID Setting Registers */ | ||
69 | #define MACIDR 0x0050 | ||
70 | #define MACIDR0 0x0050 | ||
71 | #define MACIDR4 0x0054 | ||
72 | #define BSSIDR 0x0058 | ||
73 | #define HWVID 0x005E | ||
74 | #define MAR 0x0060 | ||
75 | #define MBIDCAMCONTENT 0x0068 | ||
76 | #define MBIDCAMCFG 0x0070 | ||
77 | #define BUILDTIME 0x0074 | ||
78 | #define BUILDUSER 0x0078 | ||
79 | |||
80 | #define IDR0 MACIDR0 | ||
81 | #define IDR4 MACIDR4 | ||
82 | |||
83 | /* 4. Timing Control Registers */ | ||
84 | #define TSFR 0x0080 | ||
85 | #define SLOT_TIME 0x0089 | ||
86 | #define USTIME 0x008A | ||
87 | #define SIFS_CCK 0x008C | ||
88 | #define SIFS_OFDM 0x008E | ||
89 | #define PIFS_TIME 0x0090 | ||
90 | #define ACK_TIMEOUT 0x0091 | ||
91 | #define EIFSTR 0x0092 | ||
92 | #define BCN_INTERVAL 0x0094 | ||
93 | #define ATIMWND 0x0096 | ||
94 | #define BCN_DRV_EARLY_INT 0x0098 | ||
95 | #define BCN_DMATIME 0x009A | ||
96 | #define BCN_ERR_THRESH 0x009C | ||
97 | #define MLT 0x009D | ||
98 | #define RSVD_MAC_TUNE_US 0x009E | ||
99 | |||
100 | /* 5. FIFO Control Registers */ | ||
101 | #define RQPN 0x00A0 | ||
102 | #define RQPN1 0x00A0 | ||
103 | #define RQPN2 0x00A1 | ||
104 | #define RQPN3 0x00A2 | ||
105 | #define RQPN4 0x00A3 | ||
106 | #define RQPN5 0x00A4 | ||
107 | #define RQPN6 0x00A5 | ||
108 | #define RQPN7 0x00A6 | ||
109 | #define RQPN8 0x00A7 | ||
110 | #define RQPN9 0x00A8 | ||
111 | #define RQPN10 0x00A9 | ||
112 | #define LD_RQPN 0x00AB | ||
113 | #define RXFF_BNDY 0x00AC | ||
114 | #define RXRPT_BNDY 0x00B0 | ||
115 | #define TXPKTBUF_PGBNDY 0x00B4 | ||
116 | #define PBP 0x00B5 | ||
117 | #define RXDRVINFO_SZ 0x00B6 | ||
118 | #define TXFF_STATUS 0x00B7 | ||
119 | #define RXFF_STATUS 0x00B8 | ||
120 | #define TXFF_EMPTY_TH 0x00B9 | ||
121 | #define SDIO_RX_BLKSZ 0x00BC | ||
122 | #define RXDMA 0x00BD | ||
123 | #define RXPKT_NUM 0x00BE | ||
124 | #define C2HCMD_UDT_SIZE 0x00C0 | ||
125 | #define C2HCMD_UDT_ADDR 0x00C2 | ||
126 | #define FIFOPAGE1 0x00C4 | ||
127 | #define FIFOPAGE2 0x00C8 | ||
128 | #define FIFOPAGE3 0x00CC | ||
129 | #define FIFOPAGE4 0x00D0 | ||
130 | #define FIFOPAGE5 0x00D4 | ||
131 | #define FW_RSVD_PG_CRTL 0x00D8 | ||
132 | #define RXDMA_AGG_PG_TH 0x00D9 | ||
133 | #define TXDESC_MSK 0x00DC | ||
134 | #define TXRPTFF_RDPTR 0x00E0 | ||
135 | #define TXRPTFF_WTPTR 0x00E4 | ||
136 | #define C2HFF_RDPTR 0x00E8 | ||
137 | #define C2HFF_WTPTR 0x00EC | ||
138 | #define RXFF0_RDPTR 0x00F0 | ||
139 | #define RXFF0_WTPTR 0x00F4 | ||
140 | #define RXFF1_RDPTR 0x00F8 | ||
141 | #define RXFF1_WTPTR 0x00FC | ||
142 | #define RXRPT0_RDPTR 0x0100 | ||
143 | #define RXRPT0_WTPTR 0x0104 | ||
144 | #define RXRPT1_RDPTR 0x0108 | ||
145 | #define RXRPT1_WTPTR 0x010C | ||
146 | #define RX0_UDT_SIZE 0x0110 | ||
147 | #define RX1PKTNUM 0x0114 | ||
148 | #define RXFILTERMAP 0x0116 | ||
149 | #define RXFILTERMAP_GP1 0x0118 | ||
150 | #define RXFILTERMAP_GP2 0x011A | ||
151 | #define RXFILTERMAP_GP3 0x011C | ||
152 | #define BCNQ_CTRL 0x0120 | ||
153 | #define MGTQ_CTRL 0x0124 | ||
154 | #define HIQ_CTRL 0x0128 | ||
155 | #define VOTID7_CTRL 0x012c | ||
156 | #define VOTID6_CTRL 0x0130 | ||
157 | #define VITID5_CTRL 0x0134 | ||
158 | #define VITID4_CTRL 0x0138 | ||
159 | #define BETID3_CTRL 0x013c | ||
160 | #define BETID0_CTRL 0x0140 | ||
161 | #define BKTID2_CTRL 0x0144 | ||
162 | #define BKTID1_CTRL 0x0148 | ||
163 | #define CMDQ_CTRL 0x014c | ||
164 | #define TXPKT_NUM_CTRL 0x0150 | ||
165 | #define TXQ_PGADD 0x0152 | ||
166 | #define TXFF_PG_NUM 0x0154 | ||
167 | #define TRXDMA_STATUS 0x0156 | ||
168 | |||
169 | /* 6. Adaptive Control Registers */ | ||
170 | #define INIMCS_SEL 0x0160 | ||
171 | #define TX_RATE_REG INIMCS_SEL | ||
172 | #define INIRTSMCS_SEL 0x0180 | ||
173 | #define RRSR 0x0181 | ||
174 | #define ARFR0 0x0184 | ||
175 | #define ARFR1 0x0188 | ||
176 | #define ARFR2 0x018C | ||
177 | #define ARFR3 0x0190 | ||
178 | #define ARFR4 0x0194 | ||
179 | #define ARFR5 0x0198 | ||
180 | #define ARFR6 0x019C | ||
181 | #define ARFR7 0x01A0 | ||
182 | #define AGGLEN_LMT_H 0x01A7 | ||
183 | #define AGGLEN_LMT_L 0x01A8 | ||
184 | #define DARFRC 0x01B0 | ||
185 | #define RARFRC 0x01B8 | ||
186 | #define MCS_TXAGC 0x01C0 | ||
187 | #define CCK_TXAGC 0x01C8 | ||
188 | |||
189 | /* 7. EDCA Setting Registers */ | ||
190 | #define EDCAPARA_VO 0x01D0 | ||
191 | #define EDCAPARA_VI 0x01D4 | ||
192 | #define EDCAPARA_BE 0x01D8 | ||
193 | #define EDCAPARA_BK 0x01DC | ||
194 | #define BCNTCFG 0x01E0 | ||
195 | #define CWRR 0x01E2 | ||
196 | #define ACMAVG 0x01E4 | ||
197 | #define AcmHwCtrl 0x01E7 | ||
198 | #define VO_ADMTM 0x01E8 | ||
199 | #define VI_ADMTM 0x01EC | ||
200 | #define BE_ADMTM 0x01F0 | ||
201 | #define RETRY_LIMIT 0x01F4 | ||
202 | #define SG_RATE 0x01F6 | ||
203 | |||
204 | /* 8. WMAC, BA and CCX related Register. */ | ||
205 | #define NAV_CTRL 0x0200 | ||
206 | #define BW_OPMODE 0x0203 | ||
207 | #define BACAMCMD 0x0204 | ||
208 | #define BACAMCONTENT 0x0208 | ||
209 | |||
210 | /* the 0x2xx register WMAC definition */ | ||
211 | #define LBDLY 0x0210 | ||
212 | #define FWDLY 0x0211 | ||
213 | #define HWPC_RX_CTRL 0x0218 | ||
214 | #define MQIR 0x0220 | ||
215 | #define MAIR 0x0222 | ||
216 | #define MSIR 0x0224 | ||
217 | #define CLM_RESULT 0x0227 | ||
218 | #define NHM_RPI_CNT 0x0228 | ||
219 | #define RXERR_RPT 0x0230 | ||
220 | #define NAV_PROT_LEN 0x0234 | ||
221 | #define CFEND_TH 0x0236 | ||
222 | #define AMPDU_MIN_SPACE 0x0237 | ||
223 | #define TXOP_STALL_CTRL 0x0238 | ||
224 | |||
225 | /* 9. Security Control Registers */ | ||
226 | #define REG_RWCAM 0x0240 | ||
227 | #define REG_WCAMI 0x0244 | ||
228 | #define REG_RCAMO 0x0248 | ||
229 | #define REG_CAMDBG 0x024C | ||
230 | #define REG_SECR 0x0250 | ||
231 | |||
232 | /* 10. Power Save Control Registers */ | ||
233 | #define WOW_CTRL 0x0260 | ||
234 | #define PSSTATUS 0x0261 | ||
235 | #define PSSWITCH 0x0262 | ||
236 | #define MIMOPS_WAIT_PERIOD 0x0263 | ||
237 | #define LPNAV_CTRL 0x0264 | ||
238 | #define WFM0 0x0270 | ||
239 | #define WFM1 0x0280 | ||
240 | #define WFM2 0x0290 | ||
241 | #define WFM3 0x02A0 | ||
242 | #define WFM4 0x02B0 | ||
243 | #define WFM5 0x02C0 | ||
244 | #define WFCRC 0x02D0 | ||
245 | #define FW_RPT_REG 0x02c4 | ||
246 | |||
247 | /* 11. General Purpose Registers */ | ||
248 | #define PSTIME 0x02E0 | ||
249 | #define TIMER0 0x02E4 | ||
250 | #define TIMER1 0x02E8 | ||
251 | #define GPIO_CTRL 0x02EC | ||
252 | #define GPIO_IN 0x02EC | ||
253 | #define GPIO_OUT 0x02ED | ||
254 | #define GPIO_IO_SEL 0x02EE | ||
255 | #define GPIO_MOD 0x02EF | ||
256 | #define GPIO_INTCTRL 0x02F0 | ||
257 | #define MAC_PINMUX_CFG 0x02F1 | ||
258 | #define LEDCFG 0x02F2 | ||
259 | #define PHY_REG 0x02F3 | ||
260 | #define PHY_REG_DATA 0x02F4 | ||
261 | #define REG_EFUSE_CLK 0x02F8 | ||
262 | |||
263 | /* 12. Host Interrupt Status Registers */ | ||
264 | #define INTA_MASK 0x0300 | ||
265 | #define ISR 0x0308 | ||
266 | |||
267 | /* 13. Test Mode and Debug Control Registers */ | ||
268 | #define DBG_PORT_SWITCH 0x003A | ||
269 | #define BIST 0x0310 | ||
270 | #define DBS 0x0314 | ||
271 | #define CPUINST 0x0318 | ||
272 | #define CPUCAUSE 0x031C | ||
273 | #define LBUS_ERR_ADDR 0x0320 | ||
274 | #define LBUS_ERR_CMD 0x0324 | ||
275 | #define LBUS_ERR_DATA_L 0x0328 | ||
276 | #define LBUS_ERR_DATA_H 0x032C | ||
277 | #define LX_EXCEPTION_ADDR 0x0330 | ||
278 | #define WDG_CTRL 0x0334 | ||
279 | #define INTMTU 0x0338 | ||
280 | #define INTM 0x033A | ||
281 | #define FDLOCKTURN0 0x033C | ||
282 | #define FDLOCKTURN1 0x033D | ||
283 | #define TRXPKTBUF_DBG_DATA 0x0340 | ||
284 | #define TRXPKTBUF_DBG_CTRL 0x0348 | ||
285 | #define DPLL 0x034A | ||
286 | #define CBUS_ERR_ADDR 0x0350 | ||
287 | #define CBUS_ERR_CMD 0x0354 | ||
288 | #define CBUS_ERR_DATA_L 0x0358 | ||
289 | #define CBUS_ERR_DATA_H 0x035C | ||
290 | #define USB_SIE_INTF_ADDR 0x0360 | ||
291 | #define USB_SIE_INTF_WD 0x0361 | ||
292 | #define USB_SIE_INTF_RD 0x0362 | ||
293 | #define USB_SIE_INTF_CTRL 0x0363 | ||
294 | #define LBUS_MON_ADDR 0x0364 | ||
295 | #define LBUS_ADDR_MASK 0x0368 | ||
296 | |||
297 | /* Boundary is 0x37F */ | ||
298 | |||
299 | /* 14. PCIE config register */ | ||
300 | #define TP_POLL 0x0500 | ||
301 | #define PM_CTRL 0x0502 | ||
302 | #define PCIF 0x0503 | ||
303 | |||
304 | #define THPDA 0x0514 | ||
305 | #define TMDA 0x0518 | ||
306 | #define TCDA 0x051C | ||
307 | #define HDA 0x0520 | ||
308 | #define TVODA 0x0524 | ||
309 | #define TVIDA 0x0528 | ||
310 | #define TBEDA 0x052C | ||
311 | #define TBKDA 0x0530 | ||
312 | #define TBDA 0x0534 | ||
313 | #define RCDA 0x0538 | ||
314 | #define RDQDA 0x053C | ||
315 | #define DBI_WDATA 0x0540 | ||
316 | #define DBI_RDATA 0x0544 | ||
317 | #define DBI_CTRL 0x0548 | ||
318 | #define MDIO_DATA 0x0550 | ||
319 | #define MDIO_CTRL 0x0554 | ||
320 | #define PCI_RPWM 0x0561 | ||
321 | #define PCI_CPWM 0x0563 | ||
322 | |||
323 | /* Config register (Offset 0x800-) */ | ||
324 | #define PHY_CCA 0x803 | ||
325 | |||
326 | /* Min Spacing related settings. */ | ||
327 | #define MAX_MSS_DENSITY_2T 0x13 | ||
328 | #define MAX_MSS_DENSITY_1T 0x0A | ||
329 | |||
330 | /* Rx DMA Control related settings */ | ||
331 | #define RXDMA_AGG_EN BIT(7) | ||
332 | |||
333 | #define RPWM PCI_RPWM | ||
334 | |||
335 | /* Regsiter Bit and Content definition */ | ||
336 | |||
337 | #define ISO_MD2PP BIT(0) | ||
338 | #define ISO_PA2PCIE BIT(3) | ||
339 | #define ISO_PLL2MD BIT(4) | ||
340 | #define ISO_PWC_DV2RP BIT(11) | ||
341 | #define ISO_PWC_RV2RP BIT(12) | ||
342 | |||
343 | |||
344 | #define FEN_MREGEN BIT(15) | ||
345 | #define FEN_DCORE BIT(11) | ||
346 | #define FEN_CPUEN BIT(10) | ||
347 | |||
348 | #define PAD_HWPD_IDN BIT(22) | ||
349 | |||
350 | #define SYS_CLKSEL_80M BIT(0) | ||
351 | #define SYS_PS_CLKSEL BIT(1) | ||
352 | #define SYS_CPU_CLKSEL BIT(2) | ||
353 | #define SYS_MAC_CLK_EN BIT(11) | ||
354 | #define SYS_SWHW_SEL BIT(14) | ||
355 | #define SYS_FWHW_SEL BIT(15) | ||
356 | |||
357 | #define CmdEEPROM_En BIT(5) | ||
358 | #define CmdEERPOMSEL BIT(4) | ||
359 | #define Cmd9346CR_9356SEL BIT(4) | ||
360 | |||
361 | #define AFE_MBEN BIT(1) | ||
362 | #define AFE_BGEN BIT(0) | ||
363 | |||
364 | #define SPS1_SWEN BIT(1) | ||
365 | #define SPS1_LDEN BIT(0) | ||
366 | |||
367 | #define RF_EN BIT(0) | ||
368 | #define RF_RSTB BIT(1) | ||
369 | #define RF_SDMRSTB BIT(2) | ||
370 | |||
371 | #define LDA15_EN BIT(0) | ||
372 | |||
373 | #define LDV12_EN BIT(0) | ||
374 | #define LDV12_SDBY BIT(1) | ||
375 | |||
376 | #define XTAL_GATE_AFE BIT(10) | ||
377 | |||
378 | #define APLL_EN BIT(0) | ||
379 | |||
380 | #define AFR_CardBEn BIT(0) | ||
381 | #define AFR_CLKRUN_SEL BIT(1) | ||
382 | #define AFR_FuncRegEn BIT(2) | ||
383 | |||
384 | #define APSDOFF_STATUS BIT(15) | ||
385 | #define APSDOFF BIT(14) | ||
386 | #define BBRSTN BIT(13) | ||
387 | #define BB_GLB_RSTN BIT(12) | ||
388 | #define SCHEDULE_EN BIT(10) | ||
389 | #define MACRXEN BIT(9) | ||
390 | #define MACTXEN BIT(8) | ||
391 | #define DDMA_EN BIT(7) | ||
392 | #define FW2HW_EN BIT(6) | ||
393 | #define RXDMA_EN BIT(5) | ||
394 | #define TXDMA_EN BIT(4) | ||
395 | #define HCI_RXDMA_EN BIT(3) | ||
396 | #define HCI_TXDMA_EN BIT(2) | ||
397 | |||
398 | #define StopHCCA BIT(6) | ||
399 | #define StopHigh BIT(5) | ||
400 | #define StopMgt BIT(4) | ||
401 | #define StopVO BIT(3) | ||
402 | #define StopVI BIT(2) | ||
403 | #define StopBE BIT(1) | ||
404 | #define StopBK BIT(0) | ||
405 | |||
406 | #define LBK_NORMAL 0x00 | ||
407 | #define LBK_MAC_LB (BIT(0) | BIT(1) | BIT(3)) | ||
408 | #define LBK_MAC_DLB (BIT(0) | BIT(1)) | ||
409 | #define LBK_DMA_LB (BIT(0) | BIT(1) | BIT(2)) | ||
410 | |||
411 | #define TCP_OFDL_EN BIT(25) | ||
412 | #define HWPC_TX_EN BIT(24) | ||
413 | #define TXDMAPRE2FULL BIT(23) | ||
414 | #define DISCW BIT(20) | ||
415 | #define TCRICV BIT(19) | ||
416 | #define CfendForm BIT(17) | ||
417 | #define TCRCRC BIT(16) | ||
418 | #define FAKE_IMEM_EN BIT(15) | ||
419 | #define TSFRST BIT(9) | ||
420 | #define TSFEN BIT(8) | ||
421 | #define FWALLRDY (BIT(0) | BIT(1) | BIT(2) | \ | ||
422 | BIT(3) | BIT(4) | BIT(5) | \ | ||
423 | BIT(6) | BIT(7)) | ||
424 | #define FWRDY BIT(7) | ||
425 | #define BASECHG BIT(6) | ||
426 | #define IMEM BIT(5) | ||
427 | #define DMEM_CODE_DONE BIT(4) | ||
428 | #define EXT_IMEM_CHK_RPT BIT(3) | ||
429 | #define EXT_IMEM_CODE_DONE BIT(2) | ||
430 | #define IMEM_CHK_RPT BIT(1) | ||
431 | #define IMEM_CODE_DONE BIT(0) | ||
432 | #define IMEM_CODE_DONE BIT(0) | ||
433 | #define IMEM_CHK_RPT BIT(1) | ||
434 | #define EMEM_CODE_DONE BIT(2) | ||
435 | #define EMEM_CHK_RPT BIT(3) | ||
436 | #define DMEM_CODE_DONE BIT(4) | ||
437 | #define IMEM_RDY BIT(5) | ||
438 | #define BASECHG BIT(6) | ||
439 | #define FWRDY BIT(7) | ||
440 | #define LOAD_FW_READY (IMEM_CODE_DONE | \ | ||
441 | IMEM_CHK_RPT | \ | ||
442 | EMEM_CODE_DONE | \ | ||
443 | EMEM_CHK_RPT | \ | ||
444 | DMEM_CODE_DONE | \ | ||
445 | IMEM_RDY | \ | ||
446 | BASECHG | \ | ||
447 | FWRDY) | ||
448 | #define TCR_TSFEN BIT(8) | ||
449 | #define TCR_TSFRST BIT(9) | ||
450 | #define TCR_FAKE_IMEM_EN BIT(15) | ||
451 | #define TCR_CRC BIT(16) | ||
452 | #define TCR_ICV BIT(19) | ||
453 | #define TCR_DISCW BIT(20) | ||
454 | #define TCR_HWPC_TX_EN BIT(24) | ||
455 | #define TCR_TCP_OFDL_EN BIT(25) | ||
456 | #define TXDMA_INIT_VALUE (IMEM_CHK_RPT | \ | ||
457 | EXT_IMEM_CHK_RPT) | ||
458 | |||
459 | #define RCR_APPFCS BIT(31) | ||
460 | #define RCR_DIS_ENC_2BYTE BIT(30) | ||
461 | #define RCR_DIS_AES_2BYTE BIT(29) | ||
462 | #define RCR_HTC_LOC_CTRL BIT(28) | ||
463 | #define RCR_ENMBID BIT(27) | ||
464 | #define RCR_RX_TCPOFDL_EN BIT(26) | ||
465 | #define RCR_APP_PHYST_RXFF BIT(25) | ||
466 | #define RCR_APP_PHYST_STAFF BIT(24) | ||
467 | #define RCR_CBSSID BIT(23) | ||
468 | #define RCR_APWRMGT BIT(22) | ||
469 | #define RCR_ADD3 BIT(21) | ||
470 | #define RCR_AMF BIT(20) | ||
471 | #define RCR_ACF BIT(19) | ||
472 | #define RCR_ADF BIT(18) | ||
473 | #define RCR_APP_MIC BIT(17) | ||
474 | #define RCR_APP_ICV BIT(16) | ||
475 | #define RCR_RXFTH BIT(13) | ||
476 | #define RCR_AICV BIT(12) | ||
477 | #define RCR_RXDESC_LK_EN BIT(11) | ||
478 | #define RCR_APP_BA_SSN BIT(6) | ||
479 | #define RCR_ACRC32 BIT(5) | ||
480 | #define RCR_RXSHFT_EN BIT(4) | ||
481 | #define RCR_AB BIT(3) | ||
482 | #define RCR_AM BIT(2) | ||
483 | #define RCR_APM BIT(1) | ||
484 | #define RCR_AAP BIT(0) | ||
485 | #define RCR_MXDMA_OFFSET 8 | ||
486 | #define RCR_FIFO_OFFSET 13 | ||
487 | |||
488 | |||
489 | #define MSR_LINK_MASK ((1 << 0) | (1 << 1)) | ||
490 | #define MSR_LINK_MANAGED 2 | ||
491 | #define MSR_LINK_NONE 0 | ||
492 | #define MSR_LINK_SHIFT 0 | ||
493 | #define MSR_LINK_ADHOC 1 | ||
494 | #define MSR_LINK_MASTER 3 | ||
495 | #define MSR_NOLINK 0x00 | ||
496 | #define MSR_ADHOC 0x01 | ||
497 | #define MSR_INFRA 0x02 | ||
498 | #define MSR_AP 0x03 | ||
499 | |||
500 | #define ENUART BIT(7) | ||
501 | #define ENJTAG BIT(3) | ||
502 | #define BTMODE (BIT(2) | BIT(1)) | ||
503 | #define ENBT BIT(0) | ||
504 | |||
505 | #define ENMBID BIT(7) | ||
506 | #define BCNUM (BIT(6) | BIT(5) | BIT(4)) | ||
507 | |||
508 | #define USTIME_EDCA 0xFF00 | ||
509 | #define USTIME_TSF 0x00FF | ||
510 | |||
511 | #define SIFS_TRX 0xFF00 | ||
512 | #define SIFS_CTX 0x00FF | ||
513 | |||
514 | #define ENSWBCN BIT(15) | ||
515 | #define DRVERLY_TU 0x0FF0 | ||
516 | #define DRVERLY_US 0x000F | ||
517 | #define BCN_TCFG_CW_SHIFT 8 | ||
518 | #define BCN_TCFG_IFS 0 | ||
519 | |||
520 | #define RRSR_RSC_OFFSET 21 | ||
521 | #define RRSR_SHORT_OFFSET 23 | ||
522 | #define RRSR_RSC_BW_40M 0x600000 | ||
523 | #define RRSR_RSC_UPSUBCHNL 0x400000 | ||
524 | #define RRSR_RSC_LOWSUBCHNL 0x200000 | ||
525 | #define RRSR_SHORT 0x800000 | ||
526 | #define RRSR_1M BIT(0) | ||
527 | #define RRSR_2M BIT(1) | ||
528 | #define RRSR_5_5M BIT(2) | ||
529 | #define RRSR_11M BIT(3) | ||
530 | #define RRSR_6M BIT(4) | ||
531 | #define RRSR_9M BIT(5) | ||
532 | #define RRSR_12M BIT(6) | ||
533 | #define RRSR_18M BIT(7) | ||
534 | #define RRSR_24M BIT(8) | ||
535 | #define RRSR_36M BIT(9) | ||
536 | #define RRSR_48M BIT(10) | ||
537 | #define RRSR_54M BIT(11) | ||
538 | #define RRSR_MCS0 BIT(12) | ||
539 | #define RRSR_MCS1 BIT(13) | ||
540 | #define RRSR_MCS2 BIT(14) | ||
541 | #define RRSR_MCS3 BIT(15) | ||
542 | #define RRSR_MCS4 BIT(16) | ||
543 | #define RRSR_MCS5 BIT(17) | ||
544 | #define RRSR_MCS6 BIT(18) | ||
545 | #define RRSR_MCS7 BIT(19) | ||
546 | #define BRSR_AckShortPmb BIT(23) | ||
547 | |||
548 | #define RATR_1M 0x00000001 | ||
549 | #define RATR_2M 0x00000002 | ||
550 | #define RATR_55M 0x00000004 | ||
551 | #define RATR_11M 0x00000008 | ||
552 | #define RATR_6M 0x00000010 | ||
553 | #define RATR_9M 0x00000020 | ||
554 | #define RATR_12M 0x00000040 | ||
555 | #define RATR_18M 0x00000080 | ||
556 | #define RATR_24M 0x00000100 | ||
557 | #define RATR_36M 0x00000200 | ||
558 | #define RATR_48M 0x00000400 | ||
559 | #define RATR_54M 0x00000800 | ||
560 | #define RATR_MCS0 0x00001000 | ||
561 | #define RATR_MCS1 0x00002000 | ||
562 | #define RATR_MCS2 0x00004000 | ||
563 | #define RATR_MCS3 0x00008000 | ||
564 | #define RATR_MCS4 0x00010000 | ||
565 | #define RATR_MCS5 0x00020000 | ||
566 | #define RATR_MCS6 0x00040000 | ||
567 | #define RATR_MCS7 0x00080000 | ||
568 | #define RATR_MCS8 0x00100000 | ||
569 | #define RATR_MCS9 0x00200000 | ||
570 | #define RATR_MCS10 0x00400000 | ||
571 | #define RATR_MCS11 0x00800000 | ||
572 | #define RATR_MCS12 0x01000000 | ||
573 | #define RATR_MCS13 0x02000000 | ||
574 | #define RATR_MCS14 0x04000000 | ||
575 | #define RATR_MCS15 0x08000000 | ||
576 | |||
577 | #define RATE_ALL_CCK (RATR_1M | RATR_2M | \ | ||
578 | RATR_55M | RATR_11M) | ||
579 | #define RATE_ALL_OFDM_AG (RATR_6M | RATR_9M | \ | ||
580 | RATR_12M | RATR_18M | \ | ||
581 | RATR_24M | RATR_36M | \ | ||
582 | RATR_48M | RATR_54M) | ||
583 | #define RATE_ALL_OFDM_1SS (RATR_MCS0 | RATR_MCS1 | \ | ||
584 | RATR_MCS2 | RATR_MCS3 | \ | ||
585 | RATR_MCS4 | RATR_MCS5 | \ | ||
586 | RATR_MCS6 | RATR_MCS7) | ||
587 | #define RATE_ALL_OFDM_2SS (RATR_MCS8 | RATR_MCS9 | \ | ||
588 | RATR_MCS10 | RATR_MCS11 | \ | ||
589 | RATR_MCS12 | RATR_MCS13 | \ | ||
590 | RATR_MCS14 | RATR_MCS15) | ||
591 | |||
592 | #define AC_PARAM_TXOP_LIMIT_OFFSET 16 | ||
593 | #define AC_PARAM_ECW_MAX_OFFSET 12 | ||
594 | #define AC_PARAM_ECW_MIN_OFFSET 8 | ||
595 | #define AC_PARAM_AIFS_OFFSET 0 | ||
596 | |||
597 | #define AcmHw_HwEn BIT(0) | ||
598 | #define AcmHw_BeqEn BIT(1) | ||
599 | #define AcmHw_ViqEn BIT(2) | ||
600 | #define AcmHw_VoqEn BIT(3) | ||
601 | #define AcmHw_BeqStatus BIT(4) | ||
602 | #define AcmHw_ViqStatus BIT(5) | ||
603 | #define AcmHw_VoqStatus BIT(6) | ||
604 | |||
605 | #define RETRY_LIMIT_SHORT_SHIFT 8 | ||
606 | #define RETRY_LIMIT_LONG_SHIFT 0 | ||
607 | |||
608 | #define NAV_UPPER_EN BIT(16) | ||
609 | #define NAV_UPPER 0xFF00 | ||
610 | #define NAV_RTSRST 0xFF | ||
611 | |||
612 | #define BW_OPMODE_20MHZ BIT(2) | ||
613 | #define BW_OPMODE_5G BIT(1) | ||
614 | #define BW_OPMODE_11J BIT(0) | ||
615 | |||
616 | #define RXERR_RPT_RST BIT(27) | ||
617 | #define RXERR_OFDM_PPDU 0 | ||
618 | #define RXERR_OFDM_FALSE_ALARM 1 | ||
619 | #define RXERR_OFDM_MPDU_OK 2 | ||
620 | #define RXERR_OFDM_MPDU_FAIL 3 | ||
621 | #define RXERR_CCK_PPDU 4 | ||
622 | #define RXERR_CCK_FALSE_ALARM 5 | ||
623 | #define RXERR_CCK_MPDU_OK 6 | ||
624 | #define RXERR_CCK_MPDU_FAIL 7 | ||
625 | #define RXERR_HT_PPDU 8 | ||
626 | #define RXERR_HT_FALSE_ALARM 9 | ||
627 | #define RXERR_HT_MPDU_TOTAL 10 | ||
628 | #define RXERR_HT_MPDU_OK 11 | ||
629 | #define RXERR_HT_MPDU_FAIL 12 | ||
630 | #define RXERR_RX_FULL_DROP 15 | ||
631 | |||
632 | #define SCR_TXUSEDK BIT(0) | ||
633 | #define SCR_RXUSEDK BIT(1) | ||
634 | #define SCR_TXENCENABLE BIT(2) | ||
635 | #define SCR_RXENCENABLE BIT(3) | ||
636 | #define SCR_SKBYA2 BIT(4) | ||
637 | #define SCR_NOSKMC BIT(5) | ||
638 | |||
639 | #define CAM_VALID BIT(15) | ||
640 | #define CAM_NOTVALID 0x0000 | ||
641 | #define CAM_USEDK BIT(5) | ||
642 | |||
643 | #define CAM_NONE 0x0 | ||
644 | #define CAM_WEP40 0x01 | ||
645 | #define CAM_TKIP 0x02 | ||
646 | #define CAM_AES 0x04 | ||
647 | #define CAM_WEP104 0x05 | ||
648 | |||
649 | #define TOTAL_CAM_ENTRY 32 | ||
650 | #define HALF_CAM_ENTRY 16 | ||
651 | |||
652 | #define CAM_WRITE BIT(16) | ||
653 | #define CAM_READ 0x00000000 | ||
654 | #define CAM_POLLINIG BIT(31) | ||
655 | |||
656 | #define WOW_PMEN BIT(0) | ||
657 | #define WOW_WOMEN BIT(1) | ||
658 | #define WOW_MAGIC BIT(2) | ||
659 | #define WOW_UWF BIT(3) | ||
660 | |||
661 | #define GPIOMUX_EN BIT(3) | ||
662 | #define GPIOSEL_GPIO 0 | ||
663 | #define GPIOSEL_PHYDBG 1 | ||
664 | #define GPIOSEL_BT 2 | ||
665 | #define GPIOSEL_WLANDBG 3 | ||
666 | #define GPIOSEL_GPIO_MASK (~(BIT(0)|BIT(1))) | ||
667 | |||
668 | #define HST_RDBUSY BIT(0) | ||
669 | #define CPU_WTBUSY BIT(1) | ||
670 | |||
671 | #define IMR8190_DISABLED 0x0 | ||
672 | #define IMR_CPUERR BIT(5) | ||
673 | #define IMR_ATIMEND BIT(4) | ||
674 | #define IMR_TBDOK BIT(3) | ||
675 | #define IMR_TBDER BIT(2) | ||
676 | #define IMR_BCNDMAINT8 BIT(1) | ||
677 | #define IMR_BCNDMAINT7 BIT(0) | ||
678 | #define IMR_BCNDMAINT6 BIT(31) | ||
679 | #define IMR_BCNDMAINT5 BIT(30) | ||
680 | #define IMR_BCNDMAINT4 BIT(29) | ||
681 | #define IMR_BCNDMAINT3 BIT(28) | ||
682 | #define IMR_BCNDMAINT2 BIT(27) | ||
683 | #define IMR_BCNDMAINT1 BIT(26) | ||
684 | #define IMR_BCNDOK8 BIT(25) | ||
685 | #define IMR_BCNDOK7 BIT(24) | ||
686 | #define IMR_BCNDOK6 BIT(23) | ||
687 | #define IMR_BCNDOK5 BIT(22) | ||
688 | #define IMR_BCNDOK4 BIT(21) | ||
689 | #define IMR_BCNDOK3 BIT(20) | ||
690 | #define IMR_BCNDOK2 BIT(19) | ||
691 | #define IMR_BCNDOK1 BIT(18) | ||
692 | #define IMR_TIMEOUT2 BIT(17) | ||
693 | #define IMR_TIMEOUT1 BIT(16) | ||
694 | #define IMR_TXFOVW BIT(15) | ||
695 | #define IMR_PSTIMEOUT BIT(14) | ||
696 | #define IMR_BCNINT BIT(13) | ||
697 | #define IMR_RXFOVW BIT(12) | ||
698 | #define IMR_RDU BIT(11) | ||
699 | #define IMR_RXCMDOK BIT(10) | ||
700 | #define IMR_BDOK BIT(9) | ||
701 | #define IMR_HIGHDOK BIT(8) | ||
702 | #define IMR_COMDOK BIT(7) | ||
703 | #define IMR_MGNTDOK BIT(6) | ||
704 | #define IMR_HCCADOK BIT(5) | ||
705 | #define IMR_BKDOK BIT(4) | ||
706 | #define IMR_BEDOK BIT(3) | ||
707 | #define IMR_VIDOK BIT(2) | ||
708 | #define IMR_VODOK BIT(1) | ||
709 | #define IMR_ROK BIT(0) | ||
710 | |||
711 | #define TPPOLL_BKQ BIT(0) | ||
712 | #define TPPOLL_BEQ BIT(1) | ||
713 | #define TPPOLL_VIQ BIT(2) | ||
714 | #define TPPOLL_VOQ BIT(3) | ||
715 | #define TPPOLL_BQ BIT(4) | ||
716 | #define TPPOLL_CQ BIT(5) | ||
717 | #define TPPOLL_MQ BIT(6) | ||
718 | #define TPPOLL_HQ BIT(7) | ||
719 | #define TPPOLL_HCCAQ BIT(8) | ||
720 | #define TPPOLL_STOPBK BIT(9) | ||
721 | #define TPPOLL_STOPBE BIT(10) | ||
722 | #define TPPOLL_STOPVI BIT(11) | ||
723 | #define TPPOLL_STOPVO BIT(12) | ||
724 | #define TPPOLL_STOPMGT BIT(13) | ||
725 | #define TPPOLL_STOPHIGH BIT(14) | ||
726 | #define TPPOLL_STOPHCCA BIT(15) | ||
727 | #define TPPOLL_SHIFT 8 | ||
728 | |||
729 | #define CCX_CMD_CLM_ENABLE BIT(0) | ||
730 | #define CCX_CMD_NHM_ENABLE BIT(1) | ||
731 | #define CCX_CMD_FUNCTION_ENABLE BIT(8) | ||
732 | #define CCX_CMD_IGNORE_CCA BIT(9) | ||
733 | #define CCX_CMD_IGNORE_TXON BIT(10) | ||
734 | #define CCX_CLM_RESULT_READY BIT(16) | ||
735 | #define CCX_NHM_RESULT_READY BIT(16) | ||
736 | #define CCX_CMD_RESET 0x0 | ||
737 | |||
738 | |||
739 | #define HWSET_MAX_SIZE_92S 128 | ||
740 | #define EFUSE_MAX_SECTION 16 | ||
741 | #define EFUSE_REAL_CONTENT_LEN 512 | ||
742 | |||
743 | #define RTL8190_EEPROM_ID 0x8129 | ||
744 | #define EEPROM_HPON 0x02 | ||
745 | #define EEPROM_CLK 0x06 | ||
746 | #define EEPROM_TESTR 0x08 | ||
747 | |||
748 | #define EEPROM_VID 0x0A | ||
749 | #define EEPROM_DID 0x0C | ||
750 | #define EEPROM_SVID 0x0E | ||
751 | #define EEPROM_SMID 0x10 | ||
752 | |||
753 | #define EEPROM_MAC_ADDR 0x12 | ||
754 | #define EEPROM_NODE_ADDRESS_BYTE_0 0x12 | ||
755 | |||
756 | #define EEPROM_PWDIFF 0x54 | ||
757 | |||
758 | #define EEPROM_TXPOWERBASE 0x50 | ||
759 | #define EEPROM_TX_PWR_INDEX_RANGE 28 | ||
760 | |||
761 | #define EEPROM_TX_PWR_HT20_DIFF 0x62 | ||
762 | #define DEFAULT_HT20_TXPWR_DIFF 2 | ||
763 | #define EEPROM_TX_PWR_OFDM_DIFF 0x65 | ||
764 | |||
765 | #define EEPROM_TXPWRGROUP 0x67 | ||
766 | #define EEPROM_REGULATORY 0x6D | ||
767 | |||
768 | #define TX_PWR_SAFETY_CHK 0x6D | ||
769 | #define EEPROM_TXPWINDEX_CCK_24G 0x5D | ||
770 | #define EEPROM_TXPWINDEX_OFDM_24G 0x6B | ||
771 | #define EEPROM_HT2T_CH1_A 0x6c | ||
772 | #define EEPROM_HT2T_CH7_A 0x6d | ||
773 | #define EEPROM_HT2T_CH13_A 0x6e | ||
774 | #define EEPROM_HT2T_CH1_B 0x6f | ||
775 | #define EEPROM_HT2T_CH7_B 0x70 | ||
776 | #define EEPROM_HT2T_CH13_B 0x71 | ||
777 | |||
778 | #define EEPROM_TSSI_A 0x74 | ||
779 | #define EEPROM_TSSI_B 0x75 | ||
780 | |||
781 | #define EEPROM_RFIND_POWERDIFF 0x76 | ||
782 | #define EEPROM_DEFAULT_LEGACYHTTXPOWERDIFF 0x3 | ||
783 | |||
784 | #define EEPROM_THERMALMETER 0x77 | ||
785 | #define EEPROM_BLUETOOTH_COEXIST 0x78 | ||
786 | #define EEPROM_BLUETOOTH_TYPE 0x4f | ||
787 | |||
788 | #define EEPROM_OPTIONAL 0x78 | ||
789 | #define EEPROM_WOWLAN 0x78 | ||
790 | |||
791 | #define EEPROM_CRYSTALCAP 0x79 | ||
792 | #define EEPROM_CHANNELPLAN 0x7B | ||
793 | #define EEPROM_VERSION 0x7C | ||
794 | #define EEPROM_CUSTOMID 0x7A | ||
795 | #define EEPROM_BOARDTYPE 0x7E | ||
796 | |||
797 | #define EEPROM_CHANNEL_PLAN_FCC 0x0 | ||
798 | #define EEPROM_CHANNEL_PLAN_IC 0x1 | ||
799 | #define EEPROM_CHANNEL_PLAN_ETSI 0x2 | ||
800 | #define EEPROM_CHANNEL_PLAN_SPAIN 0x3 | ||
801 | #define EEPROM_CHANNEL_PLAN_FRANCE 0x4 | ||
802 | #define EEPROM_CHANNEL_PLAN_MKK 0x5 | ||
803 | #define EEPROM_CHANNEL_PLAN_MKK1 0x6 | ||
804 | #define EEPROM_CHANNEL_PLAN_ISRAEL 0x7 | ||
805 | #define EEPROM_CHANNEL_PLAN_TELEC 0x8 | ||
806 | #define EEPROM_CHANNEL_PLAN_GLOBAL_DOMAIN 0x9 | ||
807 | #define EEPROM_CHANNEL_PLAN_WORLD_WIDE_13 0xA | ||
808 | #define EEPROM_CHANNEL_PLAN_NCC 0xB | ||
809 | #define EEPROM_CHANNEL_PLAN_BY_HW_MASK 0x80 | ||
810 | |||
811 | #define FW_DIG_DISABLE 0xfd00cc00 | ||
812 | #define FW_DIG_ENABLE 0xfd000000 | ||
813 | #define FW_DIG_HALT 0xfd000001 | ||
814 | #define FW_DIG_RESUME 0xfd000002 | ||
815 | #define FW_HIGH_PWR_DISABLE 0xfd000008 | ||
816 | #define FW_HIGH_PWR_ENABLE 0xfd000009 | ||
817 | #define FW_ADD_A2_ENTRY 0xfd000016 | ||
818 | #define FW_TXPWR_TRACK_ENABLE 0xfd000017 | ||
819 | #define FW_TXPWR_TRACK_DISABLE 0xfd000018 | ||
820 | #define FW_TXPWR_TRACK_THERMAL 0xfd000019 | ||
821 | #define FW_TXANT_SWITCH_ENABLE 0xfd000023 | ||
822 | #define FW_TXANT_SWITCH_DISABLE 0xfd000024 | ||
823 | #define FW_RA_INIT 0xfd000026 | ||
824 | #define FW_CTRL_DM_BY_DRIVER 0Xfd00002a | ||
825 | #define FW_RA_IOT_BG_COMB 0xfd000030 | ||
826 | #define FW_RA_IOT_N_COMB 0xfd000031 | ||
827 | #define FW_RA_REFRESH 0xfd0000a0 | ||
828 | #define FW_RA_UPDATE_MASK 0xfd0000a2 | ||
829 | #define FW_RA_DISABLE 0xfd0000a4 | ||
830 | #define FW_RA_ACTIVE 0xfd0000a6 | ||
831 | #define FW_RA_DISABLE_RSSI_MASK 0xfd0000ac | ||
832 | #define FW_RA_ENABLE_RSSI_MASK 0xfd0000ad | ||
833 | #define FW_RA_RESET 0xfd0000af | ||
834 | #define FW_DM_DISABLE 0xfd00aa00 | ||
835 | #define FW_IQK_ENABLE 0xf0000020 | ||
836 | #define FW_IQK_SUCCESS 0x0000dddd | ||
837 | #define FW_IQK_FAIL 0x0000ffff | ||
838 | #define FW_OP_FAILURE 0xffffffff | ||
839 | #define FW_TX_FEEDBACK_NONE 0xfb000000 | ||
840 | #define FW_TX_FEEDBACK_DTM_ENABLE (FW_TX_FEEDBACK_NONE | 0x1) | ||
841 | #define FW_TX_FEEDBACK_CCX_ENABL (FW_TX_FEEDBACK_NONE | 0x2) | ||
842 | #define FW_BB_RESET_ENABLE 0xff00000d | ||
843 | #define FW_BB_RESET_DISABLE 0xff00000e | ||
844 | #define FW_CCA_CHK_ENABLE 0xff000011 | ||
845 | #define FW_CCK_RESET_CNT 0xff000013 | ||
846 | #define FW_LPS_ENTER 0xfe000010 | ||
847 | #define FW_LPS_LEAVE 0xfe000011 | ||
848 | #define FW_INDIRECT_READ 0xf2000000 | ||
849 | #define FW_INDIRECT_WRITE 0xf2000001 | ||
850 | #define FW_CHAN_SET 0xf3000001 | ||
851 | |||
852 | #define RFPC 0x5F | ||
853 | #define RCR_9356SEL BIT(6) | ||
854 | #define TCR_LRL_OFFSET 0 | ||
855 | #define TCR_SRL_OFFSET 8 | ||
856 | #define TCR_MXDMA_OFFSET 21 | ||
857 | #define TCR_SAT BIT(24) | ||
858 | #define RCR_MXDMA_OFFSET 8 | ||
859 | #define RCR_FIFO_OFFSET 13 | ||
860 | #define RCR_OnlyErlPkt BIT(31) | ||
861 | #define CWR 0xDC | ||
862 | #define RETRYCTR 0xDE | ||
863 | |||
864 | #define CPU_GEN_SYSTEM_RESET 0x00000001 | ||
865 | |||
866 | #define CCX_COMMAND_REG 0x890 | ||
867 | #define CLM_PERIOD_REG 0x894 | ||
868 | #define NHM_PERIOD_REG 0x896 | ||
869 | |||
870 | #define NHM_THRESHOLD0 0x898 | ||
871 | #define NHM_THRESHOLD1 0x899 | ||
872 | #define NHM_THRESHOLD2 0x89A | ||
873 | #define NHM_THRESHOLD3 0x89B | ||
874 | #define NHM_THRESHOLD4 0x89C | ||
875 | #define NHM_THRESHOLD5 0x89D | ||
876 | #define NHM_THRESHOLD6 0x89E | ||
877 | #define CLM_RESULT_REG 0x8D0 | ||
878 | #define NHM_RESULT_REG 0x8D4 | ||
879 | #define NHM_RPI_COUNTER0 0x8D8 | ||
880 | #define NHM_RPI_COUNTER1 0x8D9 | ||
881 | #define NHM_RPI_COUNTER2 0x8DA | ||
882 | #define NHM_RPI_COUNTER3 0x8DB | ||
883 | #define NHM_RPI_COUNTER4 0x8DC | ||
884 | #define NHM_RPI_COUNTER5 0x8DD | ||
885 | #define NHM_RPI_COUNTER6 0x8DE | ||
886 | #define NHM_RPI_COUNTER7 0x8DF | ||
887 | |||
888 | #define HAL_8192S_HW_GPIO_OFF_BIT BIT(3) | ||
889 | #define HAL_8192S_HW_GPIO_OFF_MASK 0xF7 | ||
890 | #define HAL_8192S_HW_GPIO_WPS_BIT BIT(4) | ||
891 | |||
892 | #define RPMAC_RESET 0x100 | ||
893 | #define RPMAC_TXSTART 0x104 | ||
894 | #define RPMAC_TXLEGACYSIG 0x108 | ||
895 | #define RPMAC_TXHTSIG1 0x10c | ||
896 | #define RPMAC_TXHTSIG2 0x110 | ||
897 | #define RPMAC_PHYDEBUG 0x114 | ||
898 | #define RPMAC_TXPACKETNNM 0x118 | ||
899 | #define RPMAC_TXIDLE 0x11c | ||
900 | #define RPMAC_TXMACHEADER0 0x120 | ||
901 | #define RPMAC_TXMACHEADER1 0x124 | ||
902 | #define RPMAC_TXMACHEADER2 0x128 | ||
903 | #define RPMAC_TXMACHEADER3 0x12c | ||
904 | #define RPMAC_TXMACHEADER4 0x130 | ||
905 | #define RPMAC_TXMACHEADER5 0x134 | ||
906 | #define RPMAC_TXDATATYPE 0x138 | ||
907 | #define RPMAC_TXRANDOMSEED 0x13c | ||
908 | #define RPMAC_CCKPLCPPREAMBLE 0x140 | ||
909 | #define RPMAC_CCKPLCPHEADER 0x144 | ||
910 | #define RPMAC_CCKCRC16 0x148 | ||
911 | #define RPMAC_OFDMRXCRC32OK 0x170 | ||
912 | #define RPMAC_OFDMRXCRC32ER 0x174 | ||
913 | #define RPMAC_OFDMRXPARITYER 0x178 | ||
914 | #define RPMAC_OFDMRXCRC8ER 0x17c | ||
915 | #define RPMAC_CCKCRXRC16ER 0x180 | ||
916 | #define RPMAC_CCKCRXRC32ER 0x184 | ||
917 | #define RPMAC_CCKCRXRC32OK 0x188 | ||
918 | #define RPMAC_TXSTATUS 0x18c | ||
919 | |||
920 | #define RF_BB_CMD_ADDR 0x02c0 | ||
921 | #define RF_BB_CMD_DATA 0x02c4 | ||
922 | |||
923 | #define RFPGA0_RFMOD 0x800 | ||
924 | |||
925 | #define RFPGA0_TXINFO 0x804 | ||
926 | #define RFPGA0_PSDFUNCTION 0x808 | ||
927 | |||
928 | #define RFPGA0_TXGAINSTAGE 0x80c | ||
929 | |||
930 | #define RFPGA0_RFTIMING1 0x810 | ||
931 | #define RFPGA0_RFTIMING2 0x814 | ||
932 | #define RFPGA0_XA_HSSIPARAMETER1 0x820 | ||
933 | #define RFPGA0_XA_HSSIPARAMETER2 0x824 | ||
934 | #define RFPGA0_XB_HSSIPARAMETER1 0x828 | ||
935 | #define RFPGA0_XB_HSSIPARAMETER2 0x82c | ||
936 | #define RFPGA0_XC_HSSIPARAMETER1 0x830 | ||
937 | #define RFPGA0_XC_HSSIPARAMETER2 0x834 | ||
938 | #define RFPGA0_XD_HSSIPARAMETER1 0x838 | ||
939 | #define RFPGA0_XD_HSSIPARAMETER2 0x83c | ||
940 | #define RFPGA0_XA_LSSIPARAMETER 0x840 | ||
941 | #define RFPGA0_XB_LSSIPARAMETER 0x844 | ||
942 | #define RFPGA0_XC_LSSIPARAMETER 0x848 | ||
943 | #define RFPGA0_XD_LSSIPARAMETER 0x84c | ||
944 | |||
945 | #define RFPGA0_RFWAKEUP_PARAMETER 0x850 | ||
946 | #define RFPGA0_RFSLEEPUP_PARAMETER 0x854 | ||
947 | |||
948 | #define RFPGA0_XAB_SWITCHCONTROL 0x858 | ||
949 | #define RFPGA0_XCD_SWITCHCONTROL 0x85c | ||
950 | |||
951 | #define RFPGA0_XA_RFINTERFACEOE 0x860 | ||
952 | #define RFPGA0_XB_RFINTERFACEOE 0x864 | ||
953 | #define RFPGA0_XC_RFINTERFACEOE 0x868 | ||
954 | #define RFPGA0_XD_RFINTERFACEOE 0x86c | ||
955 | |||
956 | #define RFPGA0_XAB_RFINTERFACESW 0x870 | ||
957 | #define RFPGA0_XCD_RFINTERFACESW 0x874 | ||
958 | |||
959 | #define RFPGA0_XAB_RFPARAMETER 0x878 | ||
960 | #define RFPGA0_XCD_RFPARAMETER 0x87c | ||
961 | |||
962 | #define RFPGA0_ANALOGPARAMETER1 0x880 | ||
963 | #define RFPGA0_ANALOGPARAMETER2 0x884 | ||
964 | #define RFPGA0_ANALOGPARAMETER3 0x888 | ||
965 | #define RFPGA0_ANALOGPARAMETER4 0x88c | ||
966 | |||
967 | #define RFPGA0_XA_LSSIREADBACK 0x8a0 | ||
968 | #define RFPGA0_XB_LSSIREADBACK 0x8a4 | ||
969 | #define RFPGA0_XC_LSSIREADBACK 0x8a8 | ||
970 | #define RFPGA0_XD_LSSIREADBACK 0x8ac | ||
971 | |||
972 | #define RFPGA0_PSDREPORT 0x8b4 | ||
973 | #define TRANSCEIVERA_HSPI_READBACK 0x8b8 | ||
974 | #define TRANSCEIVERB_HSPI_READBACK 0x8bc | ||
975 | #define RFPGA0_XAB_RFINTERFACERB 0x8e0 | ||
976 | #define RFPGA0_XCD_RFINTERFACERB 0x8e4 | ||
977 | #define RFPGA1_RFMOD 0x900 | ||
978 | |||
979 | #define RFPGA1_TXBLOCK 0x904 | ||
980 | #define RFPGA1_DEBUGSELECT 0x908 | ||
981 | #define RFPGA1_TXINFO 0x90c | ||
982 | |||
983 | #define RCCK0_SYSTEM 0xa00 | ||
984 | |||
985 | #define RCCK0_AFESETTING 0xa04 | ||
986 | #define RCCK0_CCA 0xa08 | ||
987 | |||
988 | #define RCCK0_RXAGC1 0xa0c | ||
989 | #define RCCK0_RXAGC2 0xa10 | ||
990 | |||
991 | #define RCCK0_RXHP 0xa14 | ||
992 | |||
993 | #define RCCK0_DSPPARAMETER1 0xa18 | ||
994 | #define RCCK0_DSPPARAMETER2 0xa1c | ||
995 | |||
996 | #define RCCK0_TXFILTER1 0xa20 | ||
997 | #define RCCK0_TXFILTER2 0xa24 | ||
998 | #define RCCK0_DEBUGPORT 0xa28 | ||
999 | #define RCCK0_FALSEALARMREPORT 0xa2c | ||
1000 | #define RCCK0_TRSSIREPORT 0xa50 | ||
1001 | #define RCCK0_RXREPORT 0xa54 | ||
1002 | #define RCCK0_FACOUNTERLOWER 0xa5c | ||
1003 | #define RCCK0_FACOUNTERUPPER 0xa58 | ||
1004 | |||
1005 | #define ROFDM0_LSTF 0xc00 | ||
1006 | |||
1007 | #define ROFDM0_TRXPATHENABLE 0xc04 | ||
1008 | #define ROFDM0_TRMUXPAR 0xc08 | ||
1009 | #define ROFDM0_TRSWISOLATION 0xc0c | ||
1010 | |||
1011 | #define ROFDM0_XARXAFE 0xc10 | ||
1012 | #define ROFDM0_XARXIQIMBALANCE 0xc14 | ||
1013 | #define ROFDM0_XBRXAFE 0xc18 | ||
1014 | #define ROFDM0_XBRXIQIMBALANCE 0xc1c | ||
1015 | #define ROFDM0_XCRXAFE 0xc20 | ||
1016 | #define ROFDM0_XCRXIQIMBALANCE 0xc24 | ||
1017 | #define ROFDM0_XDRXAFE 0xc28 | ||
1018 | #define ROFDM0_XDRXIQIMBALANCE 0xc2c | ||
1019 | |||
1020 | #define ROFDM0_RXDETECTOR1 0xc30 | ||
1021 | #define ROFDM0_RXDETECTOR2 0xc34 | ||
1022 | #define ROFDM0_RXDETECTOR3 0xc38 | ||
1023 | #define ROFDM0_RXDETECTOR4 0xc3c | ||
1024 | |||
1025 | #define ROFDM0_RXDSP 0xc40 | ||
1026 | #define ROFDM0_CFO_AND_DAGC 0xc44 | ||
1027 | #define ROFDM0_CCADROP_THRESHOLD 0xc48 | ||
1028 | #define ROFDM0_ECCA_THRESHOLD 0xc4c | ||
1029 | |||
1030 | #define ROFDM0_XAAGCCORE1 0xc50 | ||
1031 | #define ROFDM0_XAAGCCORE2 0xc54 | ||
1032 | #define ROFDM0_XBAGCCORE1 0xc58 | ||
1033 | #define ROFDM0_XBAGCCORE2 0xc5c | ||
1034 | #define ROFDM0_XCAGCCORE1 0xc60 | ||
1035 | #define ROFDM0_XCAGCCORE2 0xc64 | ||
1036 | #define ROFDM0_XDAGCCORE1 0xc68 | ||
1037 | #define ROFDM0_XDAGCCORE2 0xc6c | ||
1038 | |||
1039 | #define ROFDM0_AGCPARAMETER1 0xc70 | ||
1040 | #define ROFDM0_AGCPARAMETER2 0xc74 | ||
1041 | #define ROFDM0_AGCRSSITABLE 0xc78 | ||
1042 | #define ROFDM0_HTSTFAGC 0xc7c | ||
1043 | |||
1044 | #define ROFDM0_XATXIQIMBALANCE 0xc80 | ||
1045 | #define ROFDM0_XATXAFE 0xc84 | ||
1046 | #define ROFDM0_XBTXIQIMBALANCE 0xc88 | ||
1047 | #define ROFDM0_XBTXAFE 0xc8c | ||
1048 | #define ROFDM0_XCTXIQIMBALANCE 0xc90 | ||
1049 | #define ROFDM0_XCTXAFE 0xc94 | ||
1050 | #define ROFDM0_XDTXIQIMBALANCE 0xc98 | ||
1051 | #define ROFDM0_XDTXAFE 0xc9c | ||
1052 | |||
1053 | #define ROFDM0_RXHP_PARAMETER 0xce0 | ||
1054 | #define ROFDM0_TXPSEUDO_NOISE_WGT 0xce4 | ||
1055 | #define ROFDM0_FRAME_SYNC 0xcf0 | ||
1056 | #define ROFDM0_DFSREPORT 0xcf4 | ||
1057 | #define ROFDM0_TXCOEFF1 0xca4 | ||
1058 | #define ROFDM0_TXCOEFF2 0xca8 | ||
1059 | #define ROFDM0_TXCOEFF3 0xcac | ||
1060 | #define ROFDM0_TXCOEFF4 0xcb0 | ||
1061 | #define ROFDM0_TXCOEFF5 0xcb4 | ||
1062 | #define ROFDM0_TXCOEFF6 0xcb8 | ||
1063 | |||
1064 | |||
1065 | #define ROFDM1_LSTF 0xd00 | ||
1066 | #define ROFDM1_TRXPATHENABLE 0xd04 | ||
1067 | |||
1068 | #define ROFDM1_CFO 0xd08 | ||
1069 | #define ROFDM1_CSI1 0xd10 | ||
1070 | #define ROFDM1_SBD 0xd14 | ||
1071 | #define ROFDM1_CSI2 0xd18 | ||
1072 | #define ROFDM1_CFOTRACKING 0xd2c | ||
1073 | #define ROFDM1_TRXMESAURE1 0xd34 | ||
1074 | #define ROFDM1_INTF_DET 0xd3c | ||
1075 | #define ROFDM1_PSEUDO_NOISESTATEAB 0xd50 | ||
1076 | #define ROFDM1_PSEUDO_NOISESTATECD 0xd54 | ||
1077 | #define ROFDM1_RX_PSEUDO_NOISE_WGT 0xd58 | ||
1078 | |||
1079 | #define ROFDM_PHYCOUNTER1 0xda0 | ||
1080 | #define ROFDM_PHYCOUNTER2 0xda4 | ||
1081 | #define ROFDM_PHYCOUNTER3 0xda8 | ||
1082 | |||
1083 | #define ROFDM_SHORT_CFOAB 0xdac | ||
1084 | #define ROFDM_SHORT_CFOCD 0xdb0 | ||
1085 | #define ROFDM_LONG_CFOAB 0xdb4 | ||
1086 | #define ROFDM_LONG_CFOCD 0xdb8 | ||
1087 | #define ROFDM_TAIL_CFOAB 0xdbc | ||
1088 | #define ROFDM_TAIL_CFOCD 0xdc0 | ||
1089 | #define ROFDM_PW_MEASURE1 0xdc4 | ||
1090 | #define ROFDM_PW_MEASURE2 0xdc8 | ||
1091 | #define ROFDM_BW_REPORT 0xdcc | ||
1092 | #define ROFDM_AGC_REPORT 0xdd0 | ||
1093 | #define ROFDM_RXSNR 0xdd4 | ||
1094 | #define ROFDM_RXEVMCSI 0xdd8 | ||
1095 | #define ROFDM_SIG_REPORT 0xddc | ||
1096 | |||
1097 | |||
1098 | #define RTXAGC_RATE18_06 0xe00 | ||
1099 | #define RTXAGC_RATE54_24 0xe04 | ||
1100 | #define RTXAGC_CCK_MCS32 0xe08 | ||
1101 | #define RTXAGC_MCS03_MCS00 0xe10 | ||
1102 | #define RTXAGC_MCS07_MCS04 0xe14 | ||
1103 | #define RTXAGC_MCS11_MCS08 0xe18 | ||
1104 | #define RTXAGC_MCS15_MCS12 0xe1c | ||
1105 | |||
1106 | |||
1107 | #define RF_AC 0x00 | ||
1108 | #define RF_IQADJ_G1 0x01 | ||
1109 | #define RF_IQADJ_G2 0x02 | ||
1110 | #define RF_POW_TRSW 0x05 | ||
1111 | #define RF_GAIN_RX 0x06 | ||
1112 | #define RF_GAIN_TX 0x07 | ||
1113 | #define RF_TXM_IDAC 0x08 | ||
1114 | #define RF_BS_IQGEN 0x0F | ||
1115 | |||
1116 | #define RF_MODE1 0x10 | ||
1117 | #define RF_MODE2 0x11 | ||
1118 | #define RF_RX_AGC_HP 0x12 | ||
1119 | #define RF_TX_AGC 0x13 | ||
1120 | #define RF_BIAS 0x14 | ||
1121 | #define RF_IPA 0x15 | ||
1122 | #define RF_POW_ABILITY 0x17 | ||
1123 | #define RF_MODE_AG 0x18 | ||
1124 | #define RF_CHANNEL 0x18 | ||
1125 | #define RF_CHNLBW 0x18 | ||
1126 | #define RF_TOP 0x19 | ||
1127 | #define RF_RX_G1 0x1A | ||
1128 | #define RF_RX_G2 0x1B | ||
1129 | #define RF_RX_BB2 0x1C | ||
1130 | #define RF_RX_BB1 0x1D | ||
1131 | #define RF_RCK1 0x1E | ||
1132 | #define RF_RCK2 0x1F | ||
1133 | |||
1134 | #define RF_TX_G1 0x20 | ||
1135 | #define RF_TX_G2 0x21 | ||
1136 | #define RF_TX_G3 0x22 | ||
1137 | #define RF_TX_BB1 0x23 | ||
1138 | #define RF_T_METER 0x24 | ||
1139 | #define RF_SYN_G1 0x25 | ||
1140 | #define RF_SYN_G2 0x26 | ||
1141 | #define RF_SYN_G3 0x27 | ||
1142 | #define RF_SYN_G4 0x28 | ||
1143 | #define RF_SYN_G5 0x29 | ||
1144 | #define RF_SYN_G6 0x2A | ||
1145 | #define RF_SYN_G7 0x2B | ||
1146 | #define RF_SYN_G8 0x2C | ||
1147 | |||
1148 | #define RF_RCK_OS 0x30 | ||
1149 | #define RF_TXPA_G1 0x31 | ||
1150 | #define RF_TXPA_G2 0x32 | ||
1151 | #define RF_TXPA_G3 0x33 | ||
1152 | |||
1153 | #define BRFMOD 0x1 | ||
1154 | #define BCCKEN 0x1000000 | ||
1155 | #define BOFDMEN 0x2000000 | ||
1156 | |||
1157 | #define BXBTXAGC 0xf00 | ||
1158 | #define BXCTXAGC 0xf000 | ||
1159 | #define BXDTXAGC 0xf0000 | ||
1160 | |||
1161 | #define B3WIRE_DATALENGTH 0x800 | ||
1162 | #define B3WIRE_ADDRESSLENGTH 0x400 | ||
1163 | |||
1164 | #define BRFSI_RFENV 0x10 | ||
1165 | |||
1166 | #define BLSSI_READADDRESS 0x7f800000 | ||
1167 | #define BLSSI_READEDGE 0x80000000 | ||
1168 | #define BLSSI_READBACK_DATA 0xfffff | ||
1169 | |||
1170 | #define BADCLKPHASE 0x4000000 | ||
1171 | |||
1172 | #define BCCK_SIDEBAND 0x10 | ||
1173 | |||
1174 | #define BTX_AGCRATECCK 0x7f00 | ||
1175 | |||
1176 | #define MASKBYTE0 0xff | ||
1177 | #define MASKBYTE1 0xff00 | ||
1178 | #define MASKBYTE2 0xff0000 | ||
1179 | #define MASKBYTE3 0xff000000 | ||
1180 | #define MASKHWORD 0xffff0000 | ||
1181 | #define MASKLWORD 0x0000ffff | ||
1182 | #define MASKDWORD 0xffffffff | ||
1183 | |||
1184 | #define MAKS12BITS 0xfffff | ||
1185 | #define MASK20BITS 0xfffff | ||
1186 | #define RFREG_OFFSET_MASK 0xfffff | ||
1187 | |||
1188 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/rf.c b/drivers/net/wireless/rtlwifi/rtl8192se/rf.c new file mode 100644 index 000000000000..1d3a48330399 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/rf.c | |||
@@ -0,0 +1,546 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "reg.h" | ||
32 | #include "def.h" | ||
33 | #include "phy.h" | ||
34 | #include "rf.h" | ||
35 | #include "dm.h" | ||
36 | |||
37 | |||
38 | static void _rtl92s_get_powerbase(struct ieee80211_hw *hw, u8 *p_pwrlevel, | ||
39 | u8 chnl, u32 *ofdmbase, u32 *mcsbase, | ||
40 | u8 *p_final_pwridx) | ||
41 | { | ||
42 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
43 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
44 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
45 | u32 pwrbase0, pwrbase1; | ||
46 | u8 legacy_pwrdiff = 0, ht20_pwrdiff = 0; | ||
47 | u8 i, pwrlevel[4]; | ||
48 | |||
49 | for (i = 0; i < 2; i++) | ||
50 | pwrlevel[i] = p_pwrlevel[i]; | ||
51 | |||
52 | /* We only care about the path A for legacy. */ | ||
53 | if (rtlefuse->eeprom_version < 2) { | ||
54 | pwrbase0 = pwrlevel[0] + (rtlefuse->legacy_httxpowerdiff & 0xf); | ||
55 | } else if (rtlefuse->eeprom_version >= 2) { | ||
56 | legacy_pwrdiff = rtlefuse->txpwr_legacyhtdiff | ||
57 | [RF90_PATH_A][chnl - 1]; | ||
58 | |||
59 | /* For legacy OFDM, tx pwr always > HT OFDM pwr. | ||
60 | * We do not care Path B | ||
61 | * legacy OFDM pwr diff. NO BB register | ||
62 | * to notify HW. */ | ||
63 | pwrbase0 = pwrlevel[0] + legacy_pwrdiff; | ||
64 | } | ||
65 | |||
66 | pwrbase0 = (pwrbase0 << 24) | (pwrbase0 << 16) | (pwrbase0 << 8) | | ||
67 | pwrbase0; | ||
68 | *ofdmbase = pwrbase0; | ||
69 | |||
70 | /* MCS rates */ | ||
71 | if (rtlefuse->eeprom_version >= 2) { | ||
72 | /* Check HT20 to HT40 diff */ | ||
73 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20) { | ||
74 | for (i = 0; i < 2; i++) { | ||
75 | /* rf-A, rf-B */ | ||
76 | /* HT 20<->40 pwr diff */ | ||
77 | ht20_pwrdiff = rtlefuse->txpwr_ht20diff | ||
78 | [i][chnl - 1]; | ||
79 | |||
80 | if (ht20_pwrdiff < 8) /* 0~+7 */ | ||
81 | pwrlevel[i] += ht20_pwrdiff; | ||
82 | else /* index8-15=-8~-1 */ | ||
83 | pwrlevel[i] -= (16 - ht20_pwrdiff); | ||
84 | } | ||
85 | } | ||
86 | } | ||
87 | |||
88 | /* use index of rf-A */ | ||
89 | pwrbase1 = pwrlevel[0]; | ||
90 | pwrbase1 = (pwrbase1 << 24) | (pwrbase1 << 16) | (pwrbase1 << 8) | | ||
91 | pwrbase1; | ||
92 | *mcsbase = pwrbase1; | ||
93 | |||
94 | /* The following is for Antenna | ||
95 | * diff from Ant-B to Ant-A */ | ||
96 | p_final_pwridx[0] = pwrlevel[0]; | ||
97 | p_final_pwridx[1] = pwrlevel[1]; | ||
98 | |||
99 | switch (rtlefuse->eeprom_regulatory) { | ||
100 | case 3: | ||
101 | /* The following is for calculation | ||
102 | * of the power diff for Ant-B to Ant-A. */ | ||
103 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
104 | p_final_pwridx[0] += rtlefuse->pwrgroup_ht40 | ||
105 | [RF90_PATH_A][ | ||
106 | chnl - 1]; | ||
107 | p_final_pwridx[1] += rtlefuse->pwrgroup_ht40 | ||
108 | [RF90_PATH_B][ | ||
109 | chnl - 1]; | ||
110 | } else { | ||
111 | p_final_pwridx[0] += rtlefuse->pwrgroup_ht20 | ||
112 | [RF90_PATH_A][ | ||
113 | chnl - 1]; | ||
114 | p_final_pwridx[1] += rtlefuse->pwrgroup_ht20 | ||
115 | [RF90_PATH_B][ | ||
116 | chnl - 1]; | ||
117 | } | ||
118 | break; | ||
119 | default: | ||
120 | break; | ||
121 | } | ||
122 | |||
123 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
124 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("40MHz finalpwr_idx " | ||
125 | "(A / B) = 0x%x / 0x%x\n", p_final_pwridx[0], | ||
126 | p_final_pwridx[1])); | ||
127 | } else { | ||
128 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, ("20MHz finalpwr_idx " | ||
129 | "(A / B) = 0x%x / 0x%x\n", p_final_pwridx[0], | ||
130 | p_final_pwridx[1])); | ||
131 | } | ||
132 | } | ||
133 | |||
134 | static void _rtl92s_set_antennadiff(struct ieee80211_hw *hw, | ||
135 | u8 *p_final_pwridx) | ||
136 | { | ||
137 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
138 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
139 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
140 | char ant_pwr_diff = 0; | ||
141 | u32 u4reg_val = 0; | ||
142 | |||
143 | if (rtlphy->rf_type == RF_2T2R) { | ||
144 | ant_pwr_diff = p_final_pwridx[1] - p_final_pwridx[0]; | ||
145 | |||
146 | /* range is from 7~-8, | ||
147 | * index = 0x0~0xf */ | ||
148 | if (ant_pwr_diff > 7) | ||
149 | ant_pwr_diff = 7; | ||
150 | if (ant_pwr_diff < -8) | ||
151 | ant_pwr_diff = -8; | ||
152 | |||
153 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
154 | ("Antenna Diff from RF-B " | ||
155 | "to RF-A = %d (0x%x)\n", ant_pwr_diff, | ||
156 | ant_pwr_diff & 0xf)); | ||
157 | |||
158 | ant_pwr_diff &= 0xf; | ||
159 | } | ||
160 | |||
161 | /* Antenna TX power difference */ | ||
162 | rtlefuse->antenna_txpwdiff[2] = 0;/* RF-D, don't care */ | ||
163 | rtlefuse->antenna_txpwdiff[1] = 0;/* RF-C, don't care */ | ||
164 | rtlefuse->antenna_txpwdiff[0] = (u8)(ant_pwr_diff); /* RF-B */ | ||
165 | |||
166 | u4reg_val = rtlefuse->antenna_txpwdiff[2] << 8 | | ||
167 | rtlefuse->antenna_txpwdiff[1] << 4 | | ||
168 | rtlefuse->antenna_txpwdiff[0]; | ||
169 | |||
170 | rtl_set_bbreg(hw, RFPGA0_TXGAINSTAGE, (BXBTXAGC | BXCTXAGC | BXDTXAGC), | ||
171 | u4reg_val); | ||
172 | |||
173 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
174 | ("Write BCD-Diff(0x%x) = 0x%x\n", | ||
175 | RFPGA0_TXGAINSTAGE, u4reg_val)); | ||
176 | } | ||
177 | |||
178 | static void _rtl92s_get_txpower_writeval_byregulatory(struct ieee80211_hw *hw, | ||
179 | u8 chnl, u8 index, | ||
180 | u32 pwrbase0, | ||
181 | u32 pwrbase1, | ||
182 | u32 *p_outwrite_val) | ||
183 | { | ||
184 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
185 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
186 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
187 | u8 i, chnlgroup, pwrdiff_limit[4]; | ||
188 | u32 writeval, customer_limit; | ||
189 | |||
190 | /* Index 0 & 1= legacy OFDM, 2-5=HT_MCS rate */ | ||
191 | switch (rtlefuse->eeprom_regulatory) { | ||
192 | case 0: | ||
193 | /* Realtek better performance increase power diff | ||
194 | * defined by Realtek for large power */ | ||
195 | chnlgroup = 0; | ||
196 | |||
197 | writeval = rtlphy->mcs_txpwrlevel_origoffset | ||
198 | [chnlgroup][index] + | ||
199 | ((index < 2) ? pwrbase0 : pwrbase1); | ||
200 | |||
201 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
202 | ("RTK better performance, " | ||
203 | "writeval = 0x%x\n", writeval)); | ||
204 | break; | ||
205 | case 1: | ||
206 | /* Realtek regulatory increase power diff defined | ||
207 | * by Realtek for regulatory */ | ||
208 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
209 | writeval = ((index < 2) ? pwrbase0 : pwrbase1); | ||
210 | |||
211 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
212 | ("Realtek regulatory, " | ||
213 | "40MHz, writeval = 0x%x\n", writeval)); | ||
214 | } else { | ||
215 | if (rtlphy->pwrgroup_cnt == 1) | ||
216 | chnlgroup = 0; | ||
217 | |||
218 | if (rtlphy->pwrgroup_cnt >= 3) { | ||
219 | if (chnl <= 3) | ||
220 | chnlgroup = 0; | ||
221 | else if (chnl >= 4 && chnl <= 8) | ||
222 | chnlgroup = 1; | ||
223 | else if (chnl > 8) | ||
224 | chnlgroup = 2; | ||
225 | if (rtlphy->pwrgroup_cnt == 4) | ||
226 | chnlgroup++; | ||
227 | } | ||
228 | |||
229 | writeval = rtlphy->mcs_txpwrlevel_origoffset | ||
230 | [chnlgroup][index] | ||
231 | + ((index < 2) ? | ||
232 | pwrbase0 : pwrbase1); | ||
233 | |||
234 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
235 | ("Realtek regulatory, " | ||
236 | "20MHz, writeval = 0x%x\n", writeval)); | ||
237 | } | ||
238 | break; | ||
239 | case 2: | ||
240 | /* Better regulatory don't increase any power diff */ | ||
241 | writeval = ((index < 2) ? pwrbase0 : pwrbase1); | ||
242 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
243 | ("Better regulatory, " | ||
244 | "writeval = 0x%x\n", writeval)); | ||
245 | break; | ||
246 | case 3: | ||
247 | /* Customer defined power diff. increase power diff | ||
248 | defined by customer. */ | ||
249 | chnlgroup = 0; | ||
250 | |||
251 | if (rtlphy->current_chan_bw == HT_CHANNEL_WIDTH_20_40) { | ||
252 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
253 | ("customer's limit, 40MHz = 0x%x\n", | ||
254 | rtlefuse->pwrgroup_ht40 | ||
255 | [RF90_PATH_A][chnl - 1])); | ||
256 | } else { | ||
257 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
258 | ("customer's limit, 20MHz = 0x%x\n", | ||
259 | rtlefuse->pwrgroup_ht20 | ||
260 | [RF90_PATH_A][chnl - 1])); | ||
261 | } | ||
262 | |||
263 | for (i = 0; i < 4; i++) { | ||
264 | pwrdiff_limit[i] = | ||
265 | (u8)((rtlphy->mcs_txpwrlevel_origoffset | ||
266 | [chnlgroup][index] & (0x7f << (i * 8))) | ||
267 | >> (i * 8)); | ||
268 | |||
269 | if (rtlphy->current_chan_bw == | ||
270 | HT_CHANNEL_WIDTH_20_40) { | ||
271 | if (pwrdiff_limit[i] > | ||
272 | rtlefuse->pwrgroup_ht40 | ||
273 | [RF90_PATH_A][chnl - 1]) { | ||
274 | pwrdiff_limit[i] = | ||
275 | rtlefuse->pwrgroup_ht20 | ||
276 | [RF90_PATH_A][chnl - 1]; | ||
277 | } | ||
278 | } else { | ||
279 | if (pwrdiff_limit[i] > | ||
280 | rtlefuse->pwrgroup_ht20 | ||
281 | [RF90_PATH_A][chnl - 1]) { | ||
282 | pwrdiff_limit[i] = | ||
283 | rtlefuse->pwrgroup_ht20 | ||
284 | [RF90_PATH_A][chnl - 1]; | ||
285 | } | ||
286 | } | ||
287 | } | ||
288 | |||
289 | customer_limit = (pwrdiff_limit[3] << 24) | | ||
290 | (pwrdiff_limit[2] << 16) | | ||
291 | (pwrdiff_limit[1] << 8) | | ||
292 | (pwrdiff_limit[0]); | ||
293 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
294 | ("Customer's limit = 0x%x\n", | ||
295 | customer_limit)); | ||
296 | |||
297 | writeval = customer_limit + ((index < 2) ? | ||
298 | pwrbase0 : pwrbase1); | ||
299 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
300 | ("Customer, writeval = " | ||
301 | "0x%x\n", writeval)); | ||
302 | break; | ||
303 | default: | ||
304 | chnlgroup = 0; | ||
305 | writeval = rtlphy->mcs_txpwrlevel_origoffset[chnlgroup][index] + | ||
306 | ((index < 2) ? pwrbase0 : pwrbase1); | ||
307 | RT_TRACE(rtlpriv, COMP_POWER, DBG_LOUD, | ||
308 | ("RTK better performance, " | ||
309 | "writeval = 0x%x\n", writeval)); | ||
310 | break; | ||
311 | } | ||
312 | |||
313 | if (rtlpriv->dm.dynamic_txhighpower_lvl == TX_HIGH_PWR_LEVEL_LEVEL1) | ||
314 | writeval = 0x10101010; | ||
315 | else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
316 | TX_HIGH_PWR_LEVEL_LEVEL2) | ||
317 | writeval = 0x0; | ||
318 | |||
319 | *p_outwrite_val = writeval; | ||
320 | |||
321 | } | ||
322 | |||
323 | static void _rtl92s_write_ofdm_powerreg(struct ieee80211_hw *hw, | ||
324 | u8 index, u32 val) | ||
325 | { | ||
326 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
327 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
328 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
329 | u16 regoffset[6] = {0xe00, 0xe04, 0xe10, 0xe14, 0xe18, 0xe1c}; | ||
330 | u8 i, rfa_pwr[4]; | ||
331 | u8 rfa_lower_bound = 0, rfa_upper_bound = 0, rf_pwr_diff = 0; | ||
332 | u32 writeval = val; | ||
333 | |||
334 | /* If path A and Path B coexist, we must limit Path A tx power. | ||
335 | * Protect Path B pwr over or under flow. We need to calculate | ||
336 | * upper and lower bound of path A tx power. */ | ||
337 | if (rtlphy->rf_type == RF_2T2R) { | ||
338 | rf_pwr_diff = rtlefuse->antenna_txpwdiff[0]; | ||
339 | |||
340 | /* Diff=-8~-1 */ | ||
341 | if (rf_pwr_diff >= 8) { | ||
342 | /* Prevent underflow!! */ | ||
343 | rfa_lower_bound = 0x10 - rf_pwr_diff; | ||
344 | /* if (rf_pwr_diff >= 0) Diff = 0-7 */ | ||
345 | } else { | ||
346 | rfa_upper_bound = RF6052_MAX_TX_PWR - rf_pwr_diff; | ||
347 | } | ||
348 | } | ||
349 | |||
350 | for (i = 0; i < 4; i++) { | ||
351 | rfa_pwr[i] = (u8)((writeval & (0x7f << (i * 8))) >> (i * 8)); | ||
352 | if (rfa_pwr[i] > RF6052_MAX_TX_PWR) | ||
353 | rfa_pwr[i] = RF6052_MAX_TX_PWR; | ||
354 | |||
355 | /* If path A and Path B coexist, we must limit Path A tx power. | ||
356 | * Protect Path B pwr over or under flow. We need to calculate | ||
357 | * upper and lower bound of path A tx power. */ | ||
358 | if (rtlphy->rf_type == RF_2T2R) { | ||
359 | /* Diff=-8~-1 */ | ||
360 | if (rf_pwr_diff >= 8) { | ||
361 | /* Prevent underflow!! */ | ||
362 | if (rfa_pwr[i] < rfa_lower_bound) | ||
363 | rfa_pwr[i] = rfa_lower_bound; | ||
364 | /* Diff = 0-7 */ | ||
365 | } else if (rf_pwr_diff >= 1) { | ||
366 | /* Prevent overflow */ | ||
367 | if (rfa_pwr[i] > rfa_upper_bound) | ||
368 | rfa_pwr[i] = rfa_upper_bound; | ||
369 | } | ||
370 | } | ||
371 | |||
372 | } | ||
373 | |||
374 | writeval = (rfa_pwr[3] << 24) | (rfa_pwr[2] << 16) | (rfa_pwr[1] << 8) | | ||
375 | rfa_pwr[0]; | ||
376 | |||
377 | rtl_set_bbreg(hw, regoffset[index], 0x7f7f7f7f, writeval); | ||
378 | } | ||
379 | |||
380 | void rtl92s_phy_rf6052_set_ofdmtxpower(struct ieee80211_hw *hw, | ||
381 | u8 *p_pwrlevel, u8 chnl) | ||
382 | { | ||
383 | u32 writeval, pwrbase0, pwrbase1; | ||
384 | u8 index = 0; | ||
385 | u8 finalpwr_idx[4]; | ||
386 | |||
387 | _rtl92s_get_powerbase(hw, p_pwrlevel, chnl, &pwrbase0, &pwrbase1, | ||
388 | &finalpwr_idx[0]); | ||
389 | _rtl92s_set_antennadiff(hw, &finalpwr_idx[0]); | ||
390 | |||
391 | for (index = 0; index < 6; index++) { | ||
392 | _rtl92s_get_txpower_writeval_byregulatory(hw, chnl, index, | ||
393 | pwrbase0, pwrbase1, &writeval); | ||
394 | |||
395 | _rtl92s_write_ofdm_powerreg(hw, index, writeval); | ||
396 | } | ||
397 | } | ||
398 | |||
399 | void rtl92s_phy_rf6052_set_ccktxpower(struct ieee80211_hw *hw, u8 pwrlevel) | ||
400 | { | ||
401 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
402 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
403 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
404 | u32 txagc = 0; | ||
405 | bool dont_inc_cck_or_turboscanoff = false; | ||
406 | |||
407 | if (((rtlefuse->eeprom_version >= 2) && | ||
408 | (rtlefuse->txpwr_safetyflag == 1)) || | ||
409 | ((rtlefuse->eeprom_version >= 2) && | ||
410 | (rtlefuse->eeprom_regulatory != 0))) | ||
411 | dont_inc_cck_or_turboscanoff = true; | ||
412 | |||
413 | if (mac->act_scanning == true) { | ||
414 | txagc = 0x3f; | ||
415 | if (dont_inc_cck_or_turboscanoff) | ||
416 | txagc = pwrlevel; | ||
417 | } else { | ||
418 | txagc = pwrlevel; | ||
419 | |||
420 | if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
421 | TX_HIGH_PWR_LEVEL_LEVEL1) | ||
422 | txagc = 0x10; | ||
423 | else if (rtlpriv->dm.dynamic_txhighpower_lvl == | ||
424 | TX_HIGH_PWR_LEVEL_LEVEL2) | ||
425 | txagc = 0x0; | ||
426 | } | ||
427 | |||
428 | if (txagc > RF6052_MAX_TX_PWR) | ||
429 | txagc = RF6052_MAX_TX_PWR; | ||
430 | |||
431 | rtl_set_bbreg(hw, RTXAGC_CCK_MCS32, BTX_AGCRATECCK, txagc); | ||
432 | |||
433 | } | ||
434 | |||
435 | bool rtl92s_phy_rf6052_config(struct ieee80211_hw *hw) | ||
436 | { | ||
437 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
438 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
439 | u32 u4reg_val = 0; | ||
440 | u8 rfpath; | ||
441 | bool rtstatus = true; | ||
442 | struct bb_reg_def *pphyreg; | ||
443 | |||
444 | /* Initialize RF */ | ||
445 | for (rfpath = 0; rfpath < rtlphy->num_total_rfpath; rfpath++) { | ||
446 | |||
447 | pphyreg = &rtlphy->phyreg_def[rfpath]; | ||
448 | |||
449 | /* Store original RFENV control type */ | ||
450 | switch (rfpath) { | ||
451 | case RF90_PATH_A: | ||
452 | case RF90_PATH_C: | ||
453 | u4reg_val = rtl92s_phy_query_bb_reg(hw, | ||
454 | pphyreg->rfintfs, | ||
455 | BRFSI_RFENV); | ||
456 | break; | ||
457 | case RF90_PATH_B: | ||
458 | case RF90_PATH_D: | ||
459 | u4reg_val = rtl92s_phy_query_bb_reg(hw, | ||
460 | pphyreg->rfintfs, | ||
461 | BRFSI_RFENV << 16); | ||
462 | break; | ||
463 | } | ||
464 | |||
465 | /* Set RF_ENV enable */ | ||
466 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfintfe, | ||
467 | BRFSI_RFENV << 16, 0x1); | ||
468 | |||
469 | /* Set RF_ENV output high */ | ||
470 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfintfo, BRFSI_RFENV, 0x1); | ||
471 | |||
472 | /* Set bit number of Address and Data for RF register */ | ||
473 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfhssi_para2, | ||
474 | B3WIRE_ADDRESSLENGTH, 0x0); | ||
475 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfhssi_para2, | ||
476 | B3WIRE_DATALENGTH, 0x0); | ||
477 | |||
478 | /* Initialize RF fom connfiguration file */ | ||
479 | switch (rfpath) { | ||
480 | case RF90_PATH_A: | ||
481 | rtstatus = rtl92s_phy_config_rf(hw, | ||
482 | (enum radio_path)rfpath); | ||
483 | break; | ||
484 | case RF90_PATH_B: | ||
485 | rtstatus = rtl92s_phy_config_rf(hw, | ||
486 | (enum radio_path)rfpath); | ||
487 | break; | ||
488 | case RF90_PATH_C: | ||
489 | break; | ||
490 | case RF90_PATH_D: | ||
491 | break; | ||
492 | } | ||
493 | |||
494 | /* Restore RFENV control type */ | ||
495 | switch (rfpath) { | ||
496 | case RF90_PATH_A: | ||
497 | case RF90_PATH_C: | ||
498 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfintfs, BRFSI_RFENV, | ||
499 | u4reg_val); | ||
500 | break; | ||
501 | case RF90_PATH_B: | ||
502 | case RF90_PATH_D: | ||
503 | rtl92s_phy_set_bb_reg(hw, pphyreg->rfintfs, | ||
504 | BRFSI_RFENV << 16, | ||
505 | u4reg_val); | ||
506 | break; | ||
507 | } | ||
508 | |||
509 | if (rtstatus != true) { | ||
510 | printk(KERN_ERR "Radio[%d] Fail!!", rfpath); | ||
511 | goto fail; | ||
512 | } | ||
513 | |||
514 | } | ||
515 | |||
516 | return rtstatus; | ||
517 | |||
518 | fail: | ||
519 | return rtstatus; | ||
520 | } | ||
521 | |||
522 | void rtl92s_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, u8 bandwidth) | ||
523 | { | ||
524 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
525 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
526 | |||
527 | switch (bandwidth) { | ||
528 | case HT_CHANNEL_WIDTH_20: | ||
529 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
530 | 0xfffff3ff) | 0x0400); | ||
531 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
532 | rtlphy->rfreg_chnlval[0]); | ||
533 | break; | ||
534 | case HT_CHANNEL_WIDTH_20_40: | ||
535 | rtlphy->rfreg_chnlval[0] = ((rtlphy->rfreg_chnlval[0] & | ||
536 | 0xfffff3ff)); | ||
537 | rtl_set_rfreg(hw, RF90_PATH_A, RF_CHNLBW, RFREG_OFFSET_MASK, | ||
538 | rtlphy->rfreg_chnlval[0]); | ||
539 | break; | ||
540 | default: | ||
541 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
542 | ("unknown bandwidth: %#X\n", | ||
543 | bandwidth)); | ||
544 | break; | ||
545 | } | ||
546 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/rf.h b/drivers/net/wireless/rtlwifi/rtl8192se/rf.h new file mode 100644 index 000000000000..3843baa1a874 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/rf.h | |||
@@ -0,0 +1,43 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __INC_RTL92S_RF_H | ||
30 | #define __INC_RTL92S_RF_H | ||
31 | |||
32 | #define RF6052_MAX_TX_PWR 0x3F | ||
33 | |||
34 | void rtl92s_phy_rf6052_set_bandwidth(struct ieee80211_hw *hw, | ||
35 | u8 bandwidth); | ||
36 | bool rtl92s_phy_rf6052_config(struct ieee80211_hw *hw) ; | ||
37 | void rtl92s_phy_rf6052_set_ccktxpower(struct ieee80211_hw *hw, | ||
38 | u8 powerlevel); | ||
39 | void rtl92s_phy_rf6052_set_ofdmtxpower(struct ieee80211_hw *hw, | ||
40 | u8 *p_pwrlevel, u8 chnl); | ||
41 | |||
42 | #endif | ||
43 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/sw.c b/drivers/net/wireless/rtlwifi/rtl8192se/sw.c new file mode 100644 index 000000000000..1c6cb1d7d660 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/sw.c | |||
@@ -0,0 +1,423 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include <linux/vmalloc.h> | ||
31 | |||
32 | #include "../wifi.h" | ||
33 | #include "../core.h" | ||
34 | #include "../pci.h" | ||
35 | #include "reg.h" | ||
36 | #include "def.h" | ||
37 | #include "phy.h" | ||
38 | #include "dm.h" | ||
39 | #include "fw.h" | ||
40 | #include "hw.h" | ||
41 | #include "sw.h" | ||
42 | #include "trx.h" | ||
43 | #include "led.h" | ||
44 | |||
45 | static void rtl92s_init_aspm_vars(struct ieee80211_hw *hw) | ||
46 | { | ||
47 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
48 | |||
49 | /*close ASPM for AMD defaultly */ | ||
50 | rtlpci->const_amdpci_aspm = 0; | ||
51 | |||
52 | /* | ||
53 | * ASPM PS mode. | ||
54 | * 0 - Disable ASPM, | ||
55 | * 1 - Enable ASPM without Clock Req, | ||
56 | * 2 - Enable ASPM with Clock Req, | ||
57 | * 3 - Alwyas Enable ASPM with Clock Req, | ||
58 | * 4 - Always Enable ASPM without Clock Req. | ||
59 | * set defult to RTL8192CE:3 RTL8192E:2 | ||
60 | * */ | ||
61 | rtlpci->const_pci_aspm = 2; | ||
62 | |||
63 | /*Setting for PCI-E device */ | ||
64 | rtlpci->const_devicepci_aspm_setting = 0x03; | ||
65 | |||
66 | /*Setting for PCI-E bridge */ | ||
67 | rtlpci->const_hostpci_aspm_setting = 0x02; | ||
68 | |||
69 | /* | ||
70 | * In Hw/Sw Radio Off situation. | ||
71 | * 0 - Default, | ||
72 | * 1 - From ASPM setting without low Mac Pwr, | ||
73 | * 2 - From ASPM setting with low Mac Pwr, | ||
74 | * 3 - Bus D3 | ||
75 | * set default to RTL8192CE:0 RTL8192SE:2 | ||
76 | */ | ||
77 | rtlpci->const_hwsw_rfoff_d3 = 2; | ||
78 | |||
79 | /* | ||
80 | * This setting works for those device with | ||
81 | * backdoor ASPM setting such as EPHY setting. | ||
82 | * 0 - Not support ASPM, | ||
83 | * 1 - Support ASPM, | ||
84 | * 2 - According to chipset. | ||
85 | */ | ||
86 | rtlpci->const_support_pciaspm = 2; | ||
87 | } | ||
88 | |||
89 | static int rtl92s_init_sw_vars(struct ieee80211_hw *hw) | ||
90 | { | ||
91 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
92 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
93 | const struct firmware *firmware; | ||
94 | struct rt_firmware *pfirmware = NULL; | ||
95 | int err = 0; | ||
96 | u16 earlyrxthreshold = 7; | ||
97 | |||
98 | rtlpriv->dm.dm_initialgain_enable = 1; | ||
99 | rtlpriv->dm.dm_flag = 0; | ||
100 | rtlpriv->dm.disable_framebursting = 0; | ||
101 | rtlpriv->dm.thermalvalue = 0; | ||
102 | rtlpriv->dm.useramask = true; | ||
103 | |||
104 | /* compatible 5G band 91se just 2.4G band & smsp */ | ||
105 | rtlpriv->rtlhal.current_bandtype = BAND_ON_2_4G; | ||
106 | rtlpriv->rtlhal.bandset = BAND_ON_2_4G; | ||
107 | rtlpriv->rtlhal.macphymode = SINGLEMAC_SINGLEPHY; | ||
108 | |||
109 | rtlpci->transmit_config = 0; | ||
110 | |||
111 | rtlpci->receive_config = | ||
112 | RCR_APPFCS | | ||
113 | RCR_APWRMGT | | ||
114 | /*RCR_ADD3 |*/ | ||
115 | RCR_AMF | | ||
116 | RCR_ADF | | ||
117 | RCR_APP_MIC | | ||
118 | RCR_APP_ICV | | ||
119 | RCR_AICV | | ||
120 | /* Accept ICV error, CRC32 Error */ | ||
121 | RCR_ACRC32 | | ||
122 | RCR_AB | | ||
123 | /* Accept Broadcast, Multicast */ | ||
124 | RCR_AM | | ||
125 | /* Accept Physical match */ | ||
126 | RCR_APM | | ||
127 | /* Accept Destination Address packets */ | ||
128 | /*RCR_AAP |*/ | ||
129 | RCR_APP_PHYST_STAFF | | ||
130 | /* Accept PHY status */ | ||
131 | RCR_APP_PHYST_RXFF | | ||
132 | (earlyrxthreshold << RCR_FIFO_OFFSET); | ||
133 | |||
134 | rtlpci->irq_mask[0] = (u32) | ||
135 | (IMR_ROK | | ||
136 | IMR_VODOK | | ||
137 | IMR_VIDOK | | ||
138 | IMR_BEDOK | | ||
139 | IMR_BKDOK | | ||
140 | IMR_HCCADOK | | ||
141 | IMR_MGNTDOK | | ||
142 | IMR_COMDOK | | ||
143 | IMR_HIGHDOK | | ||
144 | IMR_BDOK | | ||
145 | IMR_RXCMDOK | | ||
146 | /*IMR_TIMEOUT0 |*/ | ||
147 | IMR_RDU | | ||
148 | IMR_RXFOVW | | ||
149 | IMR_BCNINT | ||
150 | /*| IMR_TXFOVW*/ | ||
151 | /*| IMR_TBDOK | | ||
152 | IMR_TBDER*/); | ||
153 | |||
154 | rtlpci->irq_mask[1] = (u32) 0; | ||
155 | |||
156 | rtlpci->shortretry_limit = 0x30; | ||
157 | rtlpci->longretry_limit = 0x30; | ||
158 | |||
159 | rtlpci->first_init = true; | ||
160 | |||
161 | /* for LPS & IPS */ | ||
162 | rtlpriv->psc.inactiveps = rtlpriv->cfg->mod_params->inactiveps; | ||
163 | rtlpriv->psc.swctrl_lps = rtlpriv->cfg->mod_params->swctrl_lps; | ||
164 | rtlpriv->psc.fwctrl_lps = rtlpriv->cfg->mod_params->fwctrl_lps; | ||
165 | rtlpriv->psc.reg_fwctrl_lps = 3; | ||
166 | rtlpriv->psc.reg_max_lps_awakeintvl = 5; | ||
167 | /* for ASPM, you can close aspm through | ||
168 | * set const_support_pciaspm = 0 */ | ||
169 | rtl92s_init_aspm_vars(hw); | ||
170 | |||
171 | if (rtlpriv->psc.reg_fwctrl_lps == 1) | ||
172 | rtlpriv->psc.fwctrl_psmode = FW_PS_MIN_MODE; | ||
173 | else if (rtlpriv->psc.reg_fwctrl_lps == 2) | ||
174 | rtlpriv->psc.fwctrl_psmode = FW_PS_MAX_MODE; | ||
175 | else if (rtlpriv->psc.reg_fwctrl_lps == 3) | ||
176 | rtlpriv->psc.fwctrl_psmode = FW_PS_DTIM_MODE; | ||
177 | |||
178 | /* for firmware buf */ | ||
179 | rtlpriv->rtlhal.pfirmware = vzalloc(sizeof(struct rt_firmware)); | ||
180 | if (!rtlpriv->rtlhal.pfirmware) { | ||
181 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
182 | ("Can't alloc buffer for fw.\n")); | ||
183 | return 1; | ||
184 | } | ||
185 | |||
186 | printk(KERN_INFO "rtl8192se: Driver for Realtek RTL8192SE/RTL8191SE\n" | ||
187 | " Loading firmware %s\n", rtlpriv->cfg->fw_name); | ||
188 | /* request fw */ | ||
189 | err = request_firmware(&firmware, rtlpriv->cfg->fw_name, | ||
190 | rtlpriv->io.dev); | ||
191 | if (err) { | ||
192 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
193 | ("Failed to request firmware!\n")); | ||
194 | return 1; | ||
195 | } | ||
196 | if (firmware->size > sizeof(struct rt_firmware)) { | ||
197 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
198 | ("Firmware is too big!\n")); | ||
199 | release_firmware(firmware); | ||
200 | return 1; | ||
201 | } | ||
202 | |||
203 | pfirmware = (struct rt_firmware *)rtlpriv->rtlhal.pfirmware; | ||
204 | memcpy(pfirmware->sz_fw_tmpbuffer, firmware->data, firmware->size); | ||
205 | pfirmware->sz_fw_tmpbufferlen = firmware->size; | ||
206 | release_firmware(firmware); | ||
207 | |||
208 | return err; | ||
209 | } | ||
210 | |||
211 | static void rtl92s_deinit_sw_vars(struct ieee80211_hw *hw) | ||
212 | { | ||
213 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
214 | |||
215 | if (rtlpriv->rtlhal.pfirmware) { | ||
216 | vfree(rtlpriv->rtlhal.pfirmware); | ||
217 | rtlpriv->rtlhal.pfirmware = NULL; | ||
218 | } | ||
219 | } | ||
220 | |||
221 | static struct rtl_hal_ops rtl8192se_hal_ops = { | ||
222 | .init_sw_vars = rtl92s_init_sw_vars, | ||
223 | .deinit_sw_vars = rtl92s_deinit_sw_vars, | ||
224 | .read_eeprom_info = rtl92se_read_eeprom_info, | ||
225 | .interrupt_recognized = rtl92se_interrupt_recognized, | ||
226 | .hw_init = rtl92se_hw_init, | ||
227 | .hw_disable = rtl92se_card_disable, | ||
228 | .hw_suspend = rtl92se_suspend, | ||
229 | .hw_resume = rtl92se_resume, | ||
230 | .enable_interrupt = rtl92se_enable_interrupt, | ||
231 | .disable_interrupt = rtl92se_disable_interrupt, | ||
232 | .set_network_type = rtl92se_set_network_type, | ||
233 | .set_chk_bssid = rtl92se_set_check_bssid, | ||
234 | .set_qos = rtl92se_set_qos, | ||
235 | .set_bcn_reg = rtl92se_set_beacon_related_registers, | ||
236 | .set_bcn_intv = rtl92se_set_beacon_interval, | ||
237 | .update_interrupt_mask = rtl92se_update_interrupt_mask, | ||
238 | .get_hw_reg = rtl92se_get_hw_reg, | ||
239 | .set_hw_reg = rtl92se_set_hw_reg, | ||
240 | .update_rate_tbl = rtl92se_update_hal_rate_tbl, | ||
241 | .fill_tx_desc = rtl92se_tx_fill_desc, | ||
242 | .fill_tx_cmddesc = rtl92se_tx_fill_cmddesc, | ||
243 | .query_rx_desc = rtl92se_rx_query_desc, | ||
244 | .set_channel_access = rtl92se_update_channel_access_setting, | ||
245 | .radio_onoff_checking = rtl92se_gpio_radio_on_off_checking, | ||
246 | .set_bw_mode = rtl92s_phy_set_bw_mode, | ||
247 | .switch_channel = rtl92s_phy_sw_chnl, | ||
248 | .dm_watchdog = rtl92s_dm_watchdog, | ||
249 | .scan_operation_backup = rtl92s_phy_scan_operation_backup, | ||
250 | .set_rf_power_state = rtl92s_phy_set_rf_power_state, | ||
251 | .led_control = rtl92se_led_control, | ||
252 | .set_desc = rtl92se_set_desc, | ||
253 | .get_desc = rtl92se_get_desc, | ||
254 | .tx_polling = rtl92se_tx_polling, | ||
255 | .enable_hw_sec = rtl92se_enable_hw_security_config, | ||
256 | .set_key = rtl92se_set_key, | ||
257 | .init_sw_leds = rtl92se_init_sw_leds, | ||
258 | .get_bbreg = rtl92s_phy_query_bb_reg, | ||
259 | .set_bbreg = rtl92s_phy_set_bb_reg, | ||
260 | .get_rfreg = rtl92s_phy_query_rf_reg, | ||
261 | .set_rfreg = rtl92s_phy_set_rf_reg, | ||
262 | }; | ||
263 | |||
264 | static struct rtl_mod_params rtl92se_mod_params = { | ||
265 | .sw_crypto = false, | ||
266 | .inactiveps = true, | ||
267 | .swctrl_lps = true, | ||
268 | .fwctrl_lps = false, | ||
269 | }; | ||
270 | |||
271 | /* Because memory R/W bursting will cause system hang/crash | ||
272 | * for 92se, so we don't read back after every write action */ | ||
273 | static struct rtl_hal_cfg rtl92se_hal_cfg = { | ||
274 | .bar_id = 1, | ||
275 | .write_readback = false, | ||
276 | .name = "rtl92s_pci", | ||
277 | .fw_name = "rtlwifi/rtl8192sefw.bin", | ||
278 | .ops = &rtl8192se_hal_ops, | ||
279 | .mod_params = &rtl92se_mod_params, | ||
280 | |||
281 | .maps[SYS_ISO_CTRL] = REG_SYS_ISO_CTRL, | ||
282 | .maps[SYS_FUNC_EN] = REG_SYS_FUNC_EN, | ||
283 | .maps[SYS_CLK] = SYS_CLKR, | ||
284 | .maps[MAC_RCR_AM] = RCR_AM, | ||
285 | .maps[MAC_RCR_AB] = RCR_AB, | ||
286 | .maps[MAC_RCR_ACRC32] = RCR_ACRC32, | ||
287 | .maps[MAC_RCR_ACF] = RCR_ACF, | ||
288 | .maps[MAC_RCR_AAP] = RCR_AAP, | ||
289 | |||
290 | .maps[EFUSE_TEST] = REG_EFUSE_TEST, | ||
291 | .maps[EFUSE_CTRL] = REG_EFUSE_CTRL, | ||
292 | .maps[EFUSE_CLK] = REG_EFUSE_CLK, | ||
293 | .maps[EFUSE_CLK_CTRL] = REG_EFUSE_CTRL, | ||
294 | .maps[EFUSE_PWC_EV12V] = 0, /* nouse for 8192se */ | ||
295 | .maps[EFUSE_FEN_ELDR] = 0, /* nouse for 8192se */ | ||
296 | .maps[EFUSE_LOADER_CLK_EN] = 0,/* nouse for 8192se */ | ||
297 | .maps[EFUSE_ANA8M] = EFUSE_ANA8M, | ||
298 | .maps[EFUSE_HWSET_MAX_SIZE] = HWSET_MAX_SIZE_92S, | ||
299 | .maps[EFUSE_MAX_SECTION_MAP] = EFUSE_MAX_SECTION, | ||
300 | .maps[EFUSE_REAL_CONTENT_SIZE] = EFUSE_REAL_CONTENT_LEN, | ||
301 | |||
302 | .maps[RWCAM] = REG_RWCAM, | ||
303 | .maps[WCAMI] = REG_WCAMI, | ||
304 | .maps[RCAMO] = REG_RCAMO, | ||
305 | .maps[CAMDBG] = REG_CAMDBG, | ||
306 | .maps[SECR] = REG_SECR, | ||
307 | .maps[SEC_CAM_NONE] = CAM_NONE, | ||
308 | .maps[SEC_CAM_WEP40] = CAM_WEP40, | ||
309 | .maps[SEC_CAM_TKIP] = CAM_TKIP, | ||
310 | .maps[SEC_CAM_AES] = CAM_AES, | ||
311 | .maps[SEC_CAM_WEP104] = CAM_WEP104, | ||
312 | |||
313 | .maps[RTL_IMR_BCNDMAINT6] = IMR_BCNDMAINT6, | ||
314 | .maps[RTL_IMR_BCNDMAINT5] = IMR_BCNDMAINT5, | ||
315 | .maps[RTL_IMR_BCNDMAINT4] = IMR_BCNDMAINT4, | ||
316 | .maps[RTL_IMR_BCNDMAINT3] = IMR_BCNDMAINT3, | ||
317 | .maps[RTL_IMR_BCNDMAINT2] = IMR_BCNDMAINT2, | ||
318 | .maps[RTL_IMR_BCNDMAINT1] = IMR_BCNDMAINT1, | ||
319 | .maps[RTL_IMR_BCNDOK8] = IMR_BCNDOK8, | ||
320 | .maps[RTL_IMR_BCNDOK7] = IMR_BCNDOK7, | ||
321 | .maps[RTL_IMR_BCNDOK6] = IMR_BCNDOK6, | ||
322 | .maps[RTL_IMR_BCNDOK5] = IMR_BCNDOK5, | ||
323 | .maps[RTL_IMR_BCNDOK4] = IMR_BCNDOK4, | ||
324 | .maps[RTL_IMR_BCNDOK3] = IMR_BCNDOK3, | ||
325 | .maps[RTL_IMR_BCNDOK2] = IMR_BCNDOK2, | ||
326 | .maps[RTL_IMR_BCNDOK1] = IMR_BCNDOK1, | ||
327 | .maps[RTL_IMR_TIMEOUT2] = IMR_TIMEOUT2, | ||
328 | .maps[RTL_IMR_TIMEOUT1] = IMR_TIMEOUT1, | ||
329 | |||
330 | .maps[RTL_IMR_TXFOVW] = IMR_TXFOVW, | ||
331 | .maps[RTL_IMR_PSTIMEOUT] = IMR_PSTIMEOUT, | ||
332 | .maps[RTL_IMR_BcnInt] = IMR_BCNINT, | ||
333 | .maps[RTL_IMR_RXFOVW] = IMR_RXFOVW, | ||
334 | .maps[RTL_IMR_RDU] = IMR_RDU, | ||
335 | .maps[RTL_IMR_ATIMEND] = IMR_ATIMEND, | ||
336 | .maps[RTL_IMR_BDOK] = IMR_BDOK, | ||
337 | .maps[RTL_IMR_MGNTDOK] = IMR_MGNTDOK, | ||
338 | .maps[RTL_IMR_TBDER] = IMR_TBDER, | ||
339 | .maps[RTL_IMR_HIGHDOK] = IMR_HIGHDOK, | ||
340 | .maps[RTL_IMR_COMDOK] = IMR_COMDOK, | ||
341 | .maps[RTL_IMR_TBDOK] = IMR_TBDOK, | ||
342 | .maps[RTL_IMR_BKDOK] = IMR_BKDOK, | ||
343 | .maps[RTL_IMR_BEDOK] = IMR_BEDOK, | ||
344 | .maps[RTL_IMR_VIDOK] = IMR_VIDOK, | ||
345 | .maps[RTL_IMR_VODOK] = IMR_VODOK, | ||
346 | .maps[RTL_IMR_ROK] = IMR_ROK, | ||
347 | .maps[RTL_IBSS_INT_MASKS] = (IMR_BCNINT | IMR_TBDOK | IMR_TBDER), | ||
348 | |||
349 | .maps[RTL_RC_CCK_RATE1M] = DESC92S_RATE1M, | ||
350 | .maps[RTL_RC_CCK_RATE2M] = DESC92S_RATE2M, | ||
351 | .maps[RTL_RC_CCK_RATE5_5M] = DESC92S_RATE5_5M, | ||
352 | .maps[RTL_RC_CCK_RATE11M] = DESC92S_RATE11M, | ||
353 | .maps[RTL_RC_OFDM_RATE6M] = DESC92S_RATE6M, | ||
354 | .maps[RTL_RC_OFDM_RATE9M] = DESC92S_RATE9M, | ||
355 | .maps[RTL_RC_OFDM_RATE12M] = DESC92S_RATE12M, | ||
356 | .maps[RTL_RC_OFDM_RATE18M] = DESC92S_RATE18M, | ||
357 | .maps[RTL_RC_OFDM_RATE24M] = DESC92S_RATE24M, | ||
358 | .maps[RTL_RC_OFDM_RATE36M] = DESC92S_RATE36M, | ||
359 | .maps[RTL_RC_OFDM_RATE48M] = DESC92S_RATE48M, | ||
360 | .maps[RTL_RC_OFDM_RATE54M] = DESC92S_RATE54M, | ||
361 | |||
362 | .maps[RTL_RC_HT_RATEMCS7] = DESC92S_RATEMCS7, | ||
363 | .maps[RTL_RC_HT_RATEMCS15] = DESC92S_RATEMCS15, | ||
364 | }; | ||
365 | |||
366 | static struct pci_device_id rtl92se_pci_ids[] __devinitdata = { | ||
367 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8192, rtl92se_hal_cfg)}, | ||
368 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8171, rtl92se_hal_cfg)}, | ||
369 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8172, rtl92se_hal_cfg)}, | ||
370 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8173, rtl92se_hal_cfg)}, | ||
371 | {RTL_PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8174, rtl92se_hal_cfg)}, | ||
372 | {}, | ||
373 | }; | ||
374 | |||
375 | MODULE_DEVICE_TABLE(pci, rtl92se_pci_ids); | ||
376 | |||
377 | MODULE_AUTHOR("lizhaoming <chaoming_li@realsil.com.cn>"); | ||
378 | MODULE_AUTHOR("Realtek WlanFAE <wlanfae@realtek.com>"); | ||
379 | MODULE_LICENSE("GPL"); | ||
380 | MODULE_DESCRIPTION("Realtek 8192S/8191S 802.11n PCI wireless"); | ||
381 | MODULE_FIRMWARE("rtlwifi/rtl8192sefw.bin"); | ||
382 | |||
383 | module_param_named(swenc, rtl92se_mod_params.sw_crypto, bool, 0444); | ||
384 | module_param_named(ips, rtl92se_mod_params.inactiveps, bool, 0444); | ||
385 | module_param_named(swlps, rtl92se_mod_params.swctrl_lps, bool, 0444); | ||
386 | module_param_named(fwlps, rtl92se_mod_params.fwctrl_lps, bool, 0444); | ||
387 | MODULE_PARM_DESC(swenc, "using hardware crypto (default 0 [hardware])\n"); | ||
388 | MODULE_PARM_DESC(ips, "using no link power save (default 1 is open)\n"); | ||
389 | MODULE_PARM_DESC(swlps, "using linked sw control power save (default 1 is " | ||
390 | "open)\n"); | ||
391 | |||
392 | |||
393 | static struct pci_driver rtl92se_driver = { | ||
394 | .name = KBUILD_MODNAME, | ||
395 | .id_table = rtl92se_pci_ids, | ||
396 | .probe = rtl_pci_probe, | ||
397 | .remove = rtl_pci_disconnect, | ||
398 | |||
399 | #ifdef CONFIG_PM | ||
400 | .suspend = rtl_pci_suspend, | ||
401 | .resume = rtl_pci_resume, | ||
402 | #endif | ||
403 | |||
404 | }; | ||
405 | |||
406 | static int __init rtl92se_module_init(void) | ||
407 | { | ||
408 | int ret = 0; | ||
409 | |||
410 | ret = pci_register_driver(&rtl92se_driver); | ||
411 | if (ret) | ||
412 | RT_ASSERT(false, (": No device found\n")); | ||
413 | |||
414 | return ret; | ||
415 | } | ||
416 | |||
417 | static void __exit rtl92se_module_exit(void) | ||
418 | { | ||
419 | pci_unregister_driver(&rtl92se_driver); | ||
420 | } | ||
421 | |||
422 | module_init(rtl92se_module_init); | ||
423 | module_exit(rtl92se_module_exit); | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/sw.h b/drivers/net/wireless/rtlwifi/rtl8192se/sw.h new file mode 100644 index 000000000000..fc4eb285a0ac --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/sw.h | |||
@@ -0,0 +1,36 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | *****************************************************************************/ | ||
27 | #ifndef __REALTEK_PCI92SE_SW_H__ | ||
28 | #define __REALTEK_PCI92SE_SW_H__ | ||
29 | |||
30 | #define EFUSE_MAX_SECTION 16 | ||
31 | |||
32 | int rtl92se_init_sw(struct ieee80211_hw *hw); | ||
33 | void rtl92se_deinit_sw(struct ieee80211_hw *hw); | ||
34 | void rtl92se_init_var_map(struct ieee80211_hw *hw); | ||
35 | |||
36 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/table.c b/drivers/net/wireless/rtlwifi/rtl8192se/table.c new file mode 100644 index 000000000000..154185b3969d --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/table.c | |||
@@ -0,0 +1,634 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | * Created on 2010/ 5/18, 1:41 | ||
29 | *****************************************************************************/ | ||
30 | |||
31 | #include "table.h" | ||
32 | |||
33 | u32 rtl8192sephy_reg_2t2rarray[PHY_REG_2T2RARRAYLENGTH] = { | ||
34 | 0x01c, 0x07000000, | ||
35 | 0x800, 0x00040000, | ||
36 | 0x804, 0x00008003, | ||
37 | 0x808, 0x0000fc00, | ||
38 | 0x80c, 0x0000000a, | ||
39 | 0x810, 0x10005088, | ||
40 | 0x814, 0x020c3d10, | ||
41 | 0x818, 0x00200185, | ||
42 | 0x81c, 0x00000000, | ||
43 | 0x820, 0x01000000, | ||
44 | 0x824, 0x00390004, | ||
45 | 0x828, 0x01000000, | ||
46 | 0x82c, 0x00390004, | ||
47 | 0x830, 0x00000004, | ||
48 | 0x834, 0x00690200, | ||
49 | 0x838, 0x00000004, | ||
50 | 0x83c, 0x00690200, | ||
51 | 0x840, 0x00010000, | ||
52 | 0x844, 0x00010000, | ||
53 | 0x848, 0x00000000, | ||
54 | 0x84c, 0x00000000, | ||
55 | 0x850, 0x00000000, | ||
56 | 0x854, 0x00000000, | ||
57 | 0x858, 0x48484848, | ||
58 | 0x85c, 0x65a965a9, | ||
59 | 0x860, 0x0f7f0130, | ||
60 | 0x864, 0x0f7f0130, | ||
61 | 0x868, 0x0f7f0130, | ||
62 | 0x86c, 0x0f7f0130, | ||
63 | 0x870, 0x03000700, | ||
64 | 0x874, 0x03000300, | ||
65 | 0x878, 0x00020002, | ||
66 | 0x87c, 0x004f0201, | ||
67 | 0x880, 0xa8300ac1, | ||
68 | 0x884, 0x00000058, | ||
69 | 0x888, 0x00000008, | ||
70 | 0x88c, 0x00000004, | ||
71 | 0x890, 0x00000000, | ||
72 | 0x894, 0xfffffffe, | ||
73 | 0x898, 0x40302010, | ||
74 | 0x89c, 0x00706050, | ||
75 | 0x8b0, 0x00000000, | ||
76 | 0x8e0, 0x00000000, | ||
77 | 0x8e4, 0x00000000, | ||
78 | 0xe00, 0x30333333, | ||
79 | 0xe04, 0x2a2d2e2f, | ||
80 | 0xe08, 0x00003232, | ||
81 | 0xe10, 0x30333333, | ||
82 | 0xe14, 0x2a2d2e2f, | ||
83 | 0xe18, 0x30333333, | ||
84 | 0xe1c, 0x2a2d2e2f, | ||
85 | 0xe30, 0x01007c00, | ||
86 | 0xe34, 0x01004800, | ||
87 | 0xe38, 0x1000dc1f, | ||
88 | 0xe3c, 0x10008c1f, | ||
89 | 0xe40, 0x021400a0, | ||
90 | 0xe44, 0x281600a0, | ||
91 | 0xe48, 0xf8000001, | ||
92 | 0xe4c, 0x00002910, | ||
93 | 0xe50, 0x01007c00, | ||
94 | 0xe54, 0x01004800, | ||
95 | 0xe58, 0x1000dc1f, | ||
96 | 0xe5c, 0x10008c1f, | ||
97 | 0xe60, 0x021400a0, | ||
98 | 0xe64, 0x281600a0, | ||
99 | 0xe6c, 0x00002910, | ||
100 | 0xe70, 0x31ed92fb, | ||
101 | 0xe74, 0x361536fb, | ||
102 | 0xe78, 0x361536fb, | ||
103 | 0xe7c, 0x361536fb, | ||
104 | 0xe80, 0x361536fb, | ||
105 | 0xe84, 0x000d92fb, | ||
106 | 0xe88, 0x000d92fb, | ||
107 | 0xe8c, 0x31ed92fb, | ||
108 | 0xed0, 0x31ed92fb, | ||
109 | 0xed4, 0x31ed92fb, | ||
110 | 0xed8, 0x000d92fb, | ||
111 | 0xedc, 0x000d92fb, | ||
112 | 0xee0, 0x000d92fb, | ||
113 | 0xee4, 0x015e5448, | ||
114 | 0xee8, 0x21555448, | ||
115 | 0x900, 0x00000000, | ||
116 | 0x904, 0x00000023, | ||
117 | 0x908, 0x00000000, | ||
118 | 0x90c, 0x01121313, | ||
119 | 0xa00, 0x00d047c8, | ||
120 | 0xa04, 0x80ff0008, | ||
121 | 0xa08, 0x8ccd8300, | ||
122 | 0xa0c, 0x2e62120f, | ||
123 | 0xa10, 0x9500bb78, | ||
124 | 0xa14, 0x11144028, | ||
125 | 0xa18, 0x00881117, | ||
126 | 0xa1c, 0x89140f00, | ||
127 | 0xa20, 0x1a1b0000, | ||
128 | 0xa24, 0x090e1317, | ||
129 | 0xa28, 0x00000204, | ||
130 | 0xa2c, 0x10d30000, | ||
131 | 0xc00, 0x40071d40, | ||
132 | 0xc04, 0x00a05633, | ||
133 | 0xc08, 0x000000e4, | ||
134 | 0xc0c, 0x6c6c6c6c, | ||
135 | 0xc10, 0x08800000, | ||
136 | 0xc14, 0x40000100, | ||
137 | 0xc18, 0x08000000, | ||
138 | 0xc1c, 0x40000100, | ||
139 | 0xc20, 0x08000000, | ||
140 | 0xc24, 0x40000100, | ||
141 | 0xc28, 0x08000000, | ||
142 | 0xc2c, 0x40000100, | ||
143 | 0xc30, 0x6de9ac44, | ||
144 | 0xc34, 0x469652cf, | ||
145 | 0xc38, 0x49795994, | ||
146 | 0xc3c, 0x0a979764, | ||
147 | 0xc40, 0x1f7c403f, | ||
148 | 0xc44, 0x000100b7, | ||
149 | 0xc48, 0xec020000, | ||
150 | 0xc4c, 0x007f037f, | ||
151 | 0xc50, 0x69543420, | ||
152 | 0xc54, 0x433c0094, | ||
153 | 0xc58, 0x69543420, | ||
154 | 0xc5c, 0x433c0094, | ||
155 | 0xc60, 0x69543420, | ||
156 | 0xc64, 0x433c0094, | ||
157 | 0xc68, 0x69543420, | ||
158 | 0xc6c, 0x433c0094, | ||
159 | 0xc70, 0x2c7f000d, | ||
160 | 0xc74, 0x0186155b, | ||
161 | 0xc78, 0x0000001f, | ||
162 | 0xc7c, 0x00b91612, | ||
163 | 0xc80, 0x40000100, | ||
164 | 0xc84, 0x20f60000, | ||
165 | 0xc88, 0x20000080, | ||
166 | 0xc8c, 0x20200000, | ||
167 | 0xc90, 0x40000100, | ||
168 | 0xc94, 0x00000000, | ||
169 | 0xc98, 0x40000100, | ||
170 | 0xc9c, 0x00000000, | ||
171 | 0xca0, 0x00492492, | ||
172 | 0xca4, 0x00000000, | ||
173 | 0xca8, 0x00000000, | ||
174 | 0xcac, 0x00000000, | ||
175 | 0xcb0, 0x00000000, | ||
176 | 0xcb4, 0x00000000, | ||
177 | 0xcb8, 0x00000000, | ||
178 | 0xcbc, 0x28000000, | ||
179 | 0xcc0, 0x00000000, | ||
180 | 0xcc4, 0x00000000, | ||
181 | 0xcc8, 0x00000000, | ||
182 | 0xccc, 0x00000000, | ||
183 | 0xcd0, 0x00000000, | ||
184 | 0xcd4, 0x00000000, | ||
185 | 0xcd8, 0x64b22427, | ||
186 | 0xcdc, 0x00766932, | ||
187 | 0xce0, 0x00222222, | ||
188 | 0xce4, 0x00000000, | ||
189 | 0xce8, 0x37644302, | ||
190 | 0xcec, 0x2f97d40c, | ||
191 | 0xd00, 0x00000750, | ||
192 | 0xd04, 0x00000403, | ||
193 | 0xd08, 0x0000907f, | ||
194 | 0xd0c, 0x00000001, | ||
195 | 0xd10, 0xa0633333, | ||
196 | 0xd14, 0x33333c63, | ||
197 | 0xd18, 0x6a8f5b6b, | ||
198 | 0xd1c, 0x00000000, | ||
199 | 0xd20, 0x00000000, | ||
200 | 0xd24, 0x00000000, | ||
201 | 0xd28, 0x00000000, | ||
202 | 0xd2c, 0xcc979975, | ||
203 | 0xd30, 0x00000000, | ||
204 | 0xd34, 0x00000000, | ||
205 | 0xd38, 0x00000000, | ||
206 | 0xd3c, 0x00027293, | ||
207 | 0xd40, 0x00000000, | ||
208 | 0xd44, 0x00000000, | ||
209 | 0xd48, 0x00000000, | ||
210 | 0xd50, 0x6437140a, | ||
211 | 0xd54, 0x024dbd02, | ||
212 | 0xd58, 0x00000000, | ||
213 | 0xd5c, 0x30032064, | ||
214 | 0xd60, 0x4653de68, | ||
215 | 0xd64, 0x00518a3c, | ||
216 | 0xd68, 0x00002101, | ||
217 | 0xf14, 0x00000003, | ||
218 | 0xf4c, 0x00000000, | ||
219 | 0xf00, 0x00000300, | ||
220 | }; | ||
221 | |||
222 | u32 rtl8192sephy_changeto_1t1rarray[PHY_CHANGETO_1T1RARRAYLENGTH] = { | ||
223 | 0x844, 0xffffffff, 0x00010000, | ||
224 | 0x804, 0x0000000f, 0x00000001, | ||
225 | 0x824, 0x00f0000f, 0x00300004, | ||
226 | 0x82c, 0x00f0000f, 0x00100002, | ||
227 | 0x870, 0x04000000, 0x00000001, | ||
228 | 0x864, 0x00000400, 0x00000000, | ||
229 | 0x878, 0x000f000f, 0x00000002, | ||
230 | 0xe74, 0x0f000000, 0x00000002, | ||
231 | 0xe78, 0x0f000000, 0x00000002, | ||
232 | 0xe7c, 0x0f000000, 0x00000002, | ||
233 | 0xe80, 0x0f000000, 0x00000002, | ||
234 | 0x90c, 0x000000ff, 0x00000011, | ||
235 | 0xc04, 0x000000ff, 0x00000011, | ||
236 | 0xd04, 0x0000000f, 0x00000001, | ||
237 | 0x1f4, 0xffff0000, 0x00007777, | ||
238 | 0x234, 0xf8000000, 0x0000000a, | ||
239 | }; | ||
240 | |||
241 | u32 rtl8192sephy_changeto_1t2rarray[PHY_CHANGETO_1T2RARRAYLENGTH] = { | ||
242 | 0x804, 0x0000000f, 0x00000003, | ||
243 | 0x824, 0x00f0000f, 0x00300004, | ||
244 | 0x82c, 0x00f0000f, 0x00300002, | ||
245 | 0x870, 0x04000000, 0x00000001, | ||
246 | 0x864, 0x00000400, 0x00000000, | ||
247 | 0x878, 0x000f000f, 0x00000002, | ||
248 | 0xe74, 0x0f000000, 0x00000002, | ||
249 | 0xe78, 0x0f000000, 0x00000002, | ||
250 | 0xe7c, 0x0f000000, 0x00000002, | ||
251 | 0xe80, 0x0f000000, 0x00000002, | ||
252 | 0x90c, 0x000000ff, 0x00000011, | ||
253 | 0xc04, 0x000000ff, 0x00000033, | ||
254 | 0xd04, 0x0000000f, 0x00000003, | ||
255 | 0x1f4, 0xffff0000, 0x00007777, | ||
256 | 0x234, 0xf8000000, 0x0000000a, | ||
257 | }; | ||
258 | |||
259 | u32 rtl8192sephy_reg_array_pg[PHY_REG_ARRAY_PGLENGTH] = { | ||
260 | 0xe00, 0xffffffff, 0x06090909, | ||
261 | 0xe04, 0xffffffff, 0x00030406, | ||
262 | 0xe08, 0x0000ff00, 0x00000000, | ||
263 | 0xe10, 0xffffffff, 0x0a0c0d0e, | ||
264 | 0xe14, 0xffffffff, 0x04070809, | ||
265 | 0xe18, 0xffffffff, 0x0a0c0d0e, | ||
266 | 0xe1c, 0xffffffff, 0x04070809, | ||
267 | 0xe00, 0xffffffff, 0x04040404, | ||
268 | 0xe04, 0xffffffff, 0x00020204, | ||
269 | 0xe08, 0x0000ff00, 0x00000000, | ||
270 | 0xe10, 0xffffffff, 0x02040404, | ||
271 | 0xe14, 0xffffffff, 0x00000002, | ||
272 | 0xe18, 0xffffffff, 0x02040404, | ||
273 | 0xe1c, 0xffffffff, 0x00000002, | ||
274 | 0xe00, 0xffffffff, 0x04040404, | ||
275 | 0xe04, 0xffffffff, 0x00020204, | ||
276 | 0xe08, 0x0000ff00, 0x00000000, | ||
277 | 0xe10, 0xffffffff, 0x02040404, | ||
278 | 0xe14, 0xffffffff, 0x00000002, | ||
279 | 0xe18, 0xffffffff, 0x02040404, | ||
280 | 0xe1c, 0xffffffff, 0x00000002, | ||
281 | 0xe00, 0xffffffff, 0x02020202, | ||
282 | 0xe04, 0xffffffff, 0x00020202, | ||
283 | 0xe08, 0x0000ff00, 0x00000000, | ||
284 | 0xe10, 0xffffffff, 0x02020202, | ||
285 | 0xe14, 0xffffffff, 0x00000002, | ||
286 | 0xe18, 0xffffffff, 0x02020202, | ||
287 | 0xe1c, 0xffffffff, 0x00000002, | ||
288 | }; | ||
289 | |||
290 | u32 rtl8192seradioa_1t_array[RADIOA_1T_ARRAYLENGTH] = { | ||
291 | 0x000, 0x00030159, | ||
292 | 0x001, 0x00030250, | ||
293 | 0x002, 0x00010000, | ||
294 | 0x010, 0x0008000f, | ||
295 | 0x011, 0x000231fc, | ||
296 | 0x010, 0x000c000f, | ||
297 | 0x011, 0x0003f9f8, | ||
298 | 0x010, 0x0002000f, | ||
299 | 0x011, 0x00020101, | ||
300 | 0x014, 0x0001093e, | ||
301 | 0x014, 0x0009093e, | ||
302 | 0x015, 0x0000f8f4, | ||
303 | 0x017, 0x000f6500, | ||
304 | 0x01a, 0x00013056, | ||
305 | 0x01b, 0x00060000, | ||
306 | 0x01c, 0x00000300, | ||
307 | 0x01e, 0x00031059, | ||
308 | 0x021, 0x00054000, | ||
309 | 0x022, 0x0000083c, | ||
310 | 0x023, 0x00001558, | ||
311 | 0x024, 0x00000060, | ||
312 | 0x025, 0x00022583, | ||
313 | 0x026, 0x0000f200, | ||
314 | 0x027, 0x000eacf1, | ||
315 | 0x028, 0x0009bd54, | ||
316 | 0x029, 0x00004582, | ||
317 | 0x02a, 0x00000001, | ||
318 | 0x02b, 0x00021334, | ||
319 | 0x02a, 0x00000000, | ||
320 | 0x02b, 0x0000000a, | ||
321 | 0x02a, 0x00000001, | ||
322 | 0x02b, 0x00000808, | ||
323 | 0x02b, 0x00053333, | ||
324 | 0x02c, 0x0000000c, | ||
325 | 0x02a, 0x00000002, | ||
326 | 0x02b, 0x00000808, | ||
327 | 0x02b, 0x0005b333, | ||
328 | 0x02c, 0x0000000d, | ||
329 | 0x02a, 0x00000003, | ||
330 | 0x02b, 0x00000808, | ||
331 | 0x02b, 0x00063333, | ||
332 | 0x02c, 0x0000000d, | ||
333 | 0x02a, 0x00000004, | ||
334 | 0x02b, 0x00000808, | ||
335 | 0x02b, 0x0006b333, | ||
336 | 0x02c, 0x0000000d, | ||
337 | 0x02a, 0x00000005, | ||
338 | 0x02b, 0x00000709, | ||
339 | 0x02b, 0x00053333, | ||
340 | 0x02c, 0x0000000d, | ||
341 | 0x02a, 0x00000006, | ||
342 | 0x02b, 0x00000709, | ||
343 | 0x02b, 0x0005b333, | ||
344 | 0x02c, 0x0000000d, | ||
345 | 0x02a, 0x00000007, | ||
346 | 0x02b, 0x00000709, | ||
347 | 0x02b, 0x00063333, | ||
348 | 0x02c, 0x0000000d, | ||
349 | 0x02a, 0x00000008, | ||
350 | 0x02b, 0x00000709, | ||
351 | 0x02b, 0x0006b333, | ||
352 | 0x02c, 0x0000000d, | ||
353 | 0x02a, 0x00000009, | ||
354 | 0x02b, 0x0000060a, | ||
355 | 0x02b, 0x00053333, | ||
356 | 0x02c, 0x0000000d, | ||
357 | 0x02a, 0x0000000a, | ||
358 | 0x02b, 0x0000060a, | ||
359 | 0x02b, 0x0005b333, | ||
360 | 0x02c, 0x0000000d, | ||
361 | 0x02a, 0x0000000b, | ||
362 | 0x02b, 0x0000060a, | ||
363 | 0x02b, 0x00063333, | ||
364 | 0x02c, 0x0000000d, | ||
365 | 0x02a, 0x0000000c, | ||
366 | 0x02b, 0x0000060a, | ||
367 | 0x02b, 0x0006b333, | ||
368 | 0x02c, 0x0000000d, | ||
369 | 0x02a, 0x0000000d, | ||
370 | 0x02b, 0x0000050b, | ||
371 | 0x02b, 0x00053333, | ||
372 | 0x02c, 0x0000000d, | ||
373 | 0x02a, 0x0000000e, | ||
374 | 0x02b, 0x0000050b, | ||
375 | 0x02b, 0x00066623, | ||
376 | 0x02c, 0x0000001a, | ||
377 | 0x02a, 0x000e4000, | ||
378 | 0x030, 0x00020000, | ||
379 | 0x031, 0x000b9631, | ||
380 | 0x032, 0x0000130d, | ||
381 | 0x033, 0x00000187, | ||
382 | 0x013, 0x00019e6c, | ||
383 | 0x013, 0x00015e94, | ||
384 | 0x000, 0x00010159, | ||
385 | 0x018, 0x0000f401, | ||
386 | 0x0fe, 0x00000000, | ||
387 | 0x01e, 0x0003105b, | ||
388 | 0x0fe, 0x00000000, | ||
389 | 0x000, 0x00030159, | ||
390 | 0x010, 0x0004000f, | ||
391 | 0x011, 0x000203f9, | ||
392 | }; | ||
393 | |||
394 | u32 rtl8192seradiob_array[RADIOB_ARRAYLENGTH] = { | ||
395 | 0x000, 0x00030159, | ||
396 | 0x001, 0x00001041, | ||
397 | 0x002, 0x00011000, | ||
398 | 0x005, 0x00080fc0, | ||
399 | 0x007, 0x000fc803, | ||
400 | 0x013, 0x00017cb0, | ||
401 | 0x013, 0x00011cc0, | ||
402 | 0x013, 0x0000dc60, | ||
403 | 0x013, 0x00008c60, | ||
404 | 0x013, 0x00004450, | ||
405 | 0x013, 0x00000020, | ||
406 | }; | ||
407 | |||
408 | u32 rtl8192seradiob_gm_array[RADIOB_GM_ARRAYLENGTH] = { | ||
409 | 0x000, 0x00030159, | ||
410 | 0x001, 0x00001041, | ||
411 | 0x002, 0x00011000, | ||
412 | 0x005, 0x00080fc0, | ||
413 | 0x007, 0x000fc803, | ||
414 | }; | ||
415 | |||
416 | u32 rtl8192semac_2t_array[MAC_2T_ARRAYLENGTH] = { | ||
417 | 0x020, 0x00000035, | ||
418 | 0x048, 0x0000000e, | ||
419 | 0x049, 0x000000f0, | ||
420 | 0x04a, 0x00000077, | ||
421 | 0x04b, 0x00000083, | ||
422 | 0x0b5, 0x00000021, | ||
423 | 0x0dc, 0x000000ff, | ||
424 | 0x0dd, 0x000000ff, | ||
425 | 0x0de, 0x000000ff, | ||
426 | 0x0df, 0x000000ff, | ||
427 | 0x116, 0x00000000, | ||
428 | 0x117, 0x00000000, | ||
429 | 0x118, 0x00000000, | ||
430 | 0x119, 0x00000000, | ||
431 | 0x11a, 0x00000000, | ||
432 | 0x11b, 0x00000000, | ||
433 | 0x11c, 0x00000000, | ||
434 | 0x11d, 0x00000000, | ||
435 | 0x160, 0x0000000b, | ||
436 | 0x161, 0x0000000b, | ||
437 | 0x162, 0x0000000b, | ||
438 | 0x163, 0x0000000b, | ||
439 | 0x164, 0x0000000b, | ||
440 | 0x165, 0x0000000b, | ||
441 | 0x166, 0x0000000b, | ||
442 | 0x167, 0x0000000b, | ||
443 | 0x168, 0x0000000b, | ||
444 | 0x169, 0x0000000b, | ||
445 | 0x16a, 0x0000000b, | ||
446 | 0x16b, 0x0000000b, | ||
447 | 0x16c, 0x0000000b, | ||
448 | 0x16d, 0x0000000b, | ||
449 | 0x16e, 0x0000000b, | ||
450 | 0x16f, 0x0000000b, | ||
451 | 0x170, 0x0000000b, | ||
452 | 0x171, 0x0000000b, | ||
453 | 0x172, 0x0000000b, | ||
454 | 0x173, 0x0000000b, | ||
455 | 0x174, 0x0000000b, | ||
456 | 0x175, 0x0000000b, | ||
457 | 0x176, 0x0000000b, | ||
458 | 0x177, 0x0000000b, | ||
459 | 0x178, 0x0000000b, | ||
460 | 0x179, 0x0000000b, | ||
461 | 0x17a, 0x0000000b, | ||
462 | 0x17b, 0x0000000b, | ||
463 | 0x17c, 0x0000000b, | ||
464 | 0x17d, 0x0000000b, | ||
465 | 0x17e, 0x0000000b, | ||
466 | 0x17f, 0x0000000b, | ||
467 | 0x236, 0x0000000c, | ||
468 | 0x503, 0x00000022, | ||
469 | 0x560, 0x00000000, | ||
470 | }; | ||
471 | |||
472 | u32 rtl8192seagctab_array[AGCTAB_ARRAYLENGTH] = { | ||
473 | 0xc78, 0x7f000001, | ||
474 | 0xc78, 0x7f010001, | ||
475 | 0xc78, 0x7e020001, | ||
476 | 0xc78, 0x7d030001, | ||
477 | 0xc78, 0x7c040001, | ||
478 | 0xc78, 0x7b050001, | ||
479 | 0xc78, 0x7a060001, | ||
480 | 0xc78, 0x79070001, | ||
481 | 0xc78, 0x78080001, | ||
482 | 0xc78, 0x77090001, | ||
483 | 0xc78, 0x760a0001, | ||
484 | 0xc78, 0x750b0001, | ||
485 | 0xc78, 0x740c0001, | ||
486 | 0xc78, 0x730d0001, | ||
487 | 0xc78, 0x720e0001, | ||
488 | 0xc78, 0x710f0001, | ||
489 | 0xc78, 0x70100001, | ||
490 | 0xc78, 0x6f110001, | ||
491 | 0xc78, 0x6f120001, | ||
492 | 0xc78, 0x6e130001, | ||
493 | 0xc78, 0x6d140001, | ||
494 | 0xc78, 0x6d150001, | ||
495 | 0xc78, 0x6c160001, | ||
496 | 0xc78, 0x6b170001, | ||
497 | 0xc78, 0x6a180001, | ||
498 | 0xc78, 0x6a190001, | ||
499 | 0xc78, 0x691a0001, | ||
500 | 0xc78, 0x681b0001, | ||
501 | 0xc78, 0x671c0001, | ||
502 | 0xc78, 0x661d0001, | ||
503 | 0xc78, 0x651e0001, | ||
504 | 0xc78, 0x641f0001, | ||
505 | 0xc78, 0x63200001, | ||
506 | 0xc78, 0x4c210001, | ||
507 | 0xc78, 0x4b220001, | ||
508 | 0xc78, 0x4a230001, | ||
509 | 0xc78, 0x49240001, | ||
510 | 0xc78, 0x48250001, | ||
511 | 0xc78, 0x47260001, | ||
512 | 0xc78, 0x46270001, | ||
513 | 0xc78, 0x45280001, | ||
514 | 0xc78, 0x44290001, | ||
515 | 0xc78, 0x2c2a0001, | ||
516 | 0xc78, 0x2b2b0001, | ||
517 | 0xc78, 0x2a2c0001, | ||
518 | 0xc78, 0x292d0001, | ||
519 | 0xc78, 0x282e0001, | ||
520 | 0xc78, 0x272f0001, | ||
521 | 0xc78, 0x26300001, | ||
522 | 0xc78, 0x25310001, | ||
523 | 0xc78, 0x24320001, | ||
524 | 0xc78, 0x23330001, | ||
525 | 0xc78, 0x22340001, | ||
526 | 0xc78, 0x09350001, | ||
527 | 0xc78, 0x08360001, | ||
528 | 0xc78, 0x07370001, | ||
529 | 0xc78, 0x06380001, | ||
530 | 0xc78, 0x05390001, | ||
531 | 0xc78, 0x043a0001, | ||
532 | 0xc78, 0x033b0001, | ||
533 | 0xc78, 0x023c0001, | ||
534 | 0xc78, 0x013d0001, | ||
535 | 0xc78, 0x003e0001, | ||
536 | 0xc78, 0x003f0001, | ||
537 | 0xc78, 0x7f400001, | ||
538 | 0xc78, 0x7f410001, | ||
539 | 0xc78, 0x7e420001, | ||
540 | 0xc78, 0x7d430001, | ||
541 | 0xc78, 0x7c440001, | ||
542 | 0xc78, 0x7b450001, | ||
543 | 0xc78, 0x7a460001, | ||
544 | 0xc78, 0x79470001, | ||
545 | 0xc78, 0x78480001, | ||
546 | 0xc78, 0x77490001, | ||
547 | 0xc78, 0x764a0001, | ||
548 | 0xc78, 0x754b0001, | ||
549 | 0xc78, 0x744c0001, | ||
550 | 0xc78, 0x734d0001, | ||
551 | 0xc78, 0x724e0001, | ||
552 | 0xc78, 0x714f0001, | ||
553 | 0xc78, 0x70500001, | ||
554 | 0xc78, 0x6f510001, | ||
555 | 0xc78, 0x6f520001, | ||
556 | 0xc78, 0x6e530001, | ||
557 | 0xc78, 0x6d540001, | ||
558 | 0xc78, 0x6d550001, | ||
559 | 0xc78, 0x6c560001, | ||
560 | 0xc78, 0x6b570001, | ||
561 | 0xc78, 0x6a580001, | ||
562 | 0xc78, 0x6a590001, | ||
563 | 0xc78, 0x695a0001, | ||
564 | 0xc78, 0x685b0001, | ||
565 | 0xc78, 0x675c0001, | ||
566 | 0xc78, 0x665d0001, | ||
567 | 0xc78, 0x655e0001, | ||
568 | 0xc78, 0x645f0001, | ||
569 | 0xc78, 0x63600001, | ||
570 | 0xc78, 0x4c610001, | ||
571 | 0xc78, 0x4b620001, | ||
572 | 0xc78, 0x4a630001, | ||
573 | 0xc78, 0x49640001, | ||
574 | 0xc78, 0x48650001, | ||
575 | 0xc78, 0x47660001, | ||
576 | 0xc78, 0x46670001, | ||
577 | 0xc78, 0x45680001, | ||
578 | 0xc78, 0x44690001, | ||
579 | 0xc78, 0x2c6a0001, | ||
580 | 0xc78, 0x2b6b0001, | ||
581 | 0xc78, 0x2a6c0001, | ||
582 | 0xc78, 0x296d0001, | ||
583 | 0xc78, 0x286e0001, | ||
584 | 0xc78, 0x276f0001, | ||
585 | 0xc78, 0x26700001, | ||
586 | 0xc78, 0x25710001, | ||
587 | 0xc78, 0x24720001, | ||
588 | 0xc78, 0x23730001, | ||
589 | 0xc78, 0x22740001, | ||
590 | 0xc78, 0x09750001, | ||
591 | 0xc78, 0x08760001, | ||
592 | 0xc78, 0x07770001, | ||
593 | 0xc78, 0x06780001, | ||
594 | 0xc78, 0x05790001, | ||
595 | 0xc78, 0x047a0001, | ||
596 | 0xc78, 0x037b0001, | ||
597 | 0xc78, 0x027c0001, | ||
598 | 0xc78, 0x017d0001, | ||
599 | 0xc78, 0x007e0001, | ||
600 | 0xc78, 0x007f0001, | ||
601 | 0xc78, 0x3000001e, | ||
602 | 0xc78, 0x3001001e, | ||
603 | 0xc78, 0x3002001e, | ||
604 | 0xc78, 0x3003001e, | ||
605 | 0xc78, 0x3004001e, | ||
606 | 0xc78, 0x3405001e, | ||
607 | 0xc78, 0x3806001e, | ||
608 | 0xc78, 0x3e07001e, | ||
609 | 0xc78, 0x3e08001e, | ||
610 | 0xc78, 0x4409001e, | ||
611 | 0xc78, 0x460a001e, | ||
612 | 0xc78, 0x480b001e, | ||
613 | 0xc78, 0x480c001e, | ||
614 | 0xc78, 0x4e0d001e, | ||
615 | 0xc78, 0x560e001e, | ||
616 | 0xc78, 0x5a0f001e, | ||
617 | 0xc78, 0x5e10001e, | ||
618 | 0xc78, 0x6211001e, | ||
619 | 0xc78, 0x6c12001e, | ||
620 | 0xc78, 0x7213001e, | ||
621 | 0xc78, 0x7214001e, | ||
622 | 0xc78, 0x7215001e, | ||
623 | 0xc78, 0x7216001e, | ||
624 | 0xc78, 0x7217001e, | ||
625 | 0xc78, 0x7218001e, | ||
626 | 0xc78, 0x7219001e, | ||
627 | 0xc78, 0x721a001e, | ||
628 | 0xc78, 0x721b001e, | ||
629 | 0xc78, 0x721c001e, | ||
630 | 0xc78, 0x721d001e, | ||
631 | 0xc78, 0x721e001e, | ||
632 | 0xc78, 0x721f001e, | ||
633 | }; | ||
634 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/table.h b/drivers/net/wireless/rtlwifi/rtl8192se/table.h new file mode 100644 index 000000000000..b4ed6d951ebb --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/table.h | |||
@@ -0,0 +1,49 @@ | |||
1 | /****************************************************************************** | ||
2 | * Copyright(c) 2008 - 2010 Realtek Corporation. All rights reserved. | ||
3 | * | ||
4 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
5 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
6 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
7 | * more details. | ||
8 | * | ||
9 | * You should have received a copy of the GNU General Public License along with | ||
10 | * this program; if not, write to the Free Software Foundation, Inc., | ||
11 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
12 | * | ||
13 | * The full GNU General Public License is included in this distribution in the | ||
14 | * file called LICENSE. | ||
15 | * | ||
16 | * Contact Information: | ||
17 | * wlanfae <wlanfae@realtek.com> | ||
18 | * | ||
19 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
20 | * | ||
21 | ******************************************************************************/ | ||
22 | #ifndef __INC_HAL8192SE_FW_IMG_H | ||
23 | #define __INC_HAL8192SE_FW_IMG_H | ||
24 | |||
25 | #include <linux/types.h> | ||
26 | |||
27 | /*Created on 2010/ 4/12, 5:56*/ | ||
28 | |||
29 | #define PHY_REG_2T2RARRAYLENGTH 372 | ||
30 | extern u32 rtl8192sephy_reg_2t2rarray[PHY_REG_2T2RARRAYLENGTH]; | ||
31 | #define PHY_CHANGETO_1T1RARRAYLENGTH 48 | ||
32 | extern u32 rtl8192sephy_changeto_1t1rarray[PHY_CHANGETO_1T1RARRAYLENGTH]; | ||
33 | #define PHY_CHANGETO_1T2RARRAYLENGTH 45 | ||
34 | extern u32 rtl8192sephy_changeto_1t2rarray[PHY_CHANGETO_1T2RARRAYLENGTH]; | ||
35 | #define PHY_REG_ARRAY_PGLENGTH 84 | ||
36 | extern u32 rtl8192sephy_reg_array_pg[PHY_REG_ARRAY_PGLENGTH]; | ||
37 | #define RADIOA_1T_ARRAYLENGTH 202 | ||
38 | extern u32 rtl8192seradioa_1t_array[RADIOA_1T_ARRAYLENGTH]; | ||
39 | #define RADIOB_ARRAYLENGTH 22 | ||
40 | extern u32 rtl8192seradiob_array[RADIOB_ARRAYLENGTH]; | ||
41 | #define RADIOB_GM_ARRAYLENGTH 10 | ||
42 | extern u32 rtl8192seradiob_gm_array[RADIOB_GM_ARRAYLENGTH]; | ||
43 | #define MAC_2T_ARRAYLENGTH 106 | ||
44 | extern u32 rtl8192semac_2t_array[MAC_2T_ARRAYLENGTH]; | ||
45 | #define AGCTAB_ARRAYLENGTH 320 | ||
46 | extern u32 rtl8192seagctab_array[AGCTAB_ARRAYLENGTH]; | ||
47 | |||
48 | #endif | ||
49 | |||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/trx.c b/drivers/net/wireless/rtlwifi/rtl8192se/trx.c new file mode 100644 index 000000000000..5cf442373d46 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/trx.c | |||
@@ -0,0 +1,976 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #include "../wifi.h" | ||
31 | #include "../pci.h" | ||
32 | #include "../base.h" | ||
33 | #include "reg.h" | ||
34 | #include "def.h" | ||
35 | #include "phy.h" | ||
36 | #include "fw.h" | ||
37 | #include "trx.h" | ||
38 | #include "led.h" | ||
39 | |||
40 | static u8 _rtl92se_map_hwqueue_to_fwqueue(struct sk_buff *skb, u8 skb_queue) | ||
41 | { | ||
42 | __le16 fc = rtl_get_fc(skb); | ||
43 | |||
44 | if (unlikely(ieee80211_is_beacon(fc))) | ||
45 | return QSLT_BEACON; | ||
46 | if (ieee80211_is_mgmt(fc)) | ||
47 | return QSLT_MGNT; | ||
48 | if (ieee80211_is_nullfunc(fc)) | ||
49 | return QSLT_HIGH; | ||
50 | |||
51 | return skb->priority; | ||
52 | } | ||
53 | |||
54 | static int _rtl92se_rate_mapping(bool isht, u8 desc_rate, bool first_ampdu) | ||
55 | { | ||
56 | int rate_idx = 0; | ||
57 | |||
58 | if (first_ampdu) { | ||
59 | if (false == isht) { | ||
60 | switch (desc_rate) { | ||
61 | case DESC92S_RATE1M: | ||
62 | rate_idx = 0; | ||
63 | break; | ||
64 | case DESC92S_RATE2M: | ||
65 | rate_idx = 1; | ||
66 | break; | ||
67 | case DESC92S_RATE5_5M: | ||
68 | rate_idx = 2; | ||
69 | break; | ||
70 | case DESC92S_RATE11M: | ||
71 | rate_idx = 3; | ||
72 | break; | ||
73 | case DESC92S_RATE6M: | ||
74 | rate_idx = 4; | ||
75 | break; | ||
76 | case DESC92S_RATE9M: | ||
77 | rate_idx = 5; | ||
78 | break; | ||
79 | case DESC92S_RATE12M: | ||
80 | rate_idx = 6; | ||
81 | break; | ||
82 | case DESC92S_RATE18M: | ||
83 | rate_idx = 7; | ||
84 | break; | ||
85 | case DESC92S_RATE24M: | ||
86 | rate_idx = 8; | ||
87 | break; | ||
88 | case DESC92S_RATE36M: | ||
89 | rate_idx = 9; | ||
90 | break; | ||
91 | case DESC92S_RATE48M: | ||
92 | rate_idx = 10; | ||
93 | break; | ||
94 | case DESC92S_RATE54M: | ||
95 | rate_idx = 11; | ||
96 | break; | ||
97 | default: | ||
98 | rate_idx = 0; | ||
99 | break; | ||
100 | } | ||
101 | } else { | ||
102 | rate_idx = 11; | ||
103 | } | ||
104 | |||
105 | return rate_idx; | ||
106 | } | ||
107 | |||
108 | switch (desc_rate) { | ||
109 | case DESC92S_RATE1M: | ||
110 | rate_idx = 0; | ||
111 | break; | ||
112 | case DESC92S_RATE2M: | ||
113 | rate_idx = 1; | ||
114 | break; | ||
115 | case DESC92S_RATE5_5M: | ||
116 | rate_idx = 2; | ||
117 | break; | ||
118 | case DESC92S_RATE11M: | ||
119 | rate_idx = 3; | ||
120 | break; | ||
121 | case DESC92S_RATE6M: | ||
122 | rate_idx = 4; | ||
123 | break; | ||
124 | case DESC92S_RATE9M: | ||
125 | rate_idx = 5; | ||
126 | break; | ||
127 | case DESC92S_RATE12M: | ||
128 | rate_idx = 6; | ||
129 | break; | ||
130 | case DESC92S_RATE18M: | ||
131 | rate_idx = 7; | ||
132 | break; | ||
133 | case DESC92S_RATE24M: | ||
134 | rate_idx = 8; | ||
135 | break; | ||
136 | case DESC92S_RATE36M: | ||
137 | rate_idx = 9; | ||
138 | break; | ||
139 | case DESC92S_RATE48M: | ||
140 | rate_idx = 10; | ||
141 | break; | ||
142 | case DESC92S_RATE54M: | ||
143 | rate_idx = 11; | ||
144 | break; | ||
145 | default: | ||
146 | rate_idx = 11; | ||
147 | break; | ||
148 | } | ||
149 | return rate_idx; | ||
150 | } | ||
151 | |||
152 | static u8 _rtl92s_query_rxpwrpercentage(char antpower) | ||
153 | { | ||
154 | if ((antpower <= -100) || (antpower >= 20)) | ||
155 | return 0; | ||
156 | else if (antpower >= 0) | ||
157 | return 100; | ||
158 | else | ||
159 | return 100 + antpower; | ||
160 | } | ||
161 | |||
162 | static u8 _rtl92s_evm_db_to_percentage(char value) | ||
163 | { | ||
164 | char ret_val; | ||
165 | ret_val = value; | ||
166 | |||
167 | if (ret_val >= 0) | ||
168 | ret_val = 0; | ||
169 | |||
170 | if (ret_val <= -33) | ||
171 | ret_val = -33; | ||
172 | |||
173 | ret_val = 0 - ret_val; | ||
174 | ret_val *= 3; | ||
175 | |||
176 | if (ret_val == 99) | ||
177 | ret_val = 100; | ||
178 | |||
179 | return ret_val; | ||
180 | } | ||
181 | |||
182 | static long _rtl92se_translate_todbm(struct ieee80211_hw *hw, | ||
183 | u8 signal_strength_index) | ||
184 | { | ||
185 | long signal_power; | ||
186 | |||
187 | signal_power = (long)((signal_strength_index + 1) >> 1); | ||
188 | signal_power -= 95; | ||
189 | return signal_power; | ||
190 | } | ||
191 | |||
192 | static long _rtl92se_signal_scale_mapping(struct ieee80211_hw *hw, | ||
193 | long currsig) | ||
194 | { | ||
195 | long retsig = 0; | ||
196 | |||
197 | /* Step 1. Scale mapping. */ | ||
198 | if (currsig > 47) | ||
199 | retsig = 100; | ||
200 | else if (currsig > 14 && currsig <= 47) | ||
201 | retsig = 100 - ((47 - currsig) * 3) / 2; | ||
202 | else if (currsig > 2 && currsig <= 14) | ||
203 | retsig = 48 - ((14 - currsig) * 15) / 7; | ||
204 | else if (currsig >= 0) | ||
205 | retsig = currsig * 9 + 1; | ||
206 | |||
207 | return retsig; | ||
208 | } | ||
209 | |||
210 | |||
211 | static void _rtl92se_query_rxphystatus(struct ieee80211_hw *hw, | ||
212 | struct rtl_stats *pstats, u8 *pdesc, | ||
213 | struct rx_fwinfo *p_drvinfo, | ||
214 | bool packet_match_bssid, | ||
215 | bool packet_toself, | ||
216 | bool packet_beacon) | ||
217 | { | ||
218 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
219 | struct phy_sts_cck_8192s_t *cck_buf; | ||
220 | s8 rx_pwr_all = 0, rx_pwr[4]; | ||
221 | u8 rf_rx_num = 0, evm, pwdb_all; | ||
222 | u8 i, max_spatial_stream; | ||
223 | u32 rssi, total_rssi = 0; | ||
224 | bool in_powersavemode = false; | ||
225 | bool is_cck_rate; | ||
226 | |||
227 | is_cck_rate = RX_HAL_IS_CCK_RATE(pdesc); | ||
228 | pstats->packet_matchbssid = packet_match_bssid; | ||
229 | pstats->packet_toself = packet_toself; | ||
230 | pstats->is_cck = is_cck_rate; | ||
231 | pstats->packet_beacon = packet_beacon; | ||
232 | pstats->is_cck = is_cck_rate; | ||
233 | pstats->rx_mimo_signalquality[0] = -1; | ||
234 | pstats->rx_mimo_signalquality[1] = -1; | ||
235 | |||
236 | if (is_cck_rate) { | ||
237 | u8 report, cck_highpwr; | ||
238 | cck_buf = (struct phy_sts_cck_8192s_t *)p_drvinfo; | ||
239 | |||
240 | if (!in_powersavemode) | ||
241 | cck_highpwr = (u8) rtl_get_bbreg(hw, | ||
242 | RFPGA0_XA_HSSIPARAMETER2, | ||
243 | 0x200); | ||
244 | else | ||
245 | cck_highpwr = false; | ||
246 | |||
247 | if (!cck_highpwr) { | ||
248 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
249 | report = cck_buf->cck_agc_rpt & 0xc0; | ||
250 | report = report >> 6; | ||
251 | switch (report) { | ||
252 | case 0x3: | ||
253 | rx_pwr_all = -40 - (cck_agc_rpt & 0x3e); | ||
254 | break; | ||
255 | case 0x2: | ||
256 | rx_pwr_all = -20 - (cck_agc_rpt & 0x3e); | ||
257 | break; | ||
258 | case 0x1: | ||
259 | rx_pwr_all = -2 - (cck_agc_rpt & 0x3e); | ||
260 | break; | ||
261 | case 0x0: | ||
262 | rx_pwr_all = 14 - (cck_agc_rpt & 0x3e); | ||
263 | break; | ||
264 | } | ||
265 | } else { | ||
266 | u8 cck_agc_rpt = cck_buf->cck_agc_rpt; | ||
267 | report = p_drvinfo->cfosho[0] & 0x60; | ||
268 | report = report >> 5; | ||
269 | switch (report) { | ||
270 | case 0x3: | ||
271 | rx_pwr_all = -40 - ((cck_agc_rpt & 0x1f) << 1); | ||
272 | break; | ||
273 | case 0x2: | ||
274 | rx_pwr_all = -20 - ((cck_agc_rpt & 0x1f) << 1); | ||
275 | break; | ||
276 | case 0x1: | ||
277 | rx_pwr_all = -2 - ((cck_agc_rpt & 0x1f) << 1); | ||
278 | break; | ||
279 | case 0x0: | ||
280 | rx_pwr_all = 14 - ((cck_agc_rpt & 0x1f) << 1); | ||
281 | break; | ||
282 | } | ||
283 | } | ||
284 | |||
285 | pwdb_all = _rtl92s_query_rxpwrpercentage(rx_pwr_all); | ||
286 | |||
287 | /* CCK gain is smaller than OFDM/MCS gain, */ | ||
288 | /* so we add gain diff by experiences, the val is 6 */ | ||
289 | pwdb_all += 6; | ||
290 | if (pwdb_all > 100) | ||
291 | pwdb_all = 100; | ||
292 | /* modify the offset to make the same gain index with OFDM. */ | ||
293 | if (pwdb_all > 34 && pwdb_all <= 42) | ||
294 | pwdb_all -= 2; | ||
295 | else if (pwdb_all > 26 && pwdb_all <= 34) | ||
296 | pwdb_all -= 6; | ||
297 | else if (pwdb_all > 14 && pwdb_all <= 26) | ||
298 | pwdb_all -= 8; | ||
299 | else if (pwdb_all > 4 && pwdb_all <= 14) | ||
300 | pwdb_all -= 4; | ||
301 | |||
302 | pstats->rx_pwdb_all = pwdb_all; | ||
303 | pstats->recvsignalpower = rx_pwr_all; | ||
304 | |||
305 | if (packet_match_bssid) { | ||
306 | u8 sq; | ||
307 | if (pstats->rx_pwdb_all > 40) { | ||
308 | sq = 100; | ||
309 | } else { | ||
310 | sq = cck_buf->sq_rpt; | ||
311 | if (sq > 64) | ||
312 | sq = 0; | ||
313 | else if (sq < 20) | ||
314 | sq = 100; | ||
315 | else | ||
316 | sq = ((64 - sq) * 100) / 44; | ||
317 | } | ||
318 | |||
319 | pstats->signalquality = sq; | ||
320 | pstats->rx_mimo_signalquality[0] = sq; | ||
321 | pstats->rx_mimo_signalquality[1] = -1; | ||
322 | } | ||
323 | } else { | ||
324 | rtlpriv->dm.rfpath_rxenable[0] = | ||
325 | rtlpriv->dm.rfpath_rxenable[1] = true; | ||
326 | for (i = RF90_PATH_A; i < RF90_PATH_MAX; i++) { | ||
327 | if (rtlpriv->dm.rfpath_rxenable[i]) | ||
328 | rf_rx_num++; | ||
329 | |||
330 | rx_pwr[i] = ((p_drvinfo->gain_trsw[i] & | ||
331 | 0x3f) * 2) - 110; | ||
332 | rssi = _rtl92s_query_rxpwrpercentage(rx_pwr[i]); | ||
333 | total_rssi += rssi; | ||
334 | rtlpriv->stats.rx_snr_db[i] = | ||
335 | (long)(p_drvinfo->rxsnr[i] / 2); | ||
336 | |||
337 | if (packet_match_bssid) | ||
338 | pstats->rx_mimo_signalstrength[i] = (u8) rssi; | ||
339 | } | ||
340 | |||
341 | rx_pwr_all = ((p_drvinfo->pwdb_all >> 1) & 0x7f) - 110; | ||
342 | pwdb_all = _rtl92s_query_rxpwrpercentage(rx_pwr_all); | ||
343 | pstats->rx_pwdb_all = pwdb_all; | ||
344 | pstats->rxpower = rx_pwr_all; | ||
345 | pstats->recvsignalpower = rx_pwr_all; | ||
346 | |||
347 | if (GET_RX_STATUS_DESC_RX_HT(pdesc) && | ||
348 | GET_RX_STATUS_DESC_RX_MCS(pdesc) >= DESC92S_RATEMCS8 && | ||
349 | GET_RX_STATUS_DESC_RX_MCS(pdesc) <= DESC92S_RATEMCS15) | ||
350 | max_spatial_stream = 2; | ||
351 | else | ||
352 | max_spatial_stream = 1; | ||
353 | |||
354 | for (i = 0; i < max_spatial_stream; i++) { | ||
355 | evm = _rtl92s_evm_db_to_percentage(p_drvinfo->rxevm[i]); | ||
356 | |||
357 | if (packet_match_bssid) { | ||
358 | if (i == 0) | ||
359 | pstats->signalquality = (u8)(evm & | ||
360 | 0xff); | ||
361 | pstats->rx_mimo_signalquality[i] = | ||
362 | (u8) (evm & 0xff); | ||
363 | } | ||
364 | } | ||
365 | } | ||
366 | |||
367 | if (is_cck_rate) | ||
368 | pstats->signalstrength = (u8)(_rtl92se_signal_scale_mapping(hw, | ||
369 | pwdb_all)); | ||
370 | else if (rf_rx_num != 0) | ||
371 | pstats->signalstrength = (u8) (_rtl92se_signal_scale_mapping(hw, | ||
372 | total_rssi /= rf_rx_num)); | ||
373 | } | ||
374 | |||
375 | static void _rtl92se_process_ui_rssi(struct ieee80211_hw *hw, | ||
376 | struct rtl_stats *pstats) | ||
377 | { | ||
378 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
379 | struct rtl_phy *rtlphy = &(rtlpriv->phy); | ||
380 | u8 rfpath; | ||
381 | u32 last_rssi, tmpval; | ||
382 | |||
383 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
384 | rtlpriv->stats.rssi_calculate_cnt++; | ||
385 | |||
386 | if (rtlpriv->stats.ui_rssi.total_num++ >= | ||
387 | PHY_RSSI_SLID_WIN_MAX) { | ||
388 | rtlpriv->stats.ui_rssi.total_num = | ||
389 | PHY_RSSI_SLID_WIN_MAX; | ||
390 | last_rssi = rtlpriv->stats.ui_rssi.elements[ | ||
391 | rtlpriv->stats.ui_rssi.index]; | ||
392 | rtlpriv->stats.ui_rssi.total_val -= last_rssi; | ||
393 | } | ||
394 | |||
395 | rtlpriv->stats.ui_rssi.total_val += pstats->signalstrength; | ||
396 | rtlpriv->stats.ui_rssi.elements[rtlpriv->stats.ui_rssi.index++] | ||
397 | = pstats->signalstrength; | ||
398 | |||
399 | if (rtlpriv->stats.ui_rssi.index >= PHY_RSSI_SLID_WIN_MAX) | ||
400 | rtlpriv->stats.ui_rssi.index = 0; | ||
401 | |||
402 | tmpval = rtlpriv->stats.ui_rssi.total_val / | ||
403 | rtlpriv->stats.ui_rssi.total_num; | ||
404 | rtlpriv->stats.signal_strength = _rtl92se_translate_todbm(hw, | ||
405 | (u8) tmpval); | ||
406 | pstats->rssi = rtlpriv->stats.signal_strength; | ||
407 | } | ||
408 | |||
409 | if (!pstats->is_cck && pstats->packet_toself) { | ||
410 | for (rfpath = RF90_PATH_A; rfpath < rtlphy->num_total_rfpath; | ||
411 | rfpath++) { | ||
412 | if (rtlpriv->stats.rx_rssi_percentage[rfpath] == 0) { | ||
413 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
414 | pstats->rx_mimo_signalstrength[rfpath]; | ||
415 | |||
416 | } | ||
417 | |||
418 | if (pstats->rx_mimo_signalstrength[rfpath] > | ||
419 | rtlpriv->stats.rx_rssi_percentage[rfpath]) { | ||
420 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
421 | ((rtlpriv->stats.rx_rssi_percentage[rfpath] | ||
422 | * (RX_SMOOTH_FACTOR - 1)) + | ||
423 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
424 | (RX_SMOOTH_FACTOR); | ||
425 | |||
426 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
427 | rtlpriv->stats.rx_rssi_percentage[rfpath] | ||
428 | + 1; | ||
429 | } else { | ||
430 | rtlpriv->stats.rx_rssi_percentage[rfpath] = | ||
431 | ((rtlpriv->stats.rx_rssi_percentage[rfpath] | ||
432 | * (RX_SMOOTH_FACTOR - 1)) + | ||
433 | (pstats->rx_mimo_signalstrength[rfpath])) / | ||
434 | (RX_SMOOTH_FACTOR); | ||
435 | } | ||
436 | |||
437 | } | ||
438 | } | ||
439 | } | ||
440 | |||
441 | static void _rtl92se_update_rxsignalstatistics(struct ieee80211_hw *hw, | ||
442 | struct rtl_stats *pstats) | ||
443 | { | ||
444 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
445 | int weighting = 0; | ||
446 | |||
447 | if (rtlpriv->stats.recv_signal_power == 0) | ||
448 | rtlpriv->stats.recv_signal_power = pstats->recvsignalpower; | ||
449 | |||
450 | if (pstats->recvsignalpower > rtlpriv->stats.recv_signal_power) | ||
451 | weighting = 5; | ||
452 | else if (pstats->recvsignalpower < rtlpriv->stats.recv_signal_power) | ||
453 | weighting = (-5); | ||
454 | |||
455 | rtlpriv->stats.recv_signal_power = (rtlpriv->stats.recv_signal_power * 5 | ||
456 | + pstats->recvsignalpower + | ||
457 | weighting) / 6; | ||
458 | } | ||
459 | |||
460 | static void _rtl92se_process_pwdb(struct ieee80211_hw *hw, | ||
461 | struct rtl_stats *pstats) | ||
462 | { | ||
463 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
464 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
465 | long undec_sm_pwdb = 0; | ||
466 | |||
467 | if (mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
468 | return; | ||
469 | } else { | ||
470 | undec_sm_pwdb = | ||
471 | rtlpriv->dm.undecorated_smoothed_pwdb; | ||
472 | } | ||
473 | |||
474 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
475 | if (undec_sm_pwdb < 0) | ||
476 | undec_sm_pwdb = pstats->rx_pwdb_all; | ||
477 | |||
478 | if (pstats->rx_pwdb_all > (u32) undec_sm_pwdb) { | ||
479 | undec_sm_pwdb = | ||
480 | (((undec_sm_pwdb) * | ||
481 | (RX_SMOOTH_FACTOR - 1)) + | ||
482 | (pstats->rx_pwdb_all)) / (RX_SMOOTH_FACTOR); | ||
483 | |||
484 | undec_sm_pwdb = undec_sm_pwdb + 1; | ||
485 | } else { | ||
486 | undec_sm_pwdb = (((undec_sm_pwdb) * | ||
487 | (RX_SMOOTH_FACTOR - 1)) + (pstats->rx_pwdb_all)) / | ||
488 | (RX_SMOOTH_FACTOR); | ||
489 | } | ||
490 | |||
491 | rtlpriv->dm.undecorated_smoothed_pwdb = undec_sm_pwdb; | ||
492 | _rtl92se_update_rxsignalstatistics(hw, pstats); | ||
493 | } | ||
494 | } | ||
495 | |||
496 | static void rtl_92s_process_streams(struct ieee80211_hw *hw, | ||
497 | struct rtl_stats *pstats) | ||
498 | { | ||
499 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
500 | u32 stream; | ||
501 | |||
502 | for (stream = 0; stream < 2; stream++) { | ||
503 | if (pstats->rx_mimo_signalquality[stream] != -1) { | ||
504 | if (rtlpriv->stats.rx_evm_percentage[stream] == 0) { | ||
505 | rtlpriv->stats.rx_evm_percentage[stream] = | ||
506 | pstats->rx_mimo_signalquality[stream]; | ||
507 | } | ||
508 | |||
509 | rtlpriv->stats.rx_evm_percentage[stream] = | ||
510 | ((rtlpriv->stats.rx_evm_percentage[stream] * | ||
511 | (RX_SMOOTH_FACTOR - 1)) + | ||
512 | (pstats->rx_mimo_signalquality[stream] * | ||
513 | 1)) / (RX_SMOOTH_FACTOR); | ||
514 | } | ||
515 | } | ||
516 | } | ||
517 | |||
518 | static void _rtl92se_process_ui_link_quality(struct ieee80211_hw *hw, | ||
519 | struct rtl_stats *pstats) | ||
520 | { | ||
521 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
522 | u32 last_evm = 0, tmpval; | ||
523 | |||
524 | if (pstats->signalquality != 0) { | ||
525 | if (pstats->packet_toself || pstats->packet_beacon) { | ||
526 | |||
527 | if (rtlpriv->stats.ui_link_quality.total_num++ >= | ||
528 | PHY_LINKQUALITY_SLID_WIN_MAX) { | ||
529 | rtlpriv->stats.ui_link_quality.total_num = | ||
530 | PHY_LINKQUALITY_SLID_WIN_MAX; | ||
531 | last_evm = | ||
532 | rtlpriv->stats.ui_link_quality.elements[ | ||
533 | rtlpriv->stats.ui_link_quality.index]; | ||
534 | rtlpriv->stats.ui_link_quality.total_val -= | ||
535 | last_evm; | ||
536 | } | ||
537 | |||
538 | rtlpriv->stats.ui_link_quality.total_val += | ||
539 | pstats->signalquality; | ||
540 | rtlpriv->stats.ui_link_quality.elements[ | ||
541 | rtlpriv->stats.ui_link_quality.index++] = | ||
542 | pstats->signalquality; | ||
543 | |||
544 | if (rtlpriv->stats.ui_link_quality.index >= | ||
545 | PHY_LINKQUALITY_SLID_WIN_MAX) | ||
546 | rtlpriv->stats.ui_link_quality.index = 0; | ||
547 | |||
548 | tmpval = rtlpriv->stats.ui_link_quality.total_val / | ||
549 | rtlpriv->stats.ui_link_quality.total_num; | ||
550 | rtlpriv->stats.signal_quality = tmpval; | ||
551 | |||
552 | rtlpriv->stats.last_sigstrength_inpercent = tmpval; | ||
553 | |||
554 | rtl_92s_process_streams(hw, pstats); | ||
555 | |||
556 | } | ||
557 | } | ||
558 | } | ||
559 | |||
560 | static void _rtl92se_process_phyinfo(struct ieee80211_hw *hw, | ||
561 | u8 *buffer, | ||
562 | struct rtl_stats *pcurrent_stats) | ||
563 | { | ||
564 | |||
565 | if (!pcurrent_stats->packet_matchbssid && | ||
566 | !pcurrent_stats->packet_beacon) | ||
567 | return; | ||
568 | |||
569 | _rtl92se_process_ui_rssi(hw, pcurrent_stats); | ||
570 | _rtl92se_process_pwdb(hw, pcurrent_stats); | ||
571 | _rtl92se_process_ui_link_quality(hw, pcurrent_stats); | ||
572 | } | ||
573 | |||
574 | static void _rtl92se_translate_rx_signal_stuff(struct ieee80211_hw *hw, | ||
575 | struct sk_buff *skb, struct rtl_stats *pstats, | ||
576 | u8 *pdesc, struct rx_fwinfo *p_drvinfo) | ||
577 | { | ||
578 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
579 | struct rtl_efuse *rtlefuse = rtl_efuse(rtl_priv(hw)); | ||
580 | |||
581 | struct ieee80211_hdr *hdr; | ||
582 | u8 *tmp_buf; | ||
583 | u8 *praddr; | ||
584 | u8 *psaddr; | ||
585 | __le16 fc; | ||
586 | u16 type, cfc; | ||
587 | bool packet_matchbssid, packet_toself, packet_beacon; | ||
588 | |||
589 | tmp_buf = skb->data + pstats->rx_drvinfo_size + pstats->rx_bufshift; | ||
590 | |||
591 | hdr = (struct ieee80211_hdr *)tmp_buf; | ||
592 | fc = hdr->frame_control; | ||
593 | cfc = le16_to_cpu(fc); | ||
594 | type = WLAN_FC_GET_TYPE(fc); | ||
595 | praddr = hdr->addr1; | ||
596 | psaddr = hdr->addr2; | ||
597 | |||
598 | packet_matchbssid = ((IEEE80211_FTYPE_CTL != type) && | ||
599 | (!compare_ether_addr(mac->bssid, (cfc & IEEE80211_FCTL_TODS) ? | ||
600 | hdr->addr1 : (cfc & IEEE80211_FCTL_FROMDS) ? | ||
601 | hdr->addr2 : hdr->addr3)) && (!pstats->hwerror) && | ||
602 | (!pstats->crc) && (!pstats->icv)); | ||
603 | |||
604 | packet_toself = packet_matchbssid && | ||
605 | (!compare_ether_addr(praddr, rtlefuse->dev_addr)); | ||
606 | |||
607 | if (ieee80211_is_beacon(fc)) | ||
608 | packet_beacon = true; | ||
609 | |||
610 | _rtl92se_query_rxphystatus(hw, pstats, pdesc, p_drvinfo, | ||
611 | packet_matchbssid, packet_toself, packet_beacon); | ||
612 | _rtl92se_process_phyinfo(hw, tmp_buf, pstats); | ||
613 | } | ||
614 | |||
615 | bool rtl92se_rx_query_desc(struct ieee80211_hw *hw, struct rtl_stats *stats, | ||
616 | struct ieee80211_rx_status *rx_status, u8 *pdesc, | ||
617 | struct sk_buff *skb) | ||
618 | { | ||
619 | struct rx_fwinfo *p_drvinfo; | ||
620 | u32 phystatus = (u32)GET_RX_STATUS_DESC_PHY_STATUS(pdesc); | ||
621 | |||
622 | stats->length = (u16)GET_RX_STATUS_DESC_PKT_LEN(pdesc); | ||
623 | stats->rx_drvinfo_size = (u8)GET_RX_STATUS_DESC_DRVINFO_SIZE(pdesc) * 8; | ||
624 | stats->rx_bufshift = (u8)(GET_RX_STATUS_DESC_SHIFT(pdesc) & 0x03); | ||
625 | stats->icv = (u16)GET_RX_STATUS_DESC_ICV(pdesc); | ||
626 | stats->crc = (u16)GET_RX_STATUS_DESC_CRC32(pdesc); | ||
627 | stats->hwerror = (u16)(stats->crc | stats->icv); | ||
628 | stats->decrypted = !GET_RX_STATUS_DESC_SWDEC(pdesc); | ||
629 | |||
630 | stats->rate = (u8)GET_RX_STATUS_DESC_RX_MCS(pdesc); | ||
631 | stats->shortpreamble = (u16)GET_RX_STATUS_DESC_SPLCP(pdesc); | ||
632 | stats->isampdu = (bool)(GET_RX_STATUS_DESC_PAGGR(pdesc) == 1); | ||
633 | stats->timestamp_low = GET_RX_STATUS_DESC_TSFL(pdesc); | ||
634 | stats->rx_is40Mhzpacket = (bool)GET_RX_STATUS_DESC_BW(pdesc); | ||
635 | |||
636 | if (stats->hwerror) | ||
637 | return false; | ||
638 | |||
639 | rx_status->freq = hw->conf.channel->center_freq; | ||
640 | rx_status->band = hw->conf.channel->band; | ||
641 | |||
642 | if (GET_RX_STATUS_DESC_CRC32(pdesc)) | ||
643 | rx_status->flag |= RX_FLAG_FAILED_FCS_CRC; | ||
644 | |||
645 | if (!GET_RX_STATUS_DESC_SWDEC(pdesc)) | ||
646 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
647 | |||
648 | if (GET_RX_STATUS_DESC_BW(pdesc)) | ||
649 | rx_status->flag |= RX_FLAG_40MHZ; | ||
650 | |||
651 | if (GET_RX_STATUS_DESC_RX_HT(pdesc)) | ||
652 | rx_status->flag |= RX_FLAG_HT; | ||
653 | |||
654 | rx_status->flag |= RX_FLAG_MACTIME_MPDU; | ||
655 | |||
656 | if (stats->decrypted) | ||
657 | rx_status->flag |= RX_FLAG_DECRYPTED; | ||
658 | |||
659 | rx_status->rate_idx = _rtl92se_rate_mapping((bool) | ||
660 | GET_RX_STATUS_DESC_RX_HT(pdesc), | ||
661 | (u8)GET_RX_STATUS_DESC_RX_MCS(pdesc), | ||
662 | (bool)GET_RX_STATUS_DESC_PAGGR(pdesc)); | ||
663 | |||
664 | |||
665 | rx_status->mactime = GET_RX_STATUS_DESC_TSFL(pdesc); | ||
666 | if (phystatus == true) { | ||
667 | p_drvinfo = (struct rx_fwinfo *)(skb->data + | ||
668 | stats->rx_bufshift); | ||
669 | _rtl92se_translate_rx_signal_stuff(hw, skb, stats, pdesc, | ||
670 | p_drvinfo); | ||
671 | } | ||
672 | |||
673 | /*rx_status->qual = stats->signal; */ | ||
674 | rx_status->signal = stats->rssi + 10; | ||
675 | /*rx_status->noise = -stats->noise; */ | ||
676 | |||
677 | return true; | ||
678 | } | ||
679 | |||
680 | void rtl92se_tx_fill_desc(struct ieee80211_hw *hw, | ||
681 | struct ieee80211_hdr *hdr, u8 *pdesc_tx, | ||
682 | struct ieee80211_tx_info *info, struct sk_buff *skb, | ||
683 | u8 hw_queue, struct rtl_tcb_desc *ptcb_desc) | ||
684 | { | ||
685 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
686 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
687 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
688 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
689 | struct ieee80211_sta *sta = info->control.sta; | ||
690 | u8 *pdesc = (u8 *) pdesc_tx; | ||
691 | u16 seq_number; | ||
692 | __le16 fc = hdr->frame_control; | ||
693 | u8 reserved_macid = 0; | ||
694 | u8 fw_qsel = _rtl92se_map_hwqueue_to_fwqueue(skb, hw_queue); | ||
695 | bool firstseg = (!(hdr->seq_ctrl & cpu_to_le16(IEEE80211_SCTL_FRAG))); | ||
696 | bool lastseg = (!(hdr->frame_control & | ||
697 | cpu_to_le16(IEEE80211_FCTL_MOREFRAGS))); | ||
698 | dma_addr_t mapping = pci_map_single(rtlpci->pdev, skb->data, skb->len, | ||
699 | PCI_DMA_TODEVICE); | ||
700 | u8 bw_40 = 0; | ||
701 | |||
702 | if (mac->opmode == NL80211_IFTYPE_STATION) { | ||
703 | bw_40 = mac->bw_40; | ||
704 | } else if (mac->opmode == NL80211_IFTYPE_AP || | ||
705 | mac->opmode == NL80211_IFTYPE_ADHOC) { | ||
706 | if (sta) | ||
707 | bw_40 = sta->ht_cap.cap & | ||
708 | IEEE80211_HT_CAP_SUP_WIDTH_20_40; | ||
709 | } | ||
710 | |||
711 | seq_number = (le16_to_cpu(hdr->seq_ctrl) & IEEE80211_SCTL_SEQ) >> 4; | ||
712 | |||
713 | rtl_get_tcb_desc(hw, info, sta, skb, ptcb_desc); | ||
714 | |||
715 | CLEAR_PCI_TX_DESC_CONTENT(pdesc, TX_DESC_SIZE_RTL8192S); | ||
716 | |||
717 | if (firstseg) { | ||
718 | if (rtlpriv->dm.useramask) { | ||
719 | /* set txdesc macId */ | ||
720 | if (ptcb_desc->mac_id < 32) { | ||
721 | SET_TX_DESC_MACID(pdesc, ptcb_desc->mac_id); | ||
722 | reserved_macid |= ptcb_desc->mac_id; | ||
723 | } | ||
724 | } | ||
725 | SET_TX_DESC_RSVD_MACID(pdesc, reserved_macid); | ||
726 | |||
727 | SET_TX_DESC_TXHT(pdesc, ((ptcb_desc->hw_rate >= | ||
728 | DESC92S_RATEMCS0) ? 1 : 0)); | ||
729 | |||
730 | if (rtlhal->version == VERSION_8192S_ACUT) { | ||
731 | if (ptcb_desc->hw_rate == DESC92S_RATE1M || | ||
732 | ptcb_desc->hw_rate == DESC92S_RATE2M || | ||
733 | ptcb_desc->hw_rate == DESC92S_RATE5_5M || | ||
734 | ptcb_desc->hw_rate == DESC92S_RATE11M) { | ||
735 | ptcb_desc->hw_rate = DESC92S_RATE12M; | ||
736 | } | ||
737 | } | ||
738 | |||
739 | SET_TX_DESC_TX_RATE(pdesc, ptcb_desc->hw_rate); | ||
740 | |||
741 | if (ptcb_desc->use_shortgi || ptcb_desc->use_shortpreamble) | ||
742 | SET_TX_DESC_TX_SHORT(pdesc, 0); | ||
743 | |||
744 | /* Aggregation related */ | ||
745 | if (info->flags & IEEE80211_TX_CTL_AMPDU) | ||
746 | SET_TX_DESC_AGG_ENABLE(pdesc, 1); | ||
747 | |||
748 | /* For AMPDU, we must insert SSN into TX_DESC */ | ||
749 | SET_TX_DESC_SEQ(pdesc, seq_number); | ||
750 | |||
751 | /* Protection mode related */ | ||
752 | /* For 92S, if RTS/CTS are set, HW will execute RTS. */ | ||
753 | /* We choose only one protection mode to execute */ | ||
754 | SET_TX_DESC_RTS_ENABLE(pdesc, ((ptcb_desc->rts_enable && | ||
755 | !ptcb_desc->cts_enable) ? 1 : 0)); | ||
756 | SET_TX_DESC_CTS_ENABLE(pdesc, ((ptcb_desc->cts_enable) ? | ||
757 | 1 : 0)); | ||
758 | SET_TX_DESC_RTS_STBC(pdesc, ((ptcb_desc->rts_stbc) ? 1 : 0)); | ||
759 | |||
760 | SET_TX_DESC_RTS_RATE(pdesc, ptcb_desc->rts_rate); | ||
761 | SET_TX_DESC_RTS_BANDWIDTH(pdesc, 0); | ||
762 | SET_TX_DESC_RTS_SUB_CARRIER(pdesc, ptcb_desc->rts_sc); | ||
763 | SET_TX_DESC_RTS_SHORT(pdesc, ((ptcb_desc->rts_rate <= | ||
764 | DESC92S_RATE54M) ? | ||
765 | (ptcb_desc->rts_use_shortpreamble ? 1 : 0) | ||
766 | : (ptcb_desc->rts_use_shortgi ? 1 : 0))); | ||
767 | |||
768 | |||
769 | /* Set Bandwidth and sub-channel settings. */ | ||
770 | if (bw_40) { | ||
771 | if (ptcb_desc->packet_bw) { | ||
772 | SET_TX_DESC_TX_BANDWIDTH(pdesc, 1); | ||
773 | /* use duplicated mode */ | ||
774 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, 0); | ||
775 | } else { | ||
776 | SET_TX_DESC_TX_BANDWIDTH(pdesc, 0); | ||
777 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, | ||
778 | mac->cur_40_prime_sc); | ||
779 | } | ||
780 | } else { | ||
781 | SET_TX_DESC_TX_BANDWIDTH(pdesc, 0); | ||
782 | SET_TX_DESC_TX_SUB_CARRIER(pdesc, 0); | ||
783 | } | ||
784 | |||
785 | /* 3 Fill necessary field in First Descriptor */ | ||
786 | /*DWORD 0*/ | ||
787 | SET_TX_DESC_LINIP(pdesc, 0); | ||
788 | SET_TX_DESC_OFFSET(pdesc, 32); | ||
789 | SET_TX_DESC_PKT_SIZE(pdesc, (u16) skb->len); | ||
790 | |||
791 | /*DWORD 1*/ | ||
792 | SET_TX_DESC_RA_BRSR_ID(pdesc, ptcb_desc->ratr_index); | ||
793 | |||
794 | /* Fill security related */ | ||
795 | if (info->control.hw_key) { | ||
796 | struct ieee80211_key_conf *keyconf; | ||
797 | |||
798 | keyconf = info->control.hw_key; | ||
799 | switch (keyconf->cipher) { | ||
800 | case WLAN_CIPHER_SUITE_WEP40: | ||
801 | case WLAN_CIPHER_SUITE_WEP104: | ||
802 | SET_TX_DESC_SEC_TYPE(pdesc, 0x1); | ||
803 | break; | ||
804 | case WLAN_CIPHER_SUITE_TKIP: | ||
805 | SET_TX_DESC_SEC_TYPE(pdesc, 0x2); | ||
806 | break; | ||
807 | case WLAN_CIPHER_SUITE_CCMP: | ||
808 | SET_TX_DESC_SEC_TYPE(pdesc, 0x3); | ||
809 | break; | ||
810 | default: | ||
811 | SET_TX_DESC_SEC_TYPE(pdesc, 0x0); | ||
812 | break; | ||
813 | |||
814 | } | ||
815 | } | ||
816 | |||
817 | /* Set Packet ID */ | ||
818 | SET_TX_DESC_PACKET_ID(pdesc, 0); | ||
819 | |||
820 | /* We will assign magement queue to BK. */ | ||
821 | SET_TX_DESC_QUEUE_SEL(pdesc, fw_qsel); | ||
822 | |||
823 | /* Alwasy enable all rate fallback range */ | ||
824 | SET_TX_DESC_DATA_RATE_FB_LIMIT(pdesc, 0x1F); | ||
825 | |||
826 | /* Fix: I don't kown why hw use 6.5M to tx when set it */ | ||
827 | SET_TX_DESC_USER_RATE(pdesc, | ||
828 | ptcb_desc->use_driver_rate ? 1 : 0); | ||
829 | |||
830 | /* Set NON_QOS bit. */ | ||
831 | if (!ieee80211_is_data_qos(fc)) | ||
832 | SET_TX_DESC_NON_QOS(pdesc, 1); | ||
833 | |||
834 | } | ||
835 | |||
836 | /* Fill fields that are required to be initialized | ||
837 | * in all of the descriptors */ | ||
838 | /*DWORD 0 */ | ||
839 | SET_TX_DESC_FIRST_SEG(pdesc, (firstseg ? 1 : 0)); | ||
840 | SET_TX_DESC_LAST_SEG(pdesc, (lastseg ? 1 : 0)); | ||
841 | |||
842 | /* DWORD 7 */ | ||
843 | SET_TX_DESC_TX_BUFFER_SIZE(pdesc, (u16) skb->len); | ||
844 | |||
845 | /* DOWRD 8 */ | ||
846 | SET_TX_DESC_TX_BUFFER_ADDRESS(pdesc, cpu_to_le32(mapping)); | ||
847 | |||
848 | RT_TRACE(rtlpriv, COMP_SEND, DBG_TRACE, ("\n")); | ||
849 | } | ||
850 | |||
851 | void rtl92se_tx_fill_cmddesc(struct ieee80211_hw *hw, u8 *pdesc, | ||
852 | bool firstseg, bool lastseg, struct sk_buff *skb) | ||
853 | { | ||
854 | struct rtl_pci *rtlpci = rtl_pcidev(rtl_pcipriv(hw)); | ||
855 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
856 | struct rtl_tcb_desc *tcb_desc = (struct rtl_tcb_desc *)(skb->cb); | ||
857 | |||
858 | dma_addr_t mapping = pci_map_single(rtlpci->pdev, skb->data, skb->len, | ||
859 | PCI_DMA_TODEVICE); | ||
860 | |||
861 | /* Clear all status */ | ||
862 | CLEAR_PCI_TX_DESC_CONTENT(pdesc, TX_CMDDESC_SIZE_RTL8192S); | ||
863 | |||
864 | /* This bit indicate this packet is used for FW download. */ | ||
865 | if (tcb_desc->cmd_or_init == DESC_PACKET_TYPE_INIT) { | ||
866 | /* For firmware downlaod we only need to set LINIP */ | ||
867 | SET_TX_DESC_LINIP(pdesc, tcb_desc->last_inipkt); | ||
868 | |||
869 | /* 92SE must set as 1 for firmware download HW DMA error */ | ||
870 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
871 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
872 | |||
873 | /* 92SE need not to set TX packet size when firmware download */ | ||
874 | SET_TX_DESC_PKT_SIZE(pdesc, (u16)(skb->len)); | ||
875 | SET_TX_DESC_TX_BUFFER_SIZE(pdesc, (u16)(skb->len)); | ||
876 | SET_TX_DESC_TX_BUFFER_ADDRESS(pdesc, cpu_to_le32(mapping)); | ||
877 | |||
878 | SET_TX_DESC_OWN(pdesc, 1); | ||
879 | } else { /* H2C Command Desc format (Host TXCMD) */ | ||
880 | /* 92SE must set as 1 for firmware download HW DMA error */ | ||
881 | SET_TX_DESC_FIRST_SEG(pdesc, 1); | ||
882 | SET_TX_DESC_LAST_SEG(pdesc, 1); | ||
883 | |||
884 | SET_TX_DESC_OFFSET(pdesc, 0x20); | ||
885 | |||
886 | /* Buffer size + command header */ | ||
887 | SET_TX_DESC_PKT_SIZE(pdesc, (u16)(skb->len)); | ||
888 | /* Fixed queue of H2C command */ | ||
889 | SET_TX_DESC_QUEUE_SEL(pdesc, 0x13); | ||
890 | |||
891 | SET_BITS_TO_LE_4BYTE(skb->data, 24, 7, rtlhal->h2c_txcmd_seq); | ||
892 | |||
893 | SET_TX_DESC_TX_BUFFER_SIZE(pdesc, (u16)(skb->len)); | ||
894 | SET_TX_DESC_TX_BUFFER_ADDRESS(pdesc, cpu_to_le32(mapping)); | ||
895 | |||
896 | SET_TX_DESC_OWN(pdesc, 1); | ||
897 | |||
898 | } | ||
899 | } | ||
900 | |||
901 | void rtl92se_set_desc(u8 *pdesc, bool istx, u8 desc_name, u8 *val) | ||
902 | { | ||
903 | if (istx == true) { | ||
904 | switch (desc_name) { | ||
905 | case HW_DESC_OWN: | ||
906 | SET_TX_DESC_OWN(pdesc, 1); | ||
907 | break; | ||
908 | case HW_DESC_TX_NEXTDESC_ADDR: | ||
909 | SET_TX_DESC_NEXT_DESC_ADDRESS(pdesc, *(u32 *) val); | ||
910 | break; | ||
911 | default: | ||
912 | RT_ASSERT(false, ("ERR txdesc :%d not process\n", | ||
913 | desc_name)); | ||
914 | break; | ||
915 | } | ||
916 | } else { | ||
917 | switch (desc_name) { | ||
918 | case HW_DESC_RXOWN: | ||
919 | SET_RX_STATUS_DESC_OWN(pdesc, 1); | ||
920 | break; | ||
921 | case HW_DESC_RXBUFF_ADDR: | ||
922 | SET_RX_STATUS__DESC_BUFF_ADDR(pdesc, *(u32 *) val); | ||
923 | break; | ||
924 | case HW_DESC_RXPKT_LEN: | ||
925 | SET_RX_STATUS_DESC_PKT_LEN(pdesc, *(u32 *) val); | ||
926 | break; | ||
927 | case HW_DESC_RXERO: | ||
928 | SET_RX_STATUS_DESC_EOR(pdesc, 1); | ||
929 | break; | ||
930 | default: | ||
931 | RT_ASSERT(false, ("ERR rxdesc :%d not process\n", | ||
932 | desc_name)); | ||
933 | break; | ||
934 | } | ||
935 | } | ||
936 | } | ||
937 | |||
938 | u32 rtl92se_get_desc(u8 *desc, bool istx, u8 desc_name) | ||
939 | { | ||
940 | u32 ret = 0; | ||
941 | |||
942 | if (istx == true) { | ||
943 | switch (desc_name) { | ||
944 | case HW_DESC_OWN: | ||
945 | ret = GET_TX_DESC_OWN(desc); | ||
946 | break; | ||
947 | case HW_DESC_TXBUFF_ADDR: | ||
948 | ret = GET_TX_DESC_TX_BUFFER_ADDRESS(desc); | ||
949 | break; | ||
950 | default: | ||
951 | RT_ASSERT(false, ("ERR txdesc :%d not process\n", | ||
952 | desc_name)); | ||
953 | break; | ||
954 | } | ||
955 | } else { | ||
956 | switch (desc_name) { | ||
957 | case HW_DESC_OWN: | ||
958 | ret = GET_RX_STATUS_DESC_OWN(desc); | ||
959 | break; | ||
960 | case HW_DESC_RXPKT_LEN: | ||
961 | ret = GET_RX_STATUS_DESC_PKT_LEN(desc); | ||
962 | break; | ||
963 | default: | ||
964 | RT_ASSERT(false, ("ERR rxdesc :%d not process\n", | ||
965 | desc_name)); | ||
966 | break; | ||
967 | } | ||
968 | } | ||
969 | return ret; | ||
970 | } | ||
971 | |||
972 | void rtl92se_tx_polling(struct ieee80211_hw *hw, u8 hw_queue) | ||
973 | { | ||
974 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
975 | rtl_write_word(rtlpriv, TP_POLL, BIT(0) << (hw_queue)); | ||
976 | } | ||
diff --git a/drivers/net/wireless/rtlwifi/rtl8192se/trx.h b/drivers/net/wireless/rtlwifi/rtl8192se/trx.h new file mode 100644 index 000000000000..05862c51b861 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/rtl8192se/trx.h | |||
@@ -0,0 +1,45 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | #ifndef __REALTEK_PCI92SE_TRX_H__ | ||
30 | #define __REALTEK_PCI92SE_TRX_H__ | ||
31 | |||
32 | void rtl92se_tx_fill_desc(struct ieee80211_hw *hw, struct ieee80211_hdr *hdr, | ||
33 | u8 *pdesc, struct ieee80211_tx_info *info, | ||
34 | struct sk_buff *skb, u8 hw_queue, | ||
35 | struct rtl_tcb_desc *ptcb_desc); | ||
36 | void rtl92se_tx_fill_cmddesc(struct ieee80211_hw *hw, u8 *pdesc, bool firstseg, | ||
37 | bool lastseg, struct sk_buff *skb); | ||
38 | bool rtl92se_rx_query_desc(struct ieee80211_hw *hw, struct rtl_stats *stats, | ||
39 | struct ieee80211_rx_status *rx_status, u8 *pdesc, | ||
40 | struct sk_buff *skb); | ||
41 | void rtl92se_set_desc(u8 *pdesc, bool istx, u8 desc_name, u8 *val); | ||
42 | u32 rtl92se_get_desc(u8 *pdesc, bool istx, u8 desc_name); | ||
43 | void rtl92se_tx_polling(struct ieee80211_hw *hw, u8 hw_queue); | ||
44 | |||
45 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/usb.c b/drivers/net/wireless/rtlwifi/usb.c new file mode 100644 index 000000000000..a9367eba1ea7 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/usb.c | |||
@@ -0,0 +1,1046 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2011 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | *****************************************************************************/ | ||
27 | #include <linux/usb.h> | ||
28 | #include "core.h" | ||
29 | #include "wifi.h" | ||
30 | #include "usb.h" | ||
31 | #include "base.h" | ||
32 | #include "ps.h" | ||
33 | |||
34 | #define REALTEK_USB_VENQT_READ 0xC0 | ||
35 | #define REALTEK_USB_VENQT_WRITE 0x40 | ||
36 | #define REALTEK_USB_VENQT_CMD_REQ 0x05 | ||
37 | #define REALTEK_USB_VENQT_CMD_IDX 0x00 | ||
38 | |||
39 | #define REALTEK_USB_VENQT_MAX_BUF_SIZE 254 | ||
40 | |||
41 | static void usbctrl_async_callback(struct urb *urb) | ||
42 | { | ||
43 | if (urb) | ||
44 | kfree(urb->context); | ||
45 | } | ||
46 | |||
47 | static int _usbctrl_vendorreq_async_write(struct usb_device *udev, u8 request, | ||
48 | u16 value, u16 index, void *pdata, | ||
49 | u16 len) | ||
50 | { | ||
51 | int rc; | ||
52 | unsigned int pipe; | ||
53 | u8 reqtype; | ||
54 | struct usb_ctrlrequest *dr; | ||
55 | struct urb *urb; | ||
56 | struct rtl819x_async_write_data { | ||
57 | u8 data[REALTEK_USB_VENQT_MAX_BUF_SIZE]; | ||
58 | struct usb_ctrlrequest dr; | ||
59 | } *buf; | ||
60 | |||
61 | pipe = usb_sndctrlpipe(udev, 0); /* write_out */ | ||
62 | reqtype = REALTEK_USB_VENQT_WRITE; | ||
63 | |||
64 | buf = kmalloc(sizeof(*buf), GFP_ATOMIC); | ||
65 | if (!buf) | ||
66 | return -ENOMEM; | ||
67 | |||
68 | urb = usb_alloc_urb(0, GFP_ATOMIC); | ||
69 | if (!urb) { | ||
70 | kfree(buf); | ||
71 | return -ENOMEM; | ||
72 | } | ||
73 | |||
74 | dr = &buf->dr; | ||
75 | |||
76 | dr->bRequestType = reqtype; | ||
77 | dr->bRequest = request; | ||
78 | dr->wValue = cpu_to_le16(value); | ||
79 | dr->wIndex = cpu_to_le16(index); | ||
80 | dr->wLength = cpu_to_le16(len); | ||
81 | memcpy(buf, pdata, len); | ||
82 | usb_fill_control_urb(urb, udev, pipe, | ||
83 | (unsigned char *)dr, buf, len, | ||
84 | usbctrl_async_callback, buf); | ||
85 | rc = usb_submit_urb(urb, GFP_ATOMIC); | ||
86 | if (rc < 0) | ||
87 | kfree(buf); | ||
88 | usb_free_urb(urb); | ||
89 | return rc; | ||
90 | } | ||
91 | |||
92 | static int _usbctrl_vendorreq_sync_read(struct usb_device *udev, u8 request, | ||
93 | u16 value, u16 index, void *pdata, | ||
94 | u16 len) | ||
95 | { | ||
96 | unsigned int pipe; | ||
97 | int status; | ||
98 | u8 reqtype; | ||
99 | |||
100 | pipe = usb_rcvctrlpipe(udev, 0); /* read_in */ | ||
101 | reqtype = REALTEK_USB_VENQT_READ; | ||
102 | |||
103 | status = usb_control_msg(udev, pipe, request, reqtype, value, index, | ||
104 | pdata, len, 0); /* max. timeout */ | ||
105 | |||
106 | if (status < 0) | ||
107 | printk(KERN_ERR "reg 0x%x, usbctrl_vendorreq TimeOut! " | ||
108 | "status:0x%x value=0x%x\n", value, status, | ||
109 | *(u32 *)pdata); | ||
110 | return status; | ||
111 | } | ||
112 | |||
113 | static u32 _usb_read_sync(struct usb_device *udev, u32 addr, u16 len) | ||
114 | { | ||
115 | u8 request; | ||
116 | u16 wvalue; | ||
117 | u16 index; | ||
118 | u32 *data; | ||
119 | u32 ret; | ||
120 | |||
121 | data = kmalloc(sizeof(u32), GFP_KERNEL); | ||
122 | if (!data) | ||
123 | return -ENOMEM; | ||
124 | request = REALTEK_USB_VENQT_CMD_REQ; | ||
125 | index = REALTEK_USB_VENQT_CMD_IDX; /* n/a */ | ||
126 | |||
127 | wvalue = (u16)addr; | ||
128 | _usbctrl_vendorreq_sync_read(udev, request, wvalue, index, data, len); | ||
129 | ret = *data; | ||
130 | kfree(data); | ||
131 | return ret; | ||
132 | } | ||
133 | |||
134 | static u8 _usb_read8_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
135 | { | ||
136 | struct device *dev = rtlpriv->io.dev; | ||
137 | |||
138 | return (u8)_usb_read_sync(to_usb_device(dev), addr, 1); | ||
139 | } | ||
140 | |||
141 | static u16 _usb_read16_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
142 | { | ||
143 | struct device *dev = rtlpriv->io.dev; | ||
144 | |||
145 | return (u16)_usb_read_sync(to_usb_device(dev), addr, 2); | ||
146 | } | ||
147 | |||
148 | static u32 _usb_read32_sync(struct rtl_priv *rtlpriv, u32 addr) | ||
149 | { | ||
150 | struct device *dev = rtlpriv->io.dev; | ||
151 | |||
152 | return _usb_read_sync(to_usb_device(dev), addr, 4); | ||
153 | } | ||
154 | |||
155 | static void _usb_write_async(struct usb_device *udev, u32 addr, u32 val, | ||
156 | u16 len) | ||
157 | { | ||
158 | u8 request; | ||
159 | u16 wvalue; | ||
160 | u16 index; | ||
161 | u32 data; | ||
162 | |||
163 | request = REALTEK_USB_VENQT_CMD_REQ; | ||
164 | index = REALTEK_USB_VENQT_CMD_IDX; /* n/a */ | ||
165 | wvalue = (u16)(addr&0x0000ffff); | ||
166 | data = val; | ||
167 | _usbctrl_vendorreq_async_write(udev, request, wvalue, index, &data, | ||
168 | len); | ||
169 | } | ||
170 | |||
171 | static void _usb_write8_async(struct rtl_priv *rtlpriv, u32 addr, u8 val) | ||
172 | { | ||
173 | struct device *dev = rtlpriv->io.dev; | ||
174 | |||
175 | _usb_write_async(to_usb_device(dev), addr, val, 1); | ||
176 | } | ||
177 | |||
178 | static void _usb_write16_async(struct rtl_priv *rtlpriv, u32 addr, u16 val) | ||
179 | { | ||
180 | struct device *dev = rtlpriv->io.dev; | ||
181 | |||
182 | _usb_write_async(to_usb_device(dev), addr, val, 2); | ||
183 | } | ||
184 | |||
185 | static void _usb_write32_async(struct rtl_priv *rtlpriv, u32 addr, u32 val) | ||
186 | { | ||
187 | struct device *dev = rtlpriv->io.dev; | ||
188 | |||
189 | _usb_write_async(to_usb_device(dev), addr, val, 4); | ||
190 | } | ||
191 | |||
192 | static int _usb_nbytes_read_write(struct usb_device *udev, bool read, u32 addr, | ||
193 | u16 len, u8 *pdata) | ||
194 | { | ||
195 | int status; | ||
196 | u8 request; | ||
197 | u16 wvalue; | ||
198 | u16 index; | ||
199 | |||
200 | request = REALTEK_USB_VENQT_CMD_REQ; | ||
201 | index = REALTEK_USB_VENQT_CMD_IDX; /* n/a */ | ||
202 | wvalue = (u16)addr; | ||
203 | if (read) | ||
204 | status = _usbctrl_vendorreq_sync_read(udev, request, wvalue, | ||
205 | index, pdata, len); | ||
206 | else | ||
207 | status = _usbctrl_vendorreq_async_write(udev, request, wvalue, | ||
208 | index, pdata, len); | ||
209 | return status; | ||
210 | } | ||
211 | |||
212 | static int _usb_readN_sync(struct rtl_priv *rtlpriv, u32 addr, u16 len, | ||
213 | u8 *pdata) | ||
214 | { | ||
215 | struct device *dev = rtlpriv->io.dev; | ||
216 | |||
217 | return _usb_nbytes_read_write(to_usb_device(dev), true, addr, len, | ||
218 | pdata); | ||
219 | } | ||
220 | |||
221 | static int _usb_writeN_async(struct rtl_priv *rtlpriv, u32 addr, u16 len, | ||
222 | u8 *pdata) | ||
223 | { | ||
224 | struct device *dev = rtlpriv->io.dev; | ||
225 | |||
226 | return _usb_nbytes_read_write(to_usb_device(dev), false, addr, len, | ||
227 | pdata); | ||
228 | } | ||
229 | |||
230 | static void _rtl_usb_io_handler_init(struct device *dev, | ||
231 | struct ieee80211_hw *hw) | ||
232 | { | ||
233 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
234 | |||
235 | rtlpriv->io.dev = dev; | ||
236 | mutex_init(&rtlpriv->io.bb_mutex); | ||
237 | rtlpriv->io.write8_async = _usb_write8_async; | ||
238 | rtlpriv->io.write16_async = _usb_write16_async; | ||
239 | rtlpriv->io.write32_async = _usb_write32_async; | ||
240 | rtlpriv->io.writeN_async = _usb_writeN_async; | ||
241 | rtlpriv->io.read8_sync = _usb_read8_sync; | ||
242 | rtlpriv->io.read16_sync = _usb_read16_sync; | ||
243 | rtlpriv->io.read32_sync = _usb_read32_sync; | ||
244 | rtlpriv->io.readN_sync = _usb_readN_sync; | ||
245 | } | ||
246 | |||
247 | static void _rtl_usb_io_handler_release(struct ieee80211_hw *hw) | ||
248 | { | ||
249 | struct rtl_priv __maybe_unused *rtlpriv = rtl_priv(hw); | ||
250 | |||
251 | mutex_destroy(&rtlpriv->io.bb_mutex); | ||
252 | } | ||
253 | |||
254 | /** | ||
255 | * | ||
256 | * Default aggregation handler. Do nothing and just return the oldest skb. | ||
257 | */ | ||
258 | static struct sk_buff *_none_usb_tx_aggregate_hdl(struct ieee80211_hw *hw, | ||
259 | struct sk_buff_head *list) | ||
260 | { | ||
261 | return skb_dequeue(list); | ||
262 | } | ||
263 | |||
264 | #define IS_HIGH_SPEED_USB(udev) \ | ||
265 | ((USB_SPEED_HIGH == (udev)->speed) ? true : false) | ||
266 | |||
267 | static int _rtl_usb_init_tx(struct ieee80211_hw *hw) | ||
268 | { | ||
269 | u32 i; | ||
270 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
271 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
272 | |||
273 | rtlusb->max_bulk_out_size = IS_HIGH_SPEED_USB(rtlusb->udev) | ||
274 | ? USB_HIGH_SPEED_BULK_SIZE | ||
275 | : USB_FULL_SPEED_BULK_SIZE; | ||
276 | |||
277 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, ("USB Max Bulk-out Size=%d\n", | ||
278 | rtlusb->max_bulk_out_size)); | ||
279 | |||
280 | for (i = 0; i < __RTL_TXQ_NUM; i++) { | ||
281 | u32 ep_num = rtlusb->ep_map.ep_mapping[i]; | ||
282 | if (!ep_num) { | ||
283 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
284 | ("Invalid endpoint map setting!\n")); | ||
285 | return -EINVAL; | ||
286 | } | ||
287 | } | ||
288 | |||
289 | rtlusb->usb_tx_post_hdl = | ||
290 | rtlpriv->cfg->usb_interface_cfg->usb_tx_post_hdl; | ||
291 | rtlusb->usb_tx_cleanup = | ||
292 | rtlpriv->cfg->usb_interface_cfg->usb_tx_cleanup; | ||
293 | rtlusb->usb_tx_aggregate_hdl = | ||
294 | (rtlpriv->cfg->usb_interface_cfg->usb_tx_aggregate_hdl) | ||
295 | ? rtlpriv->cfg->usb_interface_cfg->usb_tx_aggregate_hdl | ||
296 | : &_none_usb_tx_aggregate_hdl; | ||
297 | |||
298 | init_usb_anchor(&rtlusb->tx_submitted); | ||
299 | for (i = 0; i < RTL_USB_MAX_EP_NUM; i++) { | ||
300 | skb_queue_head_init(&rtlusb->tx_skb_queue[i]); | ||
301 | init_usb_anchor(&rtlusb->tx_pending[i]); | ||
302 | } | ||
303 | return 0; | ||
304 | } | ||
305 | |||
306 | static int _rtl_usb_init_rx(struct ieee80211_hw *hw) | ||
307 | { | ||
308 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
309 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
310 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
311 | |||
312 | rtlusb->rx_max_size = rtlpriv->cfg->usb_interface_cfg->rx_max_size; | ||
313 | rtlusb->rx_urb_num = rtlpriv->cfg->usb_interface_cfg->rx_urb_num; | ||
314 | rtlusb->in_ep = rtlpriv->cfg->usb_interface_cfg->in_ep_num; | ||
315 | rtlusb->usb_rx_hdl = rtlpriv->cfg->usb_interface_cfg->usb_rx_hdl; | ||
316 | rtlusb->usb_rx_segregate_hdl = | ||
317 | rtlpriv->cfg->usb_interface_cfg->usb_rx_segregate_hdl; | ||
318 | |||
319 | printk(KERN_INFO "rtl8192cu: rx_max_size %d, rx_urb_num %d, in_ep %d\n", | ||
320 | rtlusb->rx_max_size, rtlusb->rx_urb_num, rtlusb->in_ep); | ||
321 | init_usb_anchor(&rtlusb->rx_submitted); | ||
322 | return 0; | ||
323 | } | ||
324 | |||
325 | static int _rtl_usb_init(struct ieee80211_hw *hw) | ||
326 | { | ||
327 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
328 | struct rtl_usb_priv *usb_priv = rtl_usbpriv(hw); | ||
329 | struct rtl_usb *rtlusb = rtl_usbdev(usb_priv); | ||
330 | int err; | ||
331 | u8 epidx; | ||
332 | struct usb_interface *usb_intf = rtlusb->intf; | ||
333 | u8 epnums = usb_intf->cur_altsetting->desc.bNumEndpoints; | ||
334 | |||
335 | rtlusb->out_ep_nums = rtlusb->in_ep_nums = 0; | ||
336 | for (epidx = 0; epidx < epnums; epidx++) { | ||
337 | struct usb_endpoint_descriptor *pep_desc; | ||
338 | pep_desc = &usb_intf->cur_altsetting->endpoint[epidx].desc; | ||
339 | |||
340 | if (usb_endpoint_dir_in(pep_desc)) | ||
341 | rtlusb->in_ep_nums++; | ||
342 | else if (usb_endpoint_dir_out(pep_desc)) | ||
343 | rtlusb->out_ep_nums++; | ||
344 | |||
345 | RT_TRACE(rtlpriv, COMP_INIT, DBG_DMESG, | ||
346 | ("USB EP(0x%02x), MaxPacketSize=%d ,Interval=%d.\n", | ||
347 | pep_desc->bEndpointAddress, pep_desc->wMaxPacketSize, | ||
348 | pep_desc->bInterval)); | ||
349 | } | ||
350 | if (rtlusb->in_ep_nums < rtlpriv->cfg->usb_interface_cfg->in_ep_num) | ||
351 | return -EINVAL ; | ||
352 | |||
353 | /* usb endpoint mapping */ | ||
354 | err = rtlpriv->cfg->usb_interface_cfg->usb_endpoint_mapping(hw); | ||
355 | rtlusb->usb_mq_to_hwq = rtlpriv->cfg->usb_interface_cfg->usb_mq_to_hwq; | ||
356 | _rtl_usb_init_tx(hw); | ||
357 | _rtl_usb_init_rx(hw); | ||
358 | return err; | ||
359 | } | ||
360 | |||
361 | static int _rtl_usb_init_sw(struct ieee80211_hw *hw) | ||
362 | { | ||
363 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
364 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
365 | struct rtl_ps_ctl *ppsc = rtl_psc(rtl_priv(hw)); | ||
366 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
367 | |||
368 | rtlhal->hw = hw; | ||
369 | ppsc->inactiveps = false; | ||
370 | ppsc->leisure_ps = false; | ||
371 | ppsc->fwctrl_lps = false; | ||
372 | ppsc->reg_fwctrl_lps = 3; | ||
373 | ppsc->reg_max_lps_awakeintvl = 5; | ||
374 | ppsc->fwctrl_psmode = FW_PS_DTIM_MODE; | ||
375 | |||
376 | /* IBSS */ | ||
377 | mac->beacon_interval = 100; | ||
378 | |||
379 | /* AMPDU */ | ||
380 | mac->min_space_cfg = 0; | ||
381 | mac->max_mss_density = 0; | ||
382 | |||
383 | /* set sane AMPDU defaults */ | ||
384 | mac->current_ampdu_density = 7; | ||
385 | mac->current_ampdu_factor = 3; | ||
386 | |||
387 | /* QOS */ | ||
388 | rtlusb->acm_method = eAcmWay2_SW; | ||
389 | |||
390 | /* IRQ */ | ||
391 | /* HIMR - turn all on */ | ||
392 | rtlusb->irq_mask[0] = 0xFFFFFFFF; | ||
393 | /* HIMR_EX - turn all on */ | ||
394 | rtlusb->irq_mask[1] = 0xFFFFFFFF; | ||
395 | rtlusb->disableHWSM = true; | ||
396 | return 0; | ||
397 | } | ||
398 | |||
399 | #define __RADIO_TAP_SIZE_RSV 32 | ||
400 | |||
401 | static void _rtl_rx_completed(struct urb *urb); | ||
402 | |||
403 | static struct sk_buff *_rtl_prep_rx_urb(struct ieee80211_hw *hw, | ||
404 | struct rtl_usb *rtlusb, | ||
405 | struct urb *urb, | ||
406 | gfp_t gfp_mask) | ||
407 | { | ||
408 | struct sk_buff *skb; | ||
409 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
410 | |||
411 | skb = __dev_alloc_skb((rtlusb->rx_max_size + __RADIO_TAP_SIZE_RSV), | ||
412 | gfp_mask); | ||
413 | if (!skb) { | ||
414 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
415 | ("Failed to __dev_alloc_skb!!\n")) | ||
416 | return ERR_PTR(-ENOMEM); | ||
417 | } | ||
418 | |||
419 | /* reserve some space for mac80211's radiotap */ | ||
420 | skb_reserve(skb, __RADIO_TAP_SIZE_RSV); | ||
421 | usb_fill_bulk_urb(urb, rtlusb->udev, | ||
422 | usb_rcvbulkpipe(rtlusb->udev, rtlusb->in_ep), | ||
423 | skb->data, min(skb_tailroom(skb), | ||
424 | (int)rtlusb->rx_max_size), | ||
425 | _rtl_rx_completed, skb); | ||
426 | |||
427 | _rtl_install_trx_info(rtlusb, skb, rtlusb->in_ep); | ||
428 | return skb; | ||
429 | } | ||
430 | |||
431 | #undef __RADIO_TAP_SIZE_RSV | ||
432 | |||
433 | static void _rtl_usb_rx_process_agg(struct ieee80211_hw *hw, | ||
434 | struct sk_buff *skb) | ||
435 | { | ||
436 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
437 | u8 *rxdesc = skb->data; | ||
438 | struct ieee80211_hdr *hdr; | ||
439 | bool unicast = false; | ||
440 | __le16 fc; | ||
441 | struct ieee80211_rx_status rx_status = {0}; | ||
442 | struct rtl_stats stats = { | ||
443 | .signal = 0, | ||
444 | .noise = -98, | ||
445 | .rate = 0, | ||
446 | }; | ||
447 | |||
448 | skb_pull(skb, RTL_RX_DESC_SIZE); | ||
449 | rtlpriv->cfg->ops->query_rx_desc(hw, &stats, &rx_status, rxdesc, skb); | ||
450 | skb_pull(skb, (stats.rx_drvinfo_size + stats.rx_bufshift)); | ||
451 | hdr = (struct ieee80211_hdr *)(skb->data); | ||
452 | fc = hdr->frame_control; | ||
453 | if (!stats.crc) { | ||
454 | memcpy(IEEE80211_SKB_RXCB(skb), &rx_status, sizeof(rx_status)); | ||
455 | |||
456 | if (is_broadcast_ether_addr(hdr->addr1)) { | ||
457 | /*TODO*/; | ||
458 | } else if (is_multicast_ether_addr(hdr->addr1)) { | ||
459 | /*TODO*/ | ||
460 | } else { | ||
461 | unicast = true; | ||
462 | rtlpriv->stats.rxbytesunicast += skb->len; | ||
463 | } | ||
464 | |||
465 | rtl_is_special_data(hw, skb, false); | ||
466 | |||
467 | if (ieee80211_is_data(fc)) { | ||
468 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_RX); | ||
469 | |||
470 | if (unicast) | ||
471 | rtlpriv->link_info.num_rx_inperiod++; | ||
472 | } | ||
473 | } | ||
474 | } | ||
475 | |||
476 | static void _rtl_usb_rx_process_noagg(struct ieee80211_hw *hw, | ||
477 | struct sk_buff *skb) | ||
478 | { | ||
479 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
480 | u8 *rxdesc = skb->data; | ||
481 | struct ieee80211_hdr *hdr; | ||
482 | bool unicast = false; | ||
483 | __le16 fc; | ||
484 | struct ieee80211_rx_status rx_status = {0}; | ||
485 | struct rtl_stats stats = { | ||
486 | .signal = 0, | ||
487 | .noise = -98, | ||
488 | .rate = 0, | ||
489 | }; | ||
490 | |||
491 | skb_pull(skb, RTL_RX_DESC_SIZE); | ||
492 | rtlpriv->cfg->ops->query_rx_desc(hw, &stats, &rx_status, rxdesc, skb); | ||
493 | skb_pull(skb, (stats.rx_drvinfo_size + stats.rx_bufshift)); | ||
494 | hdr = (struct ieee80211_hdr *)(skb->data); | ||
495 | fc = hdr->frame_control; | ||
496 | if (!stats.crc) { | ||
497 | memcpy(IEEE80211_SKB_RXCB(skb), &rx_status, sizeof(rx_status)); | ||
498 | |||
499 | if (is_broadcast_ether_addr(hdr->addr1)) { | ||
500 | /*TODO*/; | ||
501 | } else if (is_multicast_ether_addr(hdr->addr1)) { | ||
502 | /*TODO*/ | ||
503 | } else { | ||
504 | unicast = true; | ||
505 | rtlpriv->stats.rxbytesunicast += skb->len; | ||
506 | } | ||
507 | |||
508 | rtl_is_special_data(hw, skb, false); | ||
509 | |||
510 | if (ieee80211_is_data(fc)) { | ||
511 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_RX); | ||
512 | |||
513 | if (unicast) | ||
514 | rtlpriv->link_info.num_rx_inperiod++; | ||
515 | } | ||
516 | if (likely(rtl_action_proc(hw, skb, false))) { | ||
517 | struct sk_buff *uskb = NULL; | ||
518 | u8 *pdata; | ||
519 | |||
520 | uskb = dev_alloc_skb(skb->len + 128); | ||
521 | memcpy(IEEE80211_SKB_RXCB(uskb), &rx_status, | ||
522 | sizeof(rx_status)); | ||
523 | pdata = (u8 *)skb_put(uskb, skb->len); | ||
524 | memcpy(pdata, skb->data, skb->len); | ||
525 | dev_kfree_skb_any(skb); | ||
526 | ieee80211_rx_irqsafe(hw, uskb); | ||
527 | } else { | ||
528 | dev_kfree_skb_any(skb); | ||
529 | } | ||
530 | } | ||
531 | } | ||
532 | |||
533 | static void _rtl_rx_pre_process(struct ieee80211_hw *hw, struct sk_buff *skb) | ||
534 | { | ||
535 | struct sk_buff *_skb; | ||
536 | struct sk_buff_head rx_queue; | ||
537 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
538 | |||
539 | skb_queue_head_init(&rx_queue); | ||
540 | if (rtlusb->usb_rx_segregate_hdl) | ||
541 | rtlusb->usb_rx_segregate_hdl(hw, skb, &rx_queue); | ||
542 | WARN_ON(skb_queue_empty(&rx_queue)); | ||
543 | while (!skb_queue_empty(&rx_queue)) { | ||
544 | _skb = skb_dequeue(&rx_queue); | ||
545 | _rtl_usb_rx_process_agg(hw, skb); | ||
546 | ieee80211_rx_irqsafe(hw, skb); | ||
547 | } | ||
548 | } | ||
549 | |||
550 | static void _rtl_rx_completed(struct urb *_urb) | ||
551 | { | ||
552 | struct sk_buff *skb = (struct sk_buff *)_urb->context; | ||
553 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
554 | struct rtl_usb *rtlusb = (struct rtl_usb *)info->rate_driver_data[0]; | ||
555 | struct ieee80211_hw *hw = usb_get_intfdata(rtlusb->intf); | ||
556 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
557 | int err = 0; | ||
558 | |||
559 | if (unlikely(IS_USB_STOP(rtlusb))) | ||
560 | goto free; | ||
561 | |||
562 | if (likely(0 == _urb->status)) { | ||
563 | /* If this code were moved to work queue, would CPU | ||
564 | * utilization be improved? NOTE: We shall allocate another skb | ||
565 | * and reuse the original one. | ||
566 | */ | ||
567 | skb_put(skb, _urb->actual_length); | ||
568 | |||
569 | if (likely(!rtlusb->usb_rx_segregate_hdl)) { | ||
570 | struct sk_buff *_skb; | ||
571 | _rtl_usb_rx_process_noagg(hw, skb); | ||
572 | _skb = _rtl_prep_rx_urb(hw, rtlusb, _urb, GFP_ATOMIC); | ||
573 | if (IS_ERR(_skb)) { | ||
574 | err = PTR_ERR(_skb); | ||
575 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
576 | ("Can't allocate skb for bulk IN!\n")); | ||
577 | return; | ||
578 | } | ||
579 | skb = _skb; | ||
580 | } else{ | ||
581 | /* TO DO */ | ||
582 | _rtl_rx_pre_process(hw, skb); | ||
583 | printk(KERN_ERR "rtlwifi: rx agg not supported\n"); | ||
584 | } | ||
585 | goto resubmit; | ||
586 | } | ||
587 | |||
588 | switch (_urb->status) { | ||
589 | /* disconnect */ | ||
590 | case -ENOENT: | ||
591 | case -ECONNRESET: | ||
592 | case -ENODEV: | ||
593 | case -ESHUTDOWN: | ||
594 | goto free; | ||
595 | default: | ||
596 | break; | ||
597 | } | ||
598 | |||
599 | resubmit: | ||
600 | skb_reset_tail_pointer(skb); | ||
601 | skb_trim(skb, 0); | ||
602 | |||
603 | usb_anchor_urb(_urb, &rtlusb->rx_submitted); | ||
604 | err = usb_submit_urb(_urb, GFP_ATOMIC); | ||
605 | if (unlikely(err)) { | ||
606 | usb_unanchor_urb(_urb); | ||
607 | goto free; | ||
608 | } | ||
609 | return; | ||
610 | |||
611 | free: | ||
612 | dev_kfree_skb_irq(skb); | ||
613 | } | ||
614 | |||
615 | static int _rtl_usb_receive(struct ieee80211_hw *hw) | ||
616 | { | ||
617 | struct urb *urb; | ||
618 | struct sk_buff *skb; | ||
619 | int err; | ||
620 | int i; | ||
621 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
622 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
623 | |||
624 | WARN_ON(0 == rtlusb->rx_urb_num); | ||
625 | /* 1600 == 1514 + max WLAN header + rtk info */ | ||
626 | WARN_ON(rtlusb->rx_max_size < 1600); | ||
627 | |||
628 | for (i = 0; i < rtlusb->rx_urb_num; i++) { | ||
629 | err = -ENOMEM; | ||
630 | urb = usb_alloc_urb(0, GFP_KERNEL); | ||
631 | if (!urb) { | ||
632 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
633 | ("Failed to alloc URB!!\n")) | ||
634 | goto err_out; | ||
635 | } | ||
636 | |||
637 | skb = _rtl_prep_rx_urb(hw, rtlusb, urb, GFP_KERNEL); | ||
638 | if (IS_ERR(skb)) { | ||
639 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
640 | ("Failed to prep_rx_urb!!\n")) | ||
641 | err = PTR_ERR(skb); | ||
642 | goto err_out; | ||
643 | } | ||
644 | |||
645 | usb_anchor_urb(urb, &rtlusb->rx_submitted); | ||
646 | err = usb_submit_urb(urb, GFP_KERNEL); | ||
647 | if (err) | ||
648 | goto err_out; | ||
649 | usb_free_urb(urb); | ||
650 | } | ||
651 | return 0; | ||
652 | |||
653 | err_out: | ||
654 | usb_kill_anchored_urbs(&rtlusb->rx_submitted); | ||
655 | return err; | ||
656 | } | ||
657 | |||
658 | static int rtl_usb_start(struct ieee80211_hw *hw) | ||
659 | { | ||
660 | int err; | ||
661 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
662 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
663 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
664 | |||
665 | err = rtlpriv->cfg->ops->hw_init(hw); | ||
666 | rtl_init_rx_config(hw); | ||
667 | |||
668 | /* Enable software */ | ||
669 | SET_USB_START(rtlusb); | ||
670 | /* should after adapter start and interrupt enable. */ | ||
671 | set_hal_start(rtlhal); | ||
672 | |||
673 | /* Start bulk IN */ | ||
674 | _rtl_usb_receive(hw); | ||
675 | |||
676 | return err; | ||
677 | } | ||
678 | /** | ||
679 | * | ||
680 | * | ||
681 | */ | ||
682 | |||
683 | /*======================= tx =========================================*/ | ||
684 | static void rtl_usb_cleanup(struct ieee80211_hw *hw) | ||
685 | { | ||
686 | u32 i; | ||
687 | struct sk_buff *_skb; | ||
688 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
689 | struct ieee80211_tx_info *txinfo; | ||
690 | |||
691 | SET_USB_STOP(rtlusb); | ||
692 | |||
693 | /* clean up rx stuff. */ | ||
694 | usb_kill_anchored_urbs(&rtlusb->rx_submitted); | ||
695 | |||
696 | /* clean up tx stuff */ | ||
697 | for (i = 0; i < RTL_USB_MAX_EP_NUM; i++) { | ||
698 | while ((_skb = skb_dequeue(&rtlusb->tx_skb_queue[i]))) { | ||
699 | rtlusb->usb_tx_cleanup(hw, _skb); | ||
700 | txinfo = IEEE80211_SKB_CB(_skb); | ||
701 | ieee80211_tx_info_clear_status(txinfo); | ||
702 | txinfo->flags |= IEEE80211_TX_STAT_ACK; | ||
703 | ieee80211_tx_status_irqsafe(hw, _skb); | ||
704 | } | ||
705 | usb_kill_anchored_urbs(&rtlusb->tx_pending[i]); | ||
706 | } | ||
707 | usb_kill_anchored_urbs(&rtlusb->tx_submitted); | ||
708 | } | ||
709 | |||
710 | /** | ||
711 | * | ||
712 | * We may add some struct into struct rtl_usb later. Do deinit here. | ||
713 | * | ||
714 | */ | ||
715 | static void rtl_usb_deinit(struct ieee80211_hw *hw) | ||
716 | { | ||
717 | rtl_usb_cleanup(hw); | ||
718 | } | ||
719 | |||
720 | static void rtl_usb_stop(struct ieee80211_hw *hw) | ||
721 | { | ||
722 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
723 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
724 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
725 | |||
726 | /* should after adapter start and interrupt enable. */ | ||
727 | set_hal_stop(rtlhal); | ||
728 | /* Enable software */ | ||
729 | SET_USB_STOP(rtlusb); | ||
730 | rtl_usb_deinit(hw); | ||
731 | rtlpriv->cfg->ops->hw_disable(hw); | ||
732 | } | ||
733 | |||
734 | static void _rtl_submit_tx_urb(struct ieee80211_hw *hw, struct urb *_urb) | ||
735 | { | ||
736 | int err; | ||
737 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
738 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
739 | |||
740 | usb_anchor_urb(_urb, &rtlusb->tx_submitted); | ||
741 | err = usb_submit_urb(_urb, GFP_ATOMIC); | ||
742 | if (err < 0) { | ||
743 | struct sk_buff *skb; | ||
744 | |||
745 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
746 | ("Failed to submit urb.\n")); | ||
747 | usb_unanchor_urb(_urb); | ||
748 | skb = (struct sk_buff *)_urb->context; | ||
749 | kfree_skb(skb); | ||
750 | } | ||
751 | usb_free_urb(_urb); | ||
752 | } | ||
753 | |||
754 | static int _usb_tx_post(struct ieee80211_hw *hw, struct urb *urb, | ||
755 | struct sk_buff *skb) | ||
756 | { | ||
757 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
758 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
759 | struct ieee80211_tx_info *txinfo; | ||
760 | |||
761 | rtlusb->usb_tx_post_hdl(hw, urb, skb); | ||
762 | skb_pull(skb, RTL_TX_HEADER_SIZE); | ||
763 | txinfo = IEEE80211_SKB_CB(skb); | ||
764 | ieee80211_tx_info_clear_status(txinfo); | ||
765 | txinfo->flags |= IEEE80211_TX_STAT_ACK; | ||
766 | |||
767 | if (urb->status) { | ||
768 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
769 | ("Urb has error status 0x%X\n", urb->status)); | ||
770 | goto out; | ||
771 | } | ||
772 | /* TODO: statistics */ | ||
773 | out: | ||
774 | ieee80211_tx_status_irqsafe(hw, skb); | ||
775 | return urb->status; | ||
776 | } | ||
777 | |||
778 | static void _rtl_tx_complete(struct urb *urb) | ||
779 | { | ||
780 | struct sk_buff *skb = (struct sk_buff *)urb->context; | ||
781 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
782 | struct rtl_usb *rtlusb = (struct rtl_usb *)info->rate_driver_data[0]; | ||
783 | struct ieee80211_hw *hw = usb_get_intfdata(rtlusb->intf); | ||
784 | int err; | ||
785 | |||
786 | if (unlikely(IS_USB_STOP(rtlusb))) | ||
787 | return; | ||
788 | err = _usb_tx_post(hw, urb, skb); | ||
789 | if (err) { | ||
790 | /* Ignore error and keep issuiing other urbs */ | ||
791 | return; | ||
792 | } | ||
793 | } | ||
794 | |||
795 | static struct urb *_rtl_usb_tx_urb_setup(struct ieee80211_hw *hw, | ||
796 | struct sk_buff *skb, u32 ep_num) | ||
797 | { | ||
798 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
799 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
800 | struct urb *_urb; | ||
801 | |||
802 | WARN_ON(NULL == skb); | ||
803 | _urb = usb_alloc_urb(0, GFP_ATOMIC); | ||
804 | if (!_urb) { | ||
805 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
806 | ("Can't allocate URB for bulk out!\n")); | ||
807 | kfree_skb(skb); | ||
808 | return NULL; | ||
809 | } | ||
810 | _rtl_install_trx_info(rtlusb, skb, ep_num); | ||
811 | usb_fill_bulk_urb(_urb, rtlusb->udev, usb_sndbulkpipe(rtlusb->udev, | ||
812 | ep_num), skb->data, skb->len, _rtl_tx_complete, skb); | ||
813 | _urb->transfer_flags |= URB_ZERO_PACKET; | ||
814 | return _urb; | ||
815 | } | ||
816 | |||
817 | static void _rtl_usb_transmit(struct ieee80211_hw *hw, struct sk_buff *skb, | ||
818 | enum rtl_txq qnum) | ||
819 | { | ||
820 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
821 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
822 | u32 ep_num; | ||
823 | struct urb *_urb = NULL; | ||
824 | struct sk_buff *_skb = NULL; | ||
825 | struct sk_buff_head *skb_list; | ||
826 | struct usb_anchor *urb_list; | ||
827 | |||
828 | WARN_ON(NULL == rtlusb->usb_tx_aggregate_hdl); | ||
829 | if (unlikely(IS_USB_STOP(rtlusb))) { | ||
830 | RT_TRACE(rtlpriv, COMP_USB, DBG_EMERG, | ||
831 | ("USB device is stopping...\n")); | ||
832 | kfree_skb(skb); | ||
833 | return; | ||
834 | } | ||
835 | ep_num = rtlusb->ep_map.ep_mapping[qnum]; | ||
836 | skb_list = &rtlusb->tx_skb_queue[ep_num]; | ||
837 | _skb = skb; | ||
838 | _urb = _rtl_usb_tx_urb_setup(hw, _skb, ep_num); | ||
839 | if (unlikely(!_urb)) { | ||
840 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
841 | ("Can't allocate urb. Drop skb!\n")); | ||
842 | return; | ||
843 | } | ||
844 | urb_list = &rtlusb->tx_pending[ep_num]; | ||
845 | _rtl_submit_tx_urb(hw, _urb); | ||
846 | } | ||
847 | |||
848 | static void _rtl_usb_tx_preprocess(struct ieee80211_hw *hw, struct sk_buff *skb, | ||
849 | u16 hw_queue) | ||
850 | { | ||
851 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
852 | struct rtl_mac *mac = rtl_mac(rtl_priv(hw)); | ||
853 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
854 | struct rtl_tx_desc *pdesc = NULL; | ||
855 | struct rtl_tcb_desc tcb_desc; | ||
856 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)(skb->data); | ||
857 | __le16 fc = hdr->frame_control; | ||
858 | u8 *pda_addr = hdr->addr1; | ||
859 | /* ssn */ | ||
860 | u8 *qc = NULL; | ||
861 | u8 tid = 0; | ||
862 | u16 seq_number = 0; | ||
863 | |||
864 | if (ieee80211_is_auth(fc)) { | ||
865 | RT_TRACE(rtlpriv, COMP_SEND, DBG_DMESG, ("MAC80211_LINKING\n")); | ||
866 | rtl_ips_nic_on(hw); | ||
867 | } | ||
868 | |||
869 | if (rtlpriv->psc.sw_ps_enabled) { | ||
870 | if (ieee80211_is_data(fc) && !ieee80211_is_nullfunc(fc) && | ||
871 | !ieee80211_has_pm(fc)) | ||
872 | hdr->frame_control |= cpu_to_le16(IEEE80211_FCTL_PM); | ||
873 | } | ||
874 | |||
875 | rtl_action_proc(hw, skb, true); | ||
876 | if (is_multicast_ether_addr(pda_addr)) | ||
877 | rtlpriv->stats.txbytesmulticast += skb->len; | ||
878 | else if (is_broadcast_ether_addr(pda_addr)) | ||
879 | rtlpriv->stats.txbytesbroadcast += skb->len; | ||
880 | else | ||
881 | rtlpriv->stats.txbytesunicast += skb->len; | ||
882 | if (ieee80211_is_data_qos(fc)) { | ||
883 | qc = ieee80211_get_qos_ctl(hdr); | ||
884 | tid = qc[0] & IEEE80211_QOS_CTL_TID_MASK; | ||
885 | seq_number = (le16_to_cpu(hdr->seq_ctrl) & | ||
886 | IEEE80211_SCTL_SEQ) >> 4; | ||
887 | seq_number += 1; | ||
888 | seq_number <<= 4; | ||
889 | } | ||
890 | rtlpriv->cfg->ops->fill_tx_desc(hw, hdr, (u8 *)pdesc, info, skb, | ||
891 | hw_queue, &tcb_desc); | ||
892 | if (!ieee80211_has_morefrags(hdr->frame_control)) { | ||
893 | if (qc) | ||
894 | mac->tids[tid].seq_number = seq_number; | ||
895 | } | ||
896 | if (ieee80211_is_data(fc)) | ||
897 | rtlpriv->cfg->ops->led_control(hw, LED_CTL_TX); | ||
898 | } | ||
899 | |||
900 | static int rtl_usb_tx(struct ieee80211_hw *hw, struct sk_buff *skb, | ||
901 | struct rtl_tcb_desc *dummy) | ||
902 | { | ||
903 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
904 | struct rtl_hal *rtlhal = rtl_hal(rtl_priv(hw)); | ||
905 | struct ieee80211_hdr *hdr = (struct ieee80211_hdr *)(skb->data); | ||
906 | __le16 fc = hdr->frame_control; | ||
907 | u16 hw_queue; | ||
908 | |||
909 | if (unlikely(is_hal_stop(rtlhal))) | ||
910 | goto err_free; | ||
911 | hw_queue = rtlusb->usb_mq_to_hwq(fc, skb_get_queue_mapping(skb)); | ||
912 | _rtl_usb_tx_preprocess(hw, skb, hw_queue); | ||
913 | _rtl_usb_transmit(hw, skb, hw_queue); | ||
914 | return NETDEV_TX_OK; | ||
915 | |||
916 | err_free: | ||
917 | dev_kfree_skb_any(skb); | ||
918 | return NETDEV_TX_OK; | ||
919 | } | ||
920 | |||
921 | static bool rtl_usb_tx_chk_waitq_insert(struct ieee80211_hw *hw, | ||
922 | struct sk_buff *skb) | ||
923 | { | ||
924 | return false; | ||
925 | } | ||
926 | |||
927 | static struct rtl_intf_ops rtl_usb_ops = { | ||
928 | .adapter_start = rtl_usb_start, | ||
929 | .adapter_stop = rtl_usb_stop, | ||
930 | .adapter_tx = rtl_usb_tx, | ||
931 | .waitq_insert = rtl_usb_tx_chk_waitq_insert, | ||
932 | }; | ||
933 | |||
934 | int __devinit rtl_usb_probe(struct usb_interface *intf, | ||
935 | const struct usb_device_id *id) | ||
936 | { | ||
937 | int err; | ||
938 | struct ieee80211_hw *hw = NULL; | ||
939 | struct rtl_priv *rtlpriv = NULL; | ||
940 | struct usb_device *udev; | ||
941 | struct rtl_usb_priv *usb_priv; | ||
942 | |||
943 | hw = ieee80211_alloc_hw(sizeof(struct rtl_priv) + | ||
944 | sizeof(struct rtl_usb_priv), &rtl_ops); | ||
945 | if (!hw) { | ||
946 | RT_ASSERT(false, ("%s : ieee80211 alloc failed\n", __func__)); | ||
947 | return -ENOMEM; | ||
948 | } | ||
949 | rtlpriv = hw->priv; | ||
950 | SET_IEEE80211_DEV(hw, &intf->dev); | ||
951 | udev = interface_to_usbdev(intf); | ||
952 | usb_get_dev(udev); | ||
953 | usb_priv = rtl_usbpriv(hw); | ||
954 | memset(usb_priv, 0, sizeof(*usb_priv)); | ||
955 | usb_priv->dev.intf = intf; | ||
956 | usb_priv->dev.udev = udev; | ||
957 | usb_set_intfdata(intf, hw); | ||
958 | /* init cfg & intf_ops */ | ||
959 | rtlpriv->rtlhal.interface = INTF_USB; | ||
960 | rtlpriv->cfg = (struct rtl_hal_cfg *)(id->driver_info); | ||
961 | rtlpriv->intf_ops = &rtl_usb_ops; | ||
962 | rtl_dbgp_flag_init(hw); | ||
963 | /* Init IO handler */ | ||
964 | _rtl_usb_io_handler_init(&udev->dev, hw); | ||
965 | rtlpriv->cfg->ops->read_chip_version(hw); | ||
966 | /*like read eeprom and so on */ | ||
967 | rtlpriv->cfg->ops->read_eeprom_info(hw); | ||
968 | if (rtlpriv->cfg->ops->init_sw_vars(hw)) { | ||
969 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
970 | ("Can't init_sw_vars.\n")); | ||
971 | goto error_out; | ||
972 | } | ||
973 | rtlpriv->cfg->ops->init_sw_leds(hw); | ||
974 | err = _rtl_usb_init(hw); | ||
975 | err = _rtl_usb_init_sw(hw); | ||
976 | /* Init mac80211 sw */ | ||
977 | err = rtl_init_core(hw); | ||
978 | if (err) { | ||
979 | RT_TRACE(rtlpriv, COMP_ERR, DBG_EMERG, | ||
980 | ("Can't allocate sw for mac80211.\n")); | ||
981 | goto error_out; | ||
982 | } | ||
983 | |||
984 | /*init rfkill */ | ||
985 | /* rtl_init_rfkill(hw); */ | ||
986 | |||
987 | err = ieee80211_register_hw(hw); | ||
988 | if (err) { | ||
989 | RT_TRACE(rtlpriv, COMP_INIT, DBG_EMERG, | ||
990 | ("Can't register mac80211 hw.\n")); | ||
991 | goto error_out; | ||
992 | } else { | ||
993 | rtlpriv->mac80211.mac80211_registered = 1; | ||
994 | } | ||
995 | set_bit(RTL_STATUS_INTERFACE_START, &rtlpriv->status); | ||
996 | return 0; | ||
997 | error_out: | ||
998 | rtl_deinit_core(hw); | ||
999 | _rtl_usb_io_handler_release(hw); | ||
1000 | ieee80211_free_hw(hw); | ||
1001 | usb_put_dev(udev); | ||
1002 | return -ENODEV; | ||
1003 | } | ||
1004 | EXPORT_SYMBOL(rtl_usb_probe); | ||
1005 | |||
1006 | void rtl_usb_disconnect(struct usb_interface *intf) | ||
1007 | { | ||
1008 | struct ieee80211_hw *hw = usb_get_intfdata(intf); | ||
1009 | struct rtl_priv *rtlpriv = rtl_priv(hw); | ||
1010 | struct rtl_mac *rtlmac = rtl_mac(rtl_priv(hw)); | ||
1011 | struct rtl_usb *rtlusb = rtl_usbdev(rtl_usbpriv(hw)); | ||
1012 | |||
1013 | if (unlikely(!rtlpriv)) | ||
1014 | return; | ||
1015 | /*ieee80211_unregister_hw will call ops_stop */ | ||
1016 | if (rtlmac->mac80211_registered == 1) { | ||
1017 | ieee80211_unregister_hw(hw); | ||
1018 | rtlmac->mac80211_registered = 0; | ||
1019 | } else { | ||
1020 | rtl_deinit_deferred_work(hw); | ||
1021 | rtlpriv->intf_ops->adapter_stop(hw); | ||
1022 | } | ||
1023 | /*deinit rfkill */ | ||
1024 | /* rtl_deinit_rfkill(hw); */ | ||
1025 | rtl_usb_deinit(hw); | ||
1026 | rtl_deinit_core(hw); | ||
1027 | rtlpriv->cfg->ops->deinit_sw_leds(hw); | ||
1028 | rtlpriv->cfg->ops->deinit_sw_vars(hw); | ||
1029 | _rtl_usb_io_handler_release(hw); | ||
1030 | usb_put_dev(rtlusb->udev); | ||
1031 | usb_set_intfdata(intf, NULL); | ||
1032 | ieee80211_free_hw(hw); | ||
1033 | } | ||
1034 | EXPORT_SYMBOL(rtl_usb_disconnect); | ||
1035 | |||
1036 | int rtl_usb_suspend(struct usb_interface *pusb_intf, pm_message_t message) | ||
1037 | { | ||
1038 | return 0; | ||
1039 | } | ||
1040 | EXPORT_SYMBOL(rtl_usb_suspend); | ||
1041 | |||
1042 | int rtl_usb_resume(struct usb_interface *pusb_intf) | ||
1043 | { | ||
1044 | return 0; | ||
1045 | } | ||
1046 | EXPORT_SYMBOL(rtl_usb_resume); | ||
diff --git a/drivers/net/wireless/rtlwifi/usb.h b/drivers/net/wireless/rtlwifi/usb.h new file mode 100644 index 000000000000..d2a63fb3e1e6 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/usb.h | |||
@@ -0,0 +1,166 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2011 Realtek Corporation. All rights reserved. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | *****************************************************************************/ | ||
27 | |||
28 | #ifndef __RTL_USB_H__ | ||
29 | #define __RTL_USB_H__ | ||
30 | |||
31 | #include <linux/usb.h> | ||
32 | #include <linux/skbuff.h> | ||
33 | |||
34 | #define RTL_RX_DESC_SIZE 24 | ||
35 | |||
36 | #define RTL_USB_DEVICE(vend, prod, cfg) \ | ||
37 | .match_flags = USB_DEVICE_ID_MATCH_DEVICE, \ | ||
38 | .idVendor = (vend), \ | ||
39 | .idProduct = (prod), \ | ||
40 | .driver_info = (kernel_ulong_t)&(cfg) | ||
41 | |||
42 | #define USB_HIGH_SPEED_BULK_SIZE 512 | ||
43 | #define USB_FULL_SPEED_BULK_SIZE 64 | ||
44 | |||
45 | |||
46 | #define RTL_USB_MAX_TXQ_NUM 4 /* max tx queue */ | ||
47 | #define RTL_USB_MAX_EP_NUM 6 /* max ep number */ | ||
48 | #define RTL_USB_MAX_TX_URBS_NUM 8 | ||
49 | |||
50 | enum rtl_txq { | ||
51 | /* These definitions shall be consistent with value | ||
52 | * returned by skb_get_queue_mapping | ||
53 | *------------------------------------*/ | ||
54 | RTL_TXQ_BK, | ||
55 | RTL_TXQ_BE, | ||
56 | RTL_TXQ_VI, | ||
57 | RTL_TXQ_VO, | ||
58 | /*------------------------------------*/ | ||
59 | RTL_TXQ_BCN, | ||
60 | RTL_TXQ_MGT, | ||
61 | RTL_TXQ_HI, | ||
62 | |||
63 | /* Must be last */ | ||
64 | __RTL_TXQ_NUM, | ||
65 | }; | ||
66 | |||
67 | struct rtl_ep_map { | ||
68 | u32 ep_mapping[__RTL_TXQ_NUM]; | ||
69 | }; | ||
70 | |||
71 | struct _trx_info { | ||
72 | struct rtl_usb *rtlusb; | ||
73 | u32 ep_num; | ||
74 | }; | ||
75 | |||
76 | static inline void _rtl_install_trx_info(struct rtl_usb *rtlusb, | ||
77 | struct sk_buff *skb, | ||
78 | u32 ep_num) | ||
79 | { | ||
80 | struct ieee80211_tx_info *info = IEEE80211_SKB_CB(skb); | ||
81 | info->rate_driver_data[0] = rtlusb; | ||
82 | info->rate_driver_data[1] = (void *)(__kernel_size_t)ep_num; | ||
83 | } | ||
84 | |||
85 | |||
86 | /* Add suspend/resume later */ | ||
87 | enum rtl_usb_state { | ||
88 | USB_STATE_STOP = 0, | ||
89 | USB_STATE_START = 1, | ||
90 | }; | ||
91 | |||
92 | #define IS_USB_STOP(rtlusb_ptr) (USB_STATE_STOP == (rtlusb_ptr)->state) | ||
93 | #define IS_USB_START(rtlusb_ptr) (USB_STATE_START == (rtlusb_ptr)->state) | ||
94 | #define SET_USB_STOP(rtlusb_ptr) \ | ||
95 | do { \ | ||
96 | (rtlusb_ptr)->state = USB_STATE_STOP; \ | ||
97 | } while (0) | ||
98 | |||
99 | #define SET_USB_START(rtlusb_ptr) \ | ||
100 | do { \ | ||
101 | (rtlusb_ptr)->state = USB_STATE_START; \ | ||
102 | } while (0) | ||
103 | |||
104 | struct rtl_usb { | ||
105 | struct usb_device *udev; | ||
106 | struct usb_interface *intf; | ||
107 | enum rtl_usb_state state; | ||
108 | |||
109 | /* Bcn control register setting */ | ||
110 | u32 reg_bcn_ctrl_val; | ||
111 | /* for 88/92cu card disable */ | ||
112 | u8 disableHWSM; | ||
113 | /*QOS & EDCA */ | ||
114 | enum acm_method acm_method; | ||
115 | /* irq . HIMR,HIMR_EX */ | ||
116 | u32 irq_mask[2]; | ||
117 | bool irq_enabled; | ||
118 | |||
119 | u16 (*usb_mq_to_hwq)(__le16 fc, u16 mac80211_queue_index); | ||
120 | |||
121 | /* Tx */ | ||
122 | u8 out_ep_nums ; | ||
123 | u8 out_queue_sel; | ||
124 | struct rtl_ep_map ep_map; | ||
125 | |||
126 | u32 max_bulk_out_size; | ||
127 | u32 tx_submitted_urbs; | ||
128 | struct sk_buff_head tx_skb_queue[RTL_USB_MAX_EP_NUM]; | ||
129 | |||
130 | struct usb_anchor tx_pending[RTL_USB_MAX_EP_NUM]; | ||
131 | struct usb_anchor tx_submitted; | ||
132 | |||
133 | struct sk_buff *(*usb_tx_aggregate_hdl)(struct ieee80211_hw *, | ||
134 | struct sk_buff_head *); | ||
135 | int (*usb_tx_post_hdl)(struct ieee80211_hw *, | ||
136 | struct urb *, struct sk_buff *); | ||
137 | void (*usb_tx_cleanup)(struct ieee80211_hw *, struct sk_buff *); | ||
138 | |||
139 | /* Rx */ | ||
140 | u8 in_ep_nums ; | ||
141 | u32 in_ep; /* Bulk IN endpoint number */ | ||
142 | u32 rx_max_size; /* Bulk IN max buffer size */ | ||
143 | u32 rx_urb_num; /* How many Bulk INs are submitted to host. */ | ||
144 | struct usb_anchor rx_submitted; | ||
145 | void (*usb_rx_segregate_hdl)(struct ieee80211_hw *, struct sk_buff *, | ||
146 | struct sk_buff_head *); | ||
147 | void (*usb_rx_hdl)(struct ieee80211_hw *, struct sk_buff *); | ||
148 | }; | ||
149 | |||
150 | struct rtl_usb_priv { | ||
151 | struct rtl_usb dev; | ||
152 | struct rtl_led_ctl ledctl; | ||
153 | }; | ||
154 | |||
155 | #define rtl_usbpriv(hw) (((struct rtl_usb_priv *)(rtl_priv(hw))->priv)) | ||
156 | #define rtl_usbdev(usbpriv) (&((usbpriv)->dev)) | ||
157 | |||
158 | |||
159 | |||
160 | int __devinit rtl_usb_probe(struct usb_interface *intf, | ||
161 | const struct usb_device_id *id); | ||
162 | void rtl_usb_disconnect(struct usb_interface *intf); | ||
163 | int rtl_usb_suspend(struct usb_interface *pusb_intf, pm_message_t message); | ||
164 | int rtl_usb_resume(struct usb_interface *pusb_intf); | ||
165 | |||
166 | #endif | ||
diff --git a/drivers/net/wireless/rtlwifi/wifi.h b/drivers/net/wireless/rtlwifi/wifi.h new file mode 100644 index 000000000000..693395ee98f9 --- /dev/null +++ b/drivers/net/wireless/rtlwifi/wifi.h | |||
@@ -0,0 +1,1991 @@ | |||
1 | /****************************************************************************** | ||
2 | * | ||
3 | * Copyright(c) 2009-2010 Realtek Corporation. | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of version 2 of the GNU General Public License as | ||
7 | * published by the Free Software Foundation. | ||
8 | * | ||
9 | * This program is distributed in the hope that it will be useful, but WITHOUT | ||
10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
11 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
12 | * more details. | ||
13 | * | ||
14 | * You should have received a copy of the GNU General Public License along with | ||
15 | * this program; if not, write to the Free Software Foundation, Inc., | ||
16 | * 51 Franklin Street, Fifth Floor, Boston, MA 02110, USA | ||
17 | * | ||
18 | * The full GNU General Public License is included in this distribution in the | ||
19 | * file called LICENSE. | ||
20 | * | ||
21 | * Contact Information: | ||
22 | * wlanfae <wlanfae@realtek.com> | ||
23 | * Realtek Corporation, No. 2, Innovation Road II, Hsinchu Science Park, | ||
24 | * Hsinchu 300, Taiwan. | ||
25 | * | ||
26 | * Larry Finger <Larry.Finger@lwfinger.net> | ||
27 | * | ||
28 | *****************************************************************************/ | ||
29 | |||
30 | #ifndef __RTL_WIFI_H__ | ||
31 | #define __RTL_WIFI_H__ | ||
32 | |||
33 | #include <linux/sched.h> | ||
34 | #include <linux/firmware.h> | ||
35 | #include <linux/version.h> | ||
36 | #include <linux/etherdevice.h> | ||
37 | #include <linux/vmalloc.h> | ||
38 | #include <linux/usb.h> | ||
39 | #include <net/mac80211.h> | ||
40 | #include "debug.h" | ||
41 | |||
42 | #define RF_CHANGE_BY_INIT 0 | ||
43 | #define RF_CHANGE_BY_IPS BIT(28) | ||
44 | #define RF_CHANGE_BY_PS BIT(29) | ||
45 | #define RF_CHANGE_BY_HW BIT(30) | ||
46 | #define RF_CHANGE_BY_SW BIT(31) | ||
47 | |||
48 | #define IQK_ADDA_REG_NUM 16 | ||
49 | #define IQK_MAC_REG_NUM 4 | ||
50 | |||
51 | #define MAX_KEY_LEN 61 | ||
52 | #define KEY_BUF_SIZE 5 | ||
53 | |||
54 | /* QoS related. */ | ||
55 | /*aci: 0x00 Best Effort*/ | ||
56 | /*aci: 0x01 Background*/ | ||
57 | /*aci: 0x10 Video*/ | ||
58 | /*aci: 0x11 Voice*/ | ||
59 | /*Max: define total number.*/ | ||
60 | #define AC0_BE 0 | ||
61 | #define AC1_BK 1 | ||
62 | #define AC2_VI 2 | ||
63 | #define AC3_VO 3 | ||
64 | #define AC_MAX 4 | ||
65 | #define QOS_QUEUE_NUM 4 | ||
66 | #define RTL_MAC80211_NUM_QUEUE 5 | ||
67 | |||
68 | #define QBSS_LOAD_SIZE 5 | ||
69 | #define MAX_WMMELE_LENGTH 64 | ||
70 | |||
71 | #define TOTAL_CAM_ENTRY 32 | ||
72 | |||
73 | /*slot time for 11g. */ | ||
74 | #define RTL_SLOT_TIME_9 9 | ||
75 | #define RTL_SLOT_TIME_20 20 | ||
76 | |||
77 | /*related with tcp/ip. */ | ||
78 | /*if_ehther.h*/ | ||
79 | #define ETH_P_PAE 0x888E /*Port Access Entity (IEEE 802.1X) */ | ||
80 | #define ETH_P_IP 0x0800 /*Internet Protocol packet */ | ||
81 | #define ETH_P_ARP 0x0806 /*Address Resolution packet */ | ||
82 | #define SNAP_SIZE 6 | ||
83 | #define PROTOC_TYPE_SIZE 2 | ||
84 | |||
85 | /*related with 802.11 frame*/ | ||
86 | #define MAC80211_3ADDR_LEN 24 | ||
87 | #define MAC80211_4ADDR_LEN 30 | ||
88 | |||
89 | #define CHANNEL_MAX_NUMBER (14 + 24 + 21) /* 14 is the max channel no */ | ||
90 | #define CHANNEL_GROUP_MAX (3 + 9) /* ch1~3, 4~9, 10~14 = three groups */ | ||
91 | #define MAX_PG_GROUP 13 | ||
92 | #define CHANNEL_GROUP_MAX_2G 3 | ||
93 | #define CHANNEL_GROUP_IDX_5GL 3 | ||
94 | #define CHANNEL_GROUP_IDX_5GM 6 | ||
95 | #define CHANNEL_GROUP_IDX_5GH 9 | ||
96 | #define CHANNEL_GROUP_MAX_5G 9 | ||
97 | #define CHANNEL_MAX_NUMBER_2G 14 | ||
98 | #define AVG_THERMAL_NUM 8 | ||
99 | #define MAX_TID_COUNT 9 | ||
100 | |||
101 | /* for early mode */ | ||
102 | #define FCS_LEN 4 | ||
103 | #define EM_HDR_LEN 8 | ||
104 | enum intf_type { | ||
105 | INTF_PCI = 0, | ||
106 | INTF_USB = 1, | ||
107 | }; | ||
108 | |||
109 | enum radio_path { | ||
110 | RF90_PATH_A = 0, | ||
111 | RF90_PATH_B = 1, | ||
112 | RF90_PATH_C = 2, | ||
113 | RF90_PATH_D = 3, | ||
114 | }; | ||
115 | |||
116 | enum rt_eeprom_type { | ||
117 | EEPROM_93C46, | ||
118 | EEPROM_93C56, | ||
119 | EEPROM_BOOT_EFUSE, | ||
120 | }; | ||
121 | |||
122 | enum rtl_status { | ||
123 | RTL_STATUS_INTERFACE_START = 0, | ||
124 | }; | ||
125 | |||
126 | enum hardware_type { | ||
127 | HARDWARE_TYPE_RTL8192E, | ||
128 | HARDWARE_TYPE_RTL8192U, | ||
129 | HARDWARE_TYPE_RTL8192SE, | ||
130 | HARDWARE_TYPE_RTL8192SU, | ||
131 | HARDWARE_TYPE_RTL8192CE, | ||
132 | HARDWARE_TYPE_RTL8192CU, | ||
133 | HARDWARE_TYPE_RTL8192DE, | ||
134 | HARDWARE_TYPE_RTL8192DU, | ||
135 | HARDWARE_TYPE_RTL8723E, | ||
136 | HARDWARE_TYPE_RTL8723U, | ||
137 | |||
138 | /* keep it last */ | ||
139 | HARDWARE_TYPE_NUM | ||
140 | }; | ||
141 | |||
142 | #define IS_HARDWARE_TYPE_8192SU(rtlhal) \ | ||
143 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SU) | ||
144 | #define IS_HARDWARE_TYPE_8192SE(rtlhal) \ | ||
145 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192SE) | ||
146 | #define IS_HARDWARE_TYPE_8192CE(rtlhal) \ | ||
147 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192CE) | ||
148 | #define IS_HARDWARE_TYPE_8192CU(rtlhal) \ | ||
149 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192CU) | ||
150 | #define IS_HARDWARE_TYPE_8192DE(rtlhal) \ | ||
151 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192DE) | ||
152 | #define IS_HARDWARE_TYPE_8192DU(rtlhal) \ | ||
153 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8192DU) | ||
154 | #define IS_HARDWARE_TYPE_8723E(rtlhal) \ | ||
155 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8723E) | ||
156 | #define IS_HARDWARE_TYPE_8723U(rtlhal) \ | ||
157 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8723U) | ||
158 | #define IS_HARDWARE_TYPE_8192S(rtlhal) \ | ||
159 | (IS_HARDWARE_TYPE_8192SE(rtlhal) || IS_HARDWARE_TYPE_8192SU(rtlhal)) | ||
160 | #define IS_HARDWARE_TYPE_8192C(rtlhal) \ | ||
161 | (IS_HARDWARE_TYPE_8192CE(rtlhal) || IS_HARDWARE_TYPE_8192CU(rtlhal)) | ||
162 | #define IS_HARDWARE_TYPE_8192D(rtlhal) \ | ||
163 | (IS_HARDWARE_TYPE_8192DE(rtlhal) || IS_HARDWARE_TYPE_8192DU(rtlhal)) | ||
164 | #define IS_HARDWARE_TYPE_8723(rtlhal) \ | ||
165 | (IS_HARDWARE_TYPE_8723E(rtlhal) || IS_HARDWARE_TYPE_8723U(rtlhal)) | ||
166 | #define IS_HARDWARE_TYPE_8723U(rtlhal) \ | ||
167 | (rtlhal->hw_type == HARDWARE_TYPE_RTL8723U) | ||
168 | |||
169 | enum scan_operation_backup_opt { | ||
170 | SCAN_OPT_BACKUP = 0, | ||
171 | SCAN_OPT_RESTORE, | ||
172 | SCAN_OPT_MAX | ||
173 | }; | ||
174 | |||
175 | /*RF state.*/ | ||
176 | enum rf_pwrstate { | ||
177 | ERFON, | ||
178 | ERFSLEEP, | ||
179 | ERFOFF | ||
180 | }; | ||
181 | |||
182 | struct bb_reg_def { | ||
183 | u32 rfintfs; | ||
184 | u32 rfintfi; | ||
185 | u32 rfintfo; | ||
186 | u32 rfintfe; | ||
187 | u32 rf3wire_offset; | ||
188 | u32 rflssi_select; | ||
189 | u32 rftxgain_stage; | ||
190 | u32 rfhssi_para1; | ||
191 | u32 rfhssi_para2; | ||
192 | u32 rfswitch_control; | ||
193 | u32 rfagc_control1; | ||
194 | u32 rfagc_control2; | ||
195 | u32 rfrxiq_imbalance; | ||
196 | u32 rfrx_afe; | ||
197 | u32 rftxiq_imbalance; | ||
198 | u32 rftx_afe; | ||
199 | u32 rflssi_readback; | ||
200 | u32 rflssi_readbackpi; | ||
201 | }; | ||
202 | |||
203 | enum io_type { | ||
204 | IO_CMD_PAUSE_DM_BY_SCAN = 0, | ||
205 | IO_CMD_RESUME_DM_BY_SCAN = 1, | ||
206 | }; | ||
207 | |||
208 | enum hw_variables { | ||
209 | HW_VAR_ETHER_ADDR, | ||
210 | HW_VAR_MULTICAST_REG, | ||
211 | HW_VAR_BASIC_RATE, | ||
212 | HW_VAR_BSSID, | ||
213 | HW_VAR_MEDIA_STATUS, | ||
214 | HW_VAR_SECURITY_CONF, | ||
215 | HW_VAR_BEACON_INTERVAL, | ||
216 | HW_VAR_ATIM_WINDOW, | ||
217 | HW_VAR_LISTEN_INTERVAL, | ||
218 | HW_VAR_CS_COUNTER, | ||
219 | HW_VAR_DEFAULTKEY0, | ||
220 | HW_VAR_DEFAULTKEY1, | ||
221 | HW_VAR_DEFAULTKEY2, | ||
222 | HW_VAR_DEFAULTKEY3, | ||
223 | HW_VAR_SIFS, | ||
224 | HW_VAR_DIFS, | ||
225 | HW_VAR_EIFS, | ||
226 | HW_VAR_SLOT_TIME, | ||
227 | HW_VAR_ACK_PREAMBLE, | ||
228 | HW_VAR_CW_CONFIG, | ||
229 | HW_VAR_CW_VALUES, | ||
230 | HW_VAR_RATE_FALLBACK_CONTROL, | ||
231 | HW_VAR_CONTENTION_WINDOW, | ||
232 | HW_VAR_RETRY_COUNT, | ||
233 | HW_VAR_TR_SWITCH, | ||
234 | HW_VAR_COMMAND, | ||
235 | HW_VAR_WPA_CONFIG, | ||
236 | HW_VAR_AMPDU_MIN_SPACE, | ||
237 | HW_VAR_SHORTGI_DENSITY, | ||
238 | HW_VAR_AMPDU_FACTOR, | ||
239 | HW_VAR_MCS_RATE_AVAILABLE, | ||
240 | HW_VAR_AC_PARAM, | ||
241 | HW_VAR_ACM_CTRL, | ||
242 | HW_VAR_DIS_Req_Qsize, | ||
243 | HW_VAR_CCX_CHNL_LOAD, | ||
244 | HW_VAR_CCX_NOISE_HISTOGRAM, | ||
245 | HW_VAR_CCX_CLM_NHM, | ||
246 | HW_VAR_TxOPLimit, | ||
247 | HW_VAR_TURBO_MODE, | ||
248 | HW_VAR_RF_STATE, | ||
249 | HW_VAR_RF_OFF_BY_HW, | ||
250 | HW_VAR_BUS_SPEED, | ||
251 | HW_VAR_SET_DEV_POWER, | ||
252 | |||
253 | HW_VAR_RCR, | ||
254 | HW_VAR_RATR_0, | ||
255 | HW_VAR_RRSR, | ||
256 | HW_VAR_CPU_RST, | ||
257 | HW_VAR_CECHK_BSSID, | ||
258 | HW_VAR_LBK_MODE, | ||
259 | HW_VAR_AES_11N_FIX, | ||
260 | HW_VAR_USB_RX_AGGR, | ||
261 | HW_VAR_USER_CONTROL_TURBO_MODE, | ||
262 | HW_VAR_RETRY_LIMIT, | ||
263 | HW_VAR_INIT_TX_RATE, | ||
264 | HW_VAR_TX_RATE_REG, | ||
265 | HW_VAR_EFUSE_USAGE, | ||
266 | HW_VAR_EFUSE_BYTES, | ||
267 | HW_VAR_AUTOLOAD_STATUS, | ||
268 | HW_VAR_RF_2R_DISABLE, | ||
269 | HW_VAR_SET_RPWM, | ||
270 | HW_VAR_H2C_FW_PWRMODE, | ||
271 | HW_VAR_H2C_FW_JOINBSSRPT, | ||
272 | HW_VAR_FW_PSMODE_STATUS, | ||
273 | HW_VAR_1X1_RECV_COMBINE, | ||
274 | HW_VAR_STOP_SEND_BEACON, | ||
275 | HW_VAR_TSF_TIMER, | ||
276 | HW_VAR_IO_CMD, | ||
277 | |||
278 | HW_VAR_RF_RECOVERY, | ||
279 | HW_VAR_H2C_FW_UPDATE_GTK, | ||
280 | HW_VAR_WF_MASK, | ||
281 | HW_VAR_WF_CRC, | ||
282 | HW_VAR_WF_IS_MAC_ADDR, | ||
283 | HW_VAR_H2C_FW_OFFLOAD, | ||
284 | HW_VAR_RESET_WFCRC, | ||
285 | |||
286 | HW_VAR_HANDLE_FW_C2H, | ||
287 | HW_VAR_DL_FW_RSVD_PAGE, | ||
288 | HW_VAR_AID, | ||
289 | HW_VAR_HW_SEQ_ENABLE, | ||
290 | HW_VAR_CORRECT_TSF, | ||
291 | HW_VAR_BCN_VALID, | ||
292 | HW_VAR_FWLPS_RF_ON, | ||
293 | HW_VAR_DUAL_TSF_RST, | ||
294 | HW_VAR_SWITCH_EPHY_WoWLAN, | ||
295 | HW_VAR_INT_MIGRATION, | ||
296 | HW_VAR_INT_AC, | ||
297 | HW_VAR_RF_TIMING, | ||
298 | |||
299 | HW_VAR_MRC, | ||
300 | |||
301 | HW_VAR_MGT_FILTER, | ||
302 | HW_VAR_CTRL_FILTER, | ||
303 | HW_VAR_DATA_FILTER, | ||
304 | }; | ||
305 | |||
306 | #define HWSET_MAX_SIZE 128 | ||
307 | #define EFUSE_MAX_SECTION 16 | ||
308 | |||
309 | enum _RT_MEDIA_STATUS { | ||
310 | RT_MEDIA_DISCONNECT = 0, | ||
311 | RT_MEDIA_CONNECT = 1 | ||
312 | }; | ||
313 | |||
314 | enum rt_oem_id { | ||
315 | RT_CID_DEFAULT = 0, | ||
316 | RT_CID_8187_ALPHA0 = 1, | ||
317 | RT_CID_8187_SERCOMM_PS = 2, | ||
318 | RT_CID_8187_HW_LED = 3, | ||
319 | RT_CID_8187_NETGEAR = 4, | ||
320 | RT_CID_WHQL = 5, | ||
321 | RT_CID_819x_CAMEO = 6, | ||
322 | RT_CID_819x_RUNTOP = 7, | ||
323 | RT_CID_819x_Senao = 8, | ||
324 | RT_CID_TOSHIBA = 9, | ||
325 | RT_CID_819x_Netcore = 10, | ||
326 | RT_CID_Nettronix = 11, | ||
327 | RT_CID_DLINK = 12, | ||
328 | RT_CID_PRONET = 13, | ||
329 | RT_CID_COREGA = 14, | ||
330 | RT_CID_819x_ALPHA = 15, | ||
331 | RT_CID_819x_Sitecom = 16, | ||
332 | RT_CID_CCX = 17, | ||
333 | RT_CID_819x_Lenovo = 18, | ||
334 | RT_CID_819x_QMI = 19, | ||
335 | RT_CID_819x_Edimax_Belkin = 20, | ||
336 | RT_CID_819x_Sercomm_Belkin = 21, | ||
337 | RT_CID_819x_CAMEO1 = 22, | ||
338 | RT_CID_819x_MSI = 23, | ||
339 | RT_CID_819x_Acer = 24, | ||
340 | RT_CID_819x_HP = 27, | ||
341 | RT_CID_819x_CLEVO = 28, | ||
342 | RT_CID_819x_Arcadyan_Belkin = 29, | ||
343 | RT_CID_819x_SAMSUNG = 30, | ||
344 | RT_CID_819x_WNC_COREGA = 31, | ||
345 | RT_CID_819x_Foxcoon = 32, | ||
346 | RT_CID_819x_DELL = 33, | ||
347 | }; | ||
348 | |||
349 | enum hw_descs { | ||
350 | HW_DESC_OWN, | ||
351 | HW_DESC_RXOWN, | ||
352 | HW_DESC_TX_NEXTDESC_ADDR, | ||
353 | HW_DESC_TXBUFF_ADDR, | ||
354 | HW_DESC_RXBUFF_ADDR, | ||
355 | HW_DESC_RXPKT_LEN, | ||
356 | HW_DESC_RXERO, | ||
357 | }; | ||
358 | |||
359 | enum prime_sc { | ||
360 | PRIME_CHNL_OFFSET_DONT_CARE = 0, | ||
361 | PRIME_CHNL_OFFSET_LOWER = 1, | ||
362 | PRIME_CHNL_OFFSET_UPPER = 2, | ||
363 | }; | ||
364 | |||
365 | enum rf_type { | ||
366 | RF_1T1R = 0, | ||
367 | RF_1T2R = 1, | ||
368 | RF_2T2R = 2, | ||
369 | RF_2T2R_GREEN = 3, | ||
370 | }; | ||
371 | |||
372 | enum ht_channel_width { | ||
373 | HT_CHANNEL_WIDTH_20 = 0, | ||
374 | HT_CHANNEL_WIDTH_20_40 = 1, | ||
375 | }; | ||
376 | |||
377 | /* Ref: 802.11i sepc D10.0 7.3.2.25.1 | ||
378 | Cipher Suites Encryption Algorithms */ | ||
379 | enum rt_enc_alg { | ||
380 | NO_ENCRYPTION = 0, | ||
381 | WEP40_ENCRYPTION = 1, | ||
382 | TKIP_ENCRYPTION = 2, | ||
383 | RSERVED_ENCRYPTION = 3, | ||
384 | AESCCMP_ENCRYPTION = 4, | ||
385 | WEP104_ENCRYPTION = 5, | ||
386 | }; | ||
387 | |||
388 | enum rtl_hal_state { | ||
389 | _HAL_STATE_STOP = 0, | ||
390 | _HAL_STATE_START = 1, | ||
391 | }; | ||
392 | |||
393 | enum rtl_var_map { | ||
394 | /*reg map */ | ||
395 | SYS_ISO_CTRL = 0, | ||
396 | SYS_FUNC_EN, | ||
397 | SYS_CLK, | ||
398 | MAC_RCR_AM, | ||
399 | MAC_RCR_AB, | ||
400 | MAC_RCR_ACRC32, | ||
401 | MAC_RCR_ACF, | ||
402 | MAC_RCR_AAP, | ||
403 | |||
404 | /*efuse map */ | ||
405 | EFUSE_TEST, | ||
406 | EFUSE_CTRL, | ||
407 | EFUSE_CLK, | ||
408 | EFUSE_CLK_CTRL, | ||
409 | EFUSE_PWC_EV12V, | ||
410 | EFUSE_FEN_ELDR, | ||
411 | EFUSE_LOADER_CLK_EN, | ||
412 | EFUSE_ANA8M, | ||
413 | EFUSE_HWSET_MAX_SIZE, | ||
414 | EFUSE_MAX_SECTION_MAP, | ||
415 | EFUSE_REAL_CONTENT_SIZE, | ||
416 | |||
417 | /*CAM map */ | ||
418 | RWCAM, | ||
419 | WCAMI, | ||
420 | RCAMO, | ||
421 | CAMDBG, | ||
422 | SECR, | ||
423 | SEC_CAM_NONE, | ||
424 | SEC_CAM_WEP40, | ||
425 | SEC_CAM_TKIP, | ||
426 | SEC_CAM_AES, | ||
427 | SEC_CAM_WEP104, | ||
428 | |||
429 | /*IMR map */ | ||
430 | RTL_IMR_BCNDMAINT6, /*Beacon DMA Interrupt 6 */ | ||
431 | RTL_IMR_BCNDMAINT5, /*Beacon DMA Interrupt 5 */ | ||
432 | RTL_IMR_BCNDMAINT4, /*Beacon DMA Interrupt 4 */ | ||
433 | RTL_IMR_BCNDMAINT3, /*Beacon DMA Interrupt 3 */ | ||
434 | RTL_IMR_BCNDMAINT2, /*Beacon DMA Interrupt 2 */ | ||
435 | RTL_IMR_BCNDMAINT1, /*Beacon DMA Interrupt 1 */ | ||
436 | RTL_IMR_BCNDOK8, /*Beacon Queue DMA OK Interrup 8 */ | ||
437 | RTL_IMR_BCNDOK7, /*Beacon Queue DMA OK Interrup 7 */ | ||
438 | RTL_IMR_BCNDOK6, /*Beacon Queue DMA OK Interrup 6 */ | ||
439 | RTL_IMR_BCNDOK5, /*Beacon Queue DMA OK Interrup 5 */ | ||
440 | RTL_IMR_BCNDOK4, /*Beacon Queue DMA OK Interrup 4 */ | ||
441 | RTL_IMR_BCNDOK3, /*Beacon Queue DMA OK Interrup 3 */ | ||
442 | RTL_IMR_BCNDOK2, /*Beacon Queue DMA OK Interrup 2 */ | ||
443 | RTL_IMR_BCNDOK1, /*Beacon Queue DMA OK Interrup 1 */ | ||
444 | RTL_IMR_TIMEOUT2, /*Timeout interrupt 2 */ | ||
445 | RTL_IMR_TIMEOUT1, /*Timeout interrupt 1 */ | ||
446 | RTL_IMR_TXFOVW, /*Transmit FIFO Overflow */ | ||
447 | RTL_IMR_PSTIMEOUT, /*Power save time out interrupt */ | ||
448 | RTL_IMR_BcnInt, /*Beacon DMA Interrupt 0 */ | ||
449 | RTL_IMR_RXFOVW, /*Receive FIFO Overflow */ | ||
450 | RTL_IMR_RDU, /*Receive Descriptor Unavailable */ | ||
451 | RTL_IMR_ATIMEND, /*For 92C,ATIM Window End Interrupt */ | ||
452 | RTL_IMR_BDOK, /*Beacon Queue DMA OK Interrup */ | ||
453 | RTL_IMR_HIGHDOK, /*High Queue DMA OK Interrupt */ | ||
454 | RTL_IMR_COMDOK, /*Command Queue DMA OK Interrupt*/ | ||
455 | RTL_IMR_TBDOK, /*Transmit Beacon OK interrup */ | ||
456 | RTL_IMR_MGNTDOK, /*Management Queue DMA OK Interrupt */ | ||
457 | RTL_IMR_TBDER, /*For 92C,Transmit Beacon Error Interrupt */ | ||
458 | RTL_IMR_BKDOK, /*AC_BK DMA OK Interrupt */ | ||
459 | RTL_IMR_BEDOK, /*AC_BE DMA OK Interrupt */ | ||
460 | RTL_IMR_VIDOK, /*AC_VI DMA OK Interrupt */ | ||
461 | RTL_IMR_VODOK, /*AC_VO DMA Interrupt */ | ||
462 | RTL_IMR_ROK, /*Receive DMA OK Interrupt */ | ||
463 | RTL_IBSS_INT_MASKS, /*(RTL_IMR_BcnInt | RTL_IMR_TBDOK | | ||
464 | * RTL_IMR_TBDER) */ | ||
465 | |||
466 | /*CCK Rates, TxHT = 0 */ | ||
467 | RTL_RC_CCK_RATE1M, | ||
468 | RTL_RC_CCK_RATE2M, | ||
469 | RTL_RC_CCK_RATE5_5M, | ||
470 | RTL_RC_CCK_RATE11M, | ||
471 | |||
472 | /*OFDM Rates, TxHT = 0 */ | ||
473 | RTL_RC_OFDM_RATE6M, | ||
474 | RTL_RC_OFDM_RATE9M, | ||
475 | RTL_RC_OFDM_RATE12M, | ||
476 | RTL_RC_OFDM_RATE18M, | ||
477 | RTL_RC_OFDM_RATE24M, | ||
478 | RTL_RC_OFDM_RATE36M, | ||
479 | RTL_RC_OFDM_RATE48M, | ||
480 | RTL_RC_OFDM_RATE54M, | ||
481 | |||
482 | RTL_RC_HT_RATEMCS7, | ||
483 | RTL_RC_HT_RATEMCS15, | ||
484 | |||
485 | /*keep it last */ | ||
486 | RTL_VAR_MAP_MAX, | ||
487 | }; | ||
488 | |||
489 | /*Firmware PS mode for control LPS.*/ | ||
490 | enum _fw_ps_mode { | ||
491 | FW_PS_ACTIVE_MODE = 0, | ||
492 | FW_PS_MIN_MODE = 1, | ||
493 | FW_PS_MAX_MODE = 2, | ||
494 | FW_PS_DTIM_MODE = 3, | ||
495 | FW_PS_VOIP_MODE = 4, | ||
496 | FW_PS_UAPSD_WMM_MODE = 5, | ||
497 | FW_PS_UAPSD_MODE = 6, | ||
498 | FW_PS_IBSS_MODE = 7, | ||
499 | FW_PS_WWLAN_MODE = 8, | ||
500 | FW_PS_PM_Radio_Off = 9, | ||
501 | FW_PS_PM_Card_Disable = 10, | ||
502 | }; | ||
503 | |||
504 | enum rt_psmode { | ||
505 | EACTIVE, /*Active/Continuous access. */ | ||
506 | EMAXPS, /*Max power save mode. */ | ||
507 | EFASTPS, /*Fast power save mode. */ | ||
508 | EAUTOPS, /*Auto power save mode. */ | ||
509 | }; | ||
510 | |||
511 | /*LED related.*/ | ||
512 | enum led_ctl_mode { | ||
513 | LED_CTL_POWER_ON = 1, | ||
514 | LED_CTL_LINK = 2, | ||
515 | LED_CTL_NO_LINK = 3, | ||
516 | LED_CTL_TX = 4, | ||
517 | LED_CTL_RX = 5, | ||
518 | LED_CTL_SITE_SURVEY = 6, | ||
519 | LED_CTL_POWER_OFF = 7, | ||
520 | LED_CTL_START_TO_LINK = 8, | ||
521 | LED_CTL_START_WPS = 9, | ||
522 | LED_CTL_STOP_WPS = 10, | ||
523 | }; | ||
524 | |||
525 | enum rtl_led_pin { | ||
526 | LED_PIN_GPIO0, | ||
527 | LED_PIN_LED0, | ||
528 | LED_PIN_LED1, | ||
529 | LED_PIN_LED2 | ||
530 | }; | ||
531 | |||
532 | /*QoS related.*/ | ||
533 | /*acm implementation method.*/ | ||
534 | enum acm_method { | ||
535 | eAcmWay0_SwAndHw = 0, | ||
536 | eAcmWay1_HW = 1, | ||
537 | eAcmWay2_SW = 2, | ||
538 | }; | ||
539 | |||
540 | enum macphy_mode { | ||
541 | SINGLEMAC_SINGLEPHY = 0, | ||
542 | DUALMAC_DUALPHY, | ||
543 | DUALMAC_SINGLEPHY, | ||
544 | }; | ||
545 | |||
546 | enum band_type { | ||
547 | BAND_ON_2_4G = 0, | ||
548 | BAND_ON_5G, | ||
549 | BAND_ON_BOTH, | ||
550 | BANDMAX | ||
551 | }; | ||
552 | |||
553 | /*aci/aifsn Field. | ||
554 | Ref: WMM spec 2.2.2: WME Parameter Element, p.12.*/ | ||
555 | union aci_aifsn { | ||
556 | u8 char_data; | ||
557 | |||
558 | struct { | ||
559 | u8 aifsn:4; | ||
560 | u8 acm:1; | ||
561 | u8 aci:2; | ||
562 | u8 reserved:1; | ||
563 | } f; /* Field */ | ||
564 | }; | ||
565 | |||
566 | /*mlme related.*/ | ||
567 | enum wireless_mode { | ||
568 | WIRELESS_MODE_UNKNOWN = 0x00, | ||
569 | WIRELESS_MODE_A = 0x01, | ||
570 | WIRELESS_MODE_B = 0x02, | ||
571 | WIRELESS_MODE_G = 0x04, | ||
572 | WIRELESS_MODE_AUTO = 0x08, | ||
573 | WIRELESS_MODE_N_24G = 0x10, | ||
574 | WIRELESS_MODE_N_5G = 0x20 | ||
575 | }; | ||
576 | |||
577 | #define IS_WIRELESS_MODE_A(wirelessmode) \ | ||
578 | (wirelessmode == WIRELESS_MODE_A) | ||
579 | #define IS_WIRELESS_MODE_B(wirelessmode) \ | ||
580 | (wirelessmode == WIRELESS_MODE_B) | ||
581 | #define IS_WIRELESS_MODE_G(wirelessmode) \ | ||
582 | (wirelessmode == WIRELESS_MODE_G) | ||
583 | #define IS_WIRELESS_MODE_N_24G(wirelessmode) \ | ||
584 | (wirelessmode == WIRELESS_MODE_N_24G) | ||
585 | #define IS_WIRELESS_MODE_N_5G(wirelessmode) \ | ||
586 | (wirelessmode == WIRELESS_MODE_N_5G) | ||
587 | |||
588 | enum ratr_table_mode { | ||
589 | RATR_INX_WIRELESS_NGB = 0, | ||
590 | RATR_INX_WIRELESS_NG = 1, | ||
591 | RATR_INX_WIRELESS_NB = 2, | ||
592 | RATR_INX_WIRELESS_N = 3, | ||
593 | RATR_INX_WIRELESS_GB = 4, | ||
594 | RATR_INX_WIRELESS_G = 5, | ||
595 | RATR_INX_WIRELESS_B = 6, | ||
596 | RATR_INX_WIRELESS_MC = 7, | ||
597 | RATR_INX_WIRELESS_A = 8, | ||
598 | }; | ||
599 | |||
600 | enum rtl_link_state { | ||
601 | MAC80211_NOLINK = 0, | ||
602 | MAC80211_LINKING = 1, | ||
603 | MAC80211_LINKED = 2, | ||
604 | MAC80211_LINKED_SCANNING = 3, | ||
605 | }; | ||
606 | |||
607 | enum act_category { | ||
608 | ACT_CAT_QOS = 1, | ||
609 | ACT_CAT_DLS = 2, | ||
610 | ACT_CAT_BA = 3, | ||
611 | ACT_CAT_HT = 7, | ||
612 | ACT_CAT_WMM = 17, | ||
613 | }; | ||
614 | |||
615 | enum ba_action { | ||
616 | ACT_ADDBAREQ = 0, | ||
617 | ACT_ADDBARSP = 1, | ||
618 | ACT_DELBA = 2, | ||
619 | }; | ||
620 | |||
621 | struct octet_string { | ||
622 | u8 *octet; | ||
623 | u16 length; | ||
624 | }; | ||
625 | |||
626 | struct rtl_hdr_3addr { | ||
627 | __le16 frame_ctl; | ||
628 | __le16 duration_id; | ||
629 | u8 addr1[ETH_ALEN]; | ||
630 | u8 addr2[ETH_ALEN]; | ||
631 | u8 addr3[ETH_ALEN]; | ||
632 | __le16 seq_ctl; | ||
633 | u8 payload[0]; | ||
634 | } __packed; | ||
635 | |||
636 | struct rtl_info_element { | ||
637 | u8 id; | ||
638 | u8 len; | ||
639 | u8 data[0]; | ||
640 | } __packed; | ||
641 | |||
642 | struct rtl_probe_rsp { | ||
643 | struct rtl_hdr_3addr header; | ||
644 | u32 time_stamp[2]; | ||
645 | __le16 beacon_interval; | ||
646 | __le16 capability; | ||
647 | /*SSID, supported rates, FH params, DS params, | ||
648 | CF params, IBSS params, TIM (if beacon), RSN */ | ||
649 | struct rtl_info_element info_element[0]; | ||
650 | } __packed; | ||
651 | |||
652 | /*LED related.*/ | ||
653 | /*ledpin Identify how to implement this SW led.*/ | ||
654 | struct rtl_led { | ||
655 | void *hw; | ||
656 | enum rtl_led_pin ledpin; | ||
657 | bool ledon; | ||
658 | }; | ||
659 | |||
660 | struct rtl_led_ctl { | ||
661 | bool led_opendrain; | ||
662 | struct rtl_led sw_led0; | ||
663 | struct rtl_led sw_led1; | ||
664 | }; | ||
665 | |||
666 | struct rtl_qos_parameters { | ||
667 | __le16 cw_min; | ||
668 | __le16 cw_max; | ||
669 | u8 aifs; | ||
670 | u8 flag; | ||
671 | __le16 tx_op; | ||
672 | } __packed; | ||
673 | |||
674 | struct rt_smooth_data { | ||
675 | u32 elements[100]; /*array to store values */ | ||
676 | u32 index; /*index to current array to store */ | ||
677 | u32 total_num; /*num of valid elements */ | ||
678 | u32 total_val; /*sum of valid elements */ | ||
679 | }; | ||
680 | |||
681 | struct false_alarm_statistics { | ||
682 | u32 cnt_parity_fail; | ||
683 | u32 cnt_rate_illegal; | ||
684 | u32 cnt_crc8_fail; | ||
685 | u32 cnt_mcs_fail; | ||
686 | u32 cnt_fast_fsync_fail; | ||
687 | u32 cnt_sb_search_fail; | ||
688 | u32 cnt_ofdm_fail; | ||
689 | u32 cnt_cck_fail; | ||
690 | u32 cnt_all; | ||
691 | }; | ||
692 | |||
693 | struct init_gain { | ||
694 | u8 xaagccore1; | ||
695 | u8 xbagccore1; | ||
696 | u8 xcagccore1; | ||
697 | u8 xdagccore1; | ||
698 | u8 cca; | ||
699 | |||
700 | }; | ||
701 | |||
702 | struct wireless_stats { | ||
703 | unsigned long txbytesunicast; | ||
704 | unsigned long txbytesmulticast; | ||
705 | unsigned long txbytesbroadcast; | ||
706 | unsigned long rxbytesunicast; | ||
707 | |||
708 | long rx_snr_db[4]; | ||
709 | /*Correct smoothed ss in Dbm, only used | ||
710 | in driver to report real power now. */ | ||
711 | long recv_signal_power; | ||
712 | long signal_quality; | ||
713 | long last_sigstrength_inpercent; | ||
714 | |||
715 | u32 rssi_calculate_cnt; | ||
716 | |||
717 | /*Transformed, in dbm. Beautified signal | ||
718 | strength for UI, not correct. */ | ||
719 | long signal_strength; | ||
720 | |||
721 | u8 rx_rssi_percentage[4]; | ||
722 | u8 rx_evm_percentage[2]; | ||
723 | |||
724 | struct rt_smooth_data ui_rssi; | ||
725 | struct rt_smooth_data ui_link_quality; | ||
726 | }; | ||
727 | |||
728 | struct rate_adaptive { | ||
729 | u8 rate_adaptive_disabled; | ||
730 | u8 ratr_state; | ||
731 | u16 reserve; | ||
732 | |||
733 | u32 high_rssi_thresh_for_ra; | ||
734 | u32 high2low_rssi_thresh_for_ra; | ||
735 | u8 low2high_rssi_thresh_for_ra40m; | ||
736 | u32 low_rssi_thresh_for_ra40M; | ||
737 | u8 low2high_rssi_thresh_for_ra20m; | ||
738 | u32 low_rssi_thresh_for_ra20M; | ||
739 | u32 upper_rssi_threshold_ratr; | ||
740 | u32 middleupper_rssi_threshold_ratr; | ||
741 | u32 middle_rssi_threshold_ratr; | ||
742 | u32 middlelow_rssi_threshold_ratr; | ||
743 | u32 low_rssi_threshold_ratr; | ||
744 | u32 ultralow_rssi_threshold_ratr; | ||
745 | u32 low_rssi_threshold_ratr_40m; | ||
746 | u32 low_rssi_threshold_ratr_20m; | ||
747 | u8 ping_rssi_enable; | ||
748 | u32 ping_rssi_ratr; | ||
749 | u32 ping_rssi_thresh_for_ra; | ||
750 | u32 last_ratr; | ||
751 | u8 pre_ratr_state; | ||
752 | }; | ||
753 | |||
754 | struct regd_pair_mapping { | ||
755 | u16 reg_dmnenum; | ||
756 | u16 reg_5ghz_ctl; | ||
757 | u16 reg_2ghz_ctl; | ||
758 | }; | ||
759 | |||
760 | struct rtl_regulatory { | ||
761 | char alpha2[2]; | ||
762 | u16 country_code; | ||
763 | u16 max_power_level; | ||
764 | u32 tp_scale; | ||
765 | u16 current_rd; | ||
766 | u16 current_rd_ext; | ||
767 | int16_t power_limit; | ||
768 | struct regd_pair_mapping *regpair; | ||
769 | }; | ||
770 | |||
771 | struct rtl_rfkill { | ||
772 | bool rfkill_state; /*0 is off, 1 is on */ | ||
773 | }; | ||
774 | |||
775 | #define IQK_MATRIX_REG_NUM 8 | ||
776 | #define IQK_MATRIX_SETTINGS_NUM (1 + 24 + 21) | ||
777 | struct iqk_matrix_regs { | ||
778 | bool iqk_done; | ||
779 | long value[1][IQK_MATRIX_REG_NUM]; | ||
780 | }; | ||
781 | |||
782 | struct phy_parameters { | ||
783 | u16 length; | ||
784 | u32 *pdata; | ||
785 | }; | ||
786 | |||
787 | enum hw_param_tab_index { | ||
788 | PHY_REG_2T, | ||
789 | PHY_REG_1T, | ||
790 | PHY_REG_PG, | ||
791 | RADIOA_2T, | ||
792 | RADIOB_2T, | ||
793 | RADIOA_1T, | ||
794 | RADIOB_1T, | ||
795 | MAC_REG, | ||
796 | AGCTAB_2T, | ||
797 | AGCTAB_1T, | ||
798 | MAX_TAB | ||
799 | }; | ||
800 | |||
801 | struct rtl_phy { | ||
802 | struct bb_reg_def phyreg_def[4]; /*Radio A/B/C/D */ | ||
803 | struct init_gain initgain_backup; | ||
804 | enum io_type current_io_type; | ||
805 | |||
806 | u8 rf_mode; | ||
807 | u8 rf_type; | ||
808 | u8 current_chan_bw; | ||
809 | u8 set_bwmode_inprogress; | ||
810 | u8 sw_chnl_inprogress; | ||
811 | u8 sw_chnl_stage; | ||
812 | u8 sw_chnl_step; | ||
813 | u8 current_channel; | ||
814 | u8 h2c_box_num; | ||
815 | u8 set_io_inprogress; | ||
816 | u8 lck_inprogress; | ||
817 | |||
818 | /* record for power tracking */ | ||
819 | s32 reg_e94; | ||
820 | s32 reg_e9c; | ||
821 | s32 reg_ea4; | ||
822 | s32 reg_eac; | ||
823 | s32 reg_eb4; | ||
824 | s32 reg_ebc; | ||
825 | s32 reg_ec4; | ||
826 | s32 reg_ecc; | ||
827 | u8 rfpienable; | ||
828 | u8 reserve_0; | ||
829 | u16 reserve_1; | ||
830 | u32 reg_c04, reg_c08, reg_874; | ||
831 | u32 adda_backup[16]; | ||
832 | u32 iqk_mac_backup[IQK_MAC_REG_NUM]; | ||
833 | u32 iqk_bb_backup[10]; | ||
834 | |||
835 | /* Dual mac */ | ||
836 | bool need_iqk; | ||
837 | struct iqk_matrix_regs iqk_matrix_regsetting[IQK_MATRIX_SETTINGS_NUM]; | ||
838 | |||
839 | bool rfpi_enable; | ||
840 | |||
841 | u8 pwrgroup_cnt; | ||
842 | u8 cck_high_power; | ||
843 | /* MAX_PG_GROUP groups of pwr diff by rates */ | ||
844 | u32 mcs_txpwrlevel_origoffset[MAX_PG_GROUP][16]; | ||
845 | u8 default_initialgain[4]; | ||
846 | |||
847 | /* the current Tx power level */ | ||
848 | u8 cur_cck_txpwridx; | ||
849 | u8 cur_ofdm24g_txpwridx; | ||
850 | |||
851 | u32 rfreg_chnlval[2]; | ||
852 | bool apk_done; | ||
853 | u32 reg_rf3c[2]; /* pathA / pathB */ | ||
854 | |||
855 | /* bfsync */ | ||
856 | u8 framesync; | ||
857 | u32 framesync_c34; | ||
858 | |||
859 | u8 num_total_rfpath; | ||
860 | struct phy_parameters hwparam_tables[MAX_TAB]; | ||
861 | u16 rf_pathmap; | ||
862 | }; | ||
863 | |||
864 | #define MAX_TID_COUNT 9 | ||
865 | #define RTL_AGG_STOP 0 | ||
866 | #define RTL_AGG_PROGRESS 1 | ||
867 | #define RTL_AGG_START 2 | ||
868 | #define RTL_AGG_OPERATIONAL 3 | ||
869 | #define RTL_AGG_OFF 0 | ||
870 | #define RTL_AGG_ON 1 | ||
871 | #define RTL_AGG_EMPTYING_HW_QUEUE_ADDBA 2 | ||
872 | #define RTL_AGG_EMPTYING_HW_QUEUE_DELBA 3 | ||
873 | |||
874 | struct rtl_ht_agg { | ||
875 | u16 txq_id; | ||
876 | u16 wait_for_ba; | ||
877 | u16 start_idx; | ||
878 | u64 bitmap; | ||
879 | u32 rate_n_flags; | ||
880 | u8 agg_state; | ||
881 | }; | ||
882 | |||
883 | struct rtl_tid_data { | ||
884 | u16 seq_number; | ||
885 | struct rtl_ht_agg agg; | ||
886 | }; | ||
887 | |||
888 | struct rtl_sta_info { | ||
889 | u8 ratr_index; | ||
890 | u8 wireless_mode; | ||
891 | u8 mimo_ps; | ||
892 | struct rtl_tid_data tids[MAX_TID_COUNT]; | ||
893 | } __packed; | ||
894 | |||
895 | struct rtl_priv; | ||
896 | struct rtl_io { | ||
897 | struct device *dev; | ||
898 | struct mutex bb_mutex; | ||
899 | |||
900 | /*PCI MEM map */ | ||
901 | unsigned long pci_mem_end; /*shared mem end */ | ||
902 | unsigned long pci_mem_start; /*shared mem start */ | ||
903 | |||
904 | /*PCI IO map */ | ||
905 | unsigned long pci_base_addr; /*device I/O address */ | ||
906 | |||
907 | void (*write8_async) (struct rtl_priv *rtlpriv, u32 addr, u8 val); | ||
908 | void (*write16_async) (struct rtl_priv *rtlpriv, u32 addr, u16 val); | ||
909 | void (*write32_async) (struct rtl_priv *rtlpriv, u32 addr, u32 val); | ||
910 | int (*writeN_async) (struct rtl_priv *rtlpriv, u32 addr, u16 len, | ||
911 | u8 *pdata); | ||
912 | |||
913 | u8(*read8_sync) (struct rtl_priv *rtlpriv, u32 addr); | ||
914 | u16(*read16_sync) (struct rtl_priv *rtlpriv, u32 addr); | ||
915 | u32(*read32_sync) (struct rtl_priv *rtlpriv, u32 addr); | ||
916 | int (*readN_sync) (struct rtl_priv *rtlpriv, u32 addr, u16 len, | ||
917 | u8 *pdata); | ||
918 | |||
919 | }; | ||
920 | |||
921 | struct rtl_mac { | ||
922 | u8 mac_addr[ETH_ALEN]; | ||
923 | u8 mac80211_registered; | ||
924 | u8 beacon_enabled; | ||
925 | |||
926 | u32 tx_ss_num; | ||
927 | u32 rx_ss_num; | ||
928 | |||
929 | struct ieee80211_supported_band bands[IEEE80211_NUM_BANDS]; | ||
930 | struct ieee80211_hw *hw; | ||
931 | struct ieee80211_vif *vif; | ||
932 | enum nl80211_iftype opmode; | ||
933 | |||
934 | /*Probe Beacon management */ | ||
935 | struct rtl_tid_data tids[MAX_TID_COUNT]; | ||
936 | enum rtl_link_state link_state; | ||
937 | |||
938 | int n_channels; | ||
939 | int n_bitrates; | ||
940 | |||
941 | bool offchan_deley; | ||
942 | |||
943 | /*filters */ | ||
944 | u32 rx_conf; | ||
945 | u16 rx_mgt_filter; | ||
946 | u16 rx_ctrl_filter; | ||
947 | u16 rx_data_filter; | ||
948 | |||
949 | bool act_scanning; | ||
950 | u8 cnt_after_linked; | ||
951 | |||
952 | /* early mode */ | ||
953 | /* skb wait queue */ | ||
954 | struct sk_buff_head skb_waitq[MAX_TID_COUNT]; | ||
955 | u8 earlymode_threshold; | ||
956 | |||
957 | /*RDG*/ | ||
958 | bool rdg_en; | ||
959 | |||
960 | /*AP*/ | ||
961 | u8 bssid[6]; | ||
962 | u32 vendor; | ||
963 | u8 mcs[16]; /* 16 bytes mcs for HT rates. */ | ||
964 | u32 basic_rates; /* b/g rates */ | ||
965 | u8 ht_enable; | ||
966 | u8 sgi_40; | ||
967 | u8 sgi_20; | ||
968 | u8 bw_40; | ||
969 | u8 mode; /* wireless mode */ | ||
970 | u8 slot_time; | ||
971 | u8 short_preamble; | ||
972 | u8 use_cts_protect; | ||
973 | u8 cur_40_prime_sc; | ||
974 | u8 cur_40_prime_sc_bk; | ||
975 | u64 tsf; | ||
976 | u8 retry_short; | ||
977 | u8 retry_long; | ||
978 | u16 assoc_id; | ||
979 | |||
980 | /*IBSS*/ | ||
981 | int beacon_interval; | ||
982 | |||
983 | /*AMPDU*/ | ||
984 | u8 min_space_cfg; /*For Min spacing configurations */ | ||
985 | u8 max_mss_density; | ||
986 | u8 current_ampdu_factor; | ||
987 | u8 current_ampdu_density; | ||
988 | |||
989 | /*QOS & EDCA */ | ||
990 | struct ieee80211_tx_queue_params edca_param[RTL_MAC80211_NUM_QUEUE]; | ||
991 | struct rtl_qos_parameters ac[AC_MAX]; | ||
992 | }; | ||
993 | |||
994 | struct rtl_hal { | ||
995 | struct ieee80211_hw *hw; | ||
996 | |||
997 | enum intf_type interface; | ||
998 | u16 hw_type; /*92c or 92d or 92s and so on */ | ||
999 | u8 ic_class; | ||
1000 | u8 oem_id; | ||
1001 | u32 version; /*version of chip */ | ||
1002 | u8 state; /*stop 0, start 1 */ | ||
1003 | |||
1004 | /*firmware */ | ||
1005 | u32 fwsize; | ||
1006 | u8 *pfirmware; | ||
1007 | u16 fw_version; | ||
1008 | u16 fw_subversion; | ||
1009 | bool h2c_setinprogress; | ||
1010 | u8 last_hmeboxnum; | ||
1011 | bool fw_ready; | ||
1012 | /*Reserve page start offset except beacon in TxQ. */ | ||
1013 | u8 fw_rsvdpage_startoffset; | ||
1014 | u8 h2c_txcmd_seq; | ||
1015 | |||
1016 | /* FW Cmd IO related */ | ||
1017 | u16 fwcmd_iomap; | ||
1018 | u32 fwcmd_ioparam; | ||
1019 | bool set_fwcmd_inprogress; | ||
1020 | u8 current_fwcmd_io; | ||
1021 | |||
1022 | /**/ | ||
1023 | bool driver_going2unload; | ||
1024 | |||
1025 | /*AMPDU init min space*/ | ||
1026 | u8 minspace_cfg; /*For Min spacing configurations */ | ||
1027 | |||
1028 | /* Dual mac */ | ||
1029 | enum macphy_mode macphymode; | ||
1030 | enum band_type current_bandtype; /* 0:2.4G, 1:5G */ | ||
1031 | enum band_type current_bandtypebackup; | ||
1032 | enum band_type bandset; | ||
1033 | /* dual MAC 0--Mac0 1--Mac1 */ | ||
1034 | u32 interfaceindex; | ||
1035 | /* just for DualMac S3S4 */ | ||
1036 | u8 macphyctl_reg; | ||
1037 | bool earlymode_enable; | ||
1038 | /* Dual mac*/ | ||
1039 | bool during_mac0init_radiob; | ||
1040 | bool during_mac1init_radioa; | ||
1041 | bool reloadtxpowerindex; | ||
1042 | /* True if IMR or IQK have done | ||
1043 | for 2.4G in scan progress */ | ||
1044 | bool load_imrandiqk_setting_for2g; | ||
1045 | |||
1046 | bool disable_amsdu_8k; | ||
1047 | }; | ||
1048 | |||
1049 | struct rtl_security { | ||
1050 | /*default 0 */ | ||
1051 | bool use_sw_sec; | ||
1052 | |||
1053 | bool being_setkey; | ||
1054 | bool use_defaultkey; | ||
1055 | /*Encryption Algorithm for Unicast Packet */ | ||
1056 | enum rt_enc_alg pairwise_enc_algorithm; | ||
1057 | /*Encryption Algorithm for Brocast/Multicast */ | ||
1058 | enum rt_enc_alg group_enc_algorithm; | ||
1059 | /*Cam Entry Bitmap */ | ||
1060 | u32 hwsec_cam_bitmap; | ||
1061 | u8 hwsec_cam_sta_addr[TOTAL_CAM_ENTRY][ETH_ALEN]; | ||
1062 | /*local Key buffer, indx 0 is for | ||
1063 | pairwise key 1-4 is for agoup key. */ | ||
1064 | u8 key_buf[KEY_BUF_SIZE][MAX_KEY_LEN]; | ||
1065 | u8 key_len[KEY_BUF_SIZE]; | ||
1066 | |||
1067 | /*The pointer of Pairwise Key, | ||
1068 | it always points to KeyBuf[4] */ | ||
1069 | u8 *pairwise_key; | ||
1070 | }; | ||
1071 | |||
1072 | struct rtl_dm { | ||
1073 | /*PHY status for Dynamic Management */ | ||
1074 | long entry_min_undecoratedsmoothed_pwdb; | ||
1075 | long undecorated_smoothed_pwdb; /*out dm */ | ||
1076 | long entry_max_undecoratedsmoothed_pwdb; | ||
1077 | bool dm_initialgain_enable; | ||
1078 | bool dynamic_txpower_enable; | ||
1079 | bool current_turbo_edca; | ||
1080 | bool is_any_nonbepkts; /*out dm */ | ||
1081 | bool is_cur_rdlstate; | ||
1082 | bool txpower_trackinginit; | ||
1083 | bool disable_framebursting; | ||
1084 | bool cck_inch14; | ||
1085 | bool txpower_tracking; | ||
1086 | bool useramask; | ||
1087 | bool rfpath_rxenable[4]; | ||
1088 | bool inform_fw_driverctrldm; | ||
1089 | bool current_mrc_switch; | ||
1090 | u8 txpowercount; | ||
1091 | |||
1092 | u8 thermalvalue_rxgain; | ||
1093 | u8 thermalvalue_iqk; | ||
1094 | u8 thermalvalue_lck; | ||
1095 | u8 thermalvalue; | ||
1096 | u8 last_dtp_lvl; | ||
1097 | u8 thermalvalue_avg[AVG_THERMAL_NUM]; | ||
1098 | u8 thermalvalue_avg_index; | ||
1099 | bool done_txpower; | ||
1100 | u8 dynamic_txhighpower_lvl; /*Tx high power level */ | ||
1101 | u8 dm_flag; /*Indicate each dynamic mechanism's status. */ | ||
1102 | u8 dm_type; | ||
1103 | u8 txpower_track_control; | ||
1104 | bool interrupt_migration; | ||
1105 | bool disable_tx_int; | ||
1106 | char ofdm_index[2]; | ||
1107 | char cck_index; | ||
1108 | }; | ||
1109 | |||
1110 | #define EFUSE_MAX_LOGICAL_SIZE 256 | ||
1111 | |||
1112 | struct rtl_efuse { | ||
1113 | bool autoLoad_ok; | ||
1114 | bool bootfromefuse; | ||
1115 | u16 max_physical_size; | ||
1116 | |||
1117 | u8 efuse_map[2][EFUSE_MAX_LOGICAL_SIZE]; | ||
1118 | u16 efuse_usedbytes; | ||
1119 | u8 efuse_usedpercentage; | ||
1120 | #ifdef EFUSE_REPG_WORKAROUND | ||
1121 | bool efuse_re_pg_sec1flag; | ||
1122 | u8 efuse_re_pg_data[8]; | ||
1123 | #endif | ||
1124 | |||
1125 | u8 autoload_failflag; | ||
1126 | u8 autoload_status; | ||
1127 | |||
1128 | short epromtype; | ||
1129 | u16 eeprom_vid; | ||
1130 | u16 eeprom_did; | ||
1131 | u16 eeprom_svid; | ||
1132 | u16 eeprom_smid; | ||
1133 | u8 eeprom_oemid; | ||
1134 | u16 eeprom_channelplan; | ||
1135 | u8 eeprom_version; | ||
1136 | u8 board_type; | ||
1137 | u8 external_pa; | ||
1138 | |||
1139 | u8 dev_addr[6]; | ||
1140 | |||
1141 | bool txpwr_fromeprom; | ||
1142 | u8 eeprom_crystalcap; | ||
1143 | u8 eeprom_tssi[2]; | ||
1144 | u8 eeprom_tssi_5g[3][2]; /* for 5GL/5GM/5GH band. */ | ||
1145 | u8 eeprom_pwrlimit_ht20[CHANNEL_GROUP_MAX]; | ||
1146 | u8 eeprom_pwrlimit_ht40[CHANNEL_GROUP_MAX]; | ||
1147 | u8 eeprom_chnlarea_txpwr_cck[2][CHANNEL_GROUP_MAX_2G]; | ||
1148 | u8 eeprom_chnlarea_txpwr_ht40_1s[2][CHANNEL_GROUP_MAX]; | ||
1149 | u8 eeprom_chnlarea_txpwr_ht40_2sdiif[2][CHANNEL_GROUP_MAX]; | ||
1150 | u8 txpwrlevel_cck[2][CHANNEL_MAX_NUMBER_2G]; | ||
1151 | u8 txpwrlevel_ht40_1s[2][CHANNEL_MAX_NUMBER]; /*For HT 40MHZ pwr */ | ||
1152 | u8 txpwrlevel_ht40_2s[2][CHANNEL_MAX_NUMBER]; /*For HT 40MHZ pwr */ | ||
1153 | |||
1154 | u8 internal_pa_5g[2]; /* pathA / pathB */ | ||
1155 | u8 eeprom_c9; | ||
1156 | u8 eeprom_cc; | ||
1157 | |||
1158 | /*For power group */ | ||
1159 | u8 eeprom_pwrgroup[2][3]; | ||
1160 | u8 pwrgroup_ht20[2][CHANNEL_MAX_NUMBER]; | ||
1161 | u8 pwrgroup_ht40[2][CHANNEL_MAX_NUMBER]; | ||
1162 | |||
1163 | char txpwr_ht20diff[2][CHANNEL_MAX_NUMBER]; /*HT 20<->40 Pwr diff */ | ||
1164 | /*For HT<->legacy pwr diff*/ | ||
1165 | u8 txpwr_legacyhtdiff[2][CHANNEL_MAX_NUMBER]; | ||
1166 | u8 txpwr_safetyflag; /* Band edge enable flag */ | ||
1167 | u16 eeprom_txpowerdiff; | ||
1168 | u8 legacy_httxpowerdiff; /* Legacy to HT rate power diff */ | ||
1169 | u8 antenna_txpwdiff[3]; | ||
1170 | |||
1171 | u8 eeprom_regulatory; | ||
1172 | u8 eeprom_thermalmeter; | ||
1173 | u8 thermalmeter[2]; /*ThermalMeter, index 0 for RFIC0, 1 for RFIC1 */ | ||
1174 | u16 tssi_13dbm; | ||
1175 | u8 crystalcap; /* CrystalCap. */ | ||
1176 | u8 delta_iqk; | ||
1177 | u8 delta_lck; | ||
1178 | |||
1179 | u8 legacy_ht_txpowerdiff; /*Legacy to HT rate power diff */ | ||
1180 | bool apk_thermalmeterignore; | ||
1181 | |||
1182 | bool b1x1_recvcombine; | ||
1183 | bool b1ss_support; | ||
1184 | |||
1185 | /*channel plan */ | ||
1186 | u8 channel_plan; | ||
1187 | }; | ||
1188 | |||
1189 | struct rtl_ps_ctl { | ||
1190 | bool pwrdomain_protect; | ||
1191 | bool set_rfpowerstate_inprogress; | ||
1192 | bool in_powersavemode; | ||
1193 | bool rfchange_inprogress; | ||
1194 | bool swrf_processing; | ||
1195 | bool hwradiooff; | ||
1196 | |||
1197 | /* | ||
1198 | * just for PCIE ASPM | ||
1199 | * If it supports ASPM, Offset[560h] = 0x40, | ||
1200 | * otherwise Offset[560h] = 0x00. | ||
1201 | * */ | ||
1202 | bool support_aspm; | ||
1203 | |||
1204 | bool support_backdoor; | ||
1205 | |||
1206 | /*for LPS */ | ||
1207 | enum rt_psmode dot11_psmode; /*Power save mode configured. */ | ||
1208 | bool swctrl_lps; | ||
1209 | bool leisure_ps; | ||
1210 | bool fwctrl_lps; | ||
1211 | u8 fwctrl_psmode; | ||
1212 | /*For Fw control LPS mode */ | ||
1213 | u8 reg_fwctrl_lps; | ||
1214 | /*Record Fw PS mode status. */ | ||
1215 | bool fw_current_inpsmode; | ||
1216 | u8 reg_max_lps_awakeintvl; | ||
1217 | bool report_linked; | ||
1218 | |||
1219 | /*for IPS */ | ||
1220 | bool inactiveps; | ||
1221 | |||
1222 | u32 rfoff_reason; | ||
1223 | |||
1224 | /*RF OFF Level */ | ||
1225 | u32 cur_ps_level; | ||
1226 | u32 reg_rfps_level; | ||
1227 | |||
1228 | /*just for PCIE ASPM */ | ||
1229 | u8 const_amdpci_aspm; | ||
1230 | bool pwrdown_mode; | ||
1231 | |||
1232 | enum rf_pwrstate inactive_pwrstate; | ||
1233 | enum rf_pwrstate rfpwr_state; /*cur power state */ | ||
1234 | |||
1235 | /* for SW LPS*/ | ||
1236 | bool sw_ps_enabled; | ||
1237 | bool state; | ||
1238 | bool state_inap; | ||
1239 | bool multi_buffered; | ||
1240 | u16 nullfunc_seq; | ||
1241 | unsigned int dtim_counter; | ||
1242 | unsigned int sleep_ms; | ||
1243 | unsigned long last_sleep_jiffies; | ||
1244 | unsigned long last_awake_jiffies; | ||
1245 | unsigned long last_delaylps_stamp_jiffies; | ||
1246 | unsigned long last_dtim; | ||
1247 | unsigned long last_beacon; | ||
1248 | unsigned long last_action; | ||
1249 | unsigned long last_slept; | ||
1250 | }; | ||
1251 | |||
1252 | struct rtl_stats { | ||
1253 | u32 mac_time[2]; | ||
1254 | s8 rssi; | ||
1255 | u8 signal; | ||
1256 | u8 noise; | ||
1257 | u16 rate; /*in 100 kbps */ | ||
1258 | u8 received_channel; | ||
1259 | u8 control; | ||
1260 | u8 mask; | ||
1261 | u8 freq; | ||
1262 | u16 len; | ||
1263 | u64 tsf; | ||
1264 | u32 beacon_time; | ||
1265 | u8 nic_type; | ||
1266 | u16 length; | ||
1267 | u8 signalquality; /*in 0-100 index. */ | ||
1268 | /* | ||
1269 | * Real power in dBm for this packet, | ||
1270 | * no beautification and aggregation. | ||
1271 | * */ | ||
1272 | s32 recvsignalpower; | ||
1273 | s8 rxpower; /*in dBm Translate from PWdB */ | ||
1274 | u8 signalstrength; /*in 0-100 index. */ | ||
1275 | u16 hwerror:1; | ||
1276 | u16 crc:1; | ||
1277 | u16 icv:1; | ||
1278 | u16 shortpreamble:1; | ||
1279 | u16 antenna:1; | ||
1280 | u16 decrypted:1; | ||
1281 | u16 wakeup:1; | ||
1282 | u32 timestamp_low; | ||
1283 | u32 timestamp_high; | ||
1284 | |||
1285 | u8 rx_drvinfo_size; | ||
1286 | u8 rx_bufshift; | ||
1287 | bool isampdu; | ||
1288 | bool isfirst_ampdu; | ||
1289 | bool rx_is40Mhzpacket; | ||
1290 | u32 rx_pwdb_all; | ||
1291 | u8 rx_mimo_signalstrength[4]; /*in 0~100 index */ | ||
1292 | s8 rx_mimo_signalquality[2]; | ||
1293 | bool packet_matchbssid; | ||
1294 | bool is_cck; | ||
1295 | bool packet_toself; | ||
1296 | bool packet_beacon; /*for rssi */ | ||
1297 | char cck_adc_pwdb[4]; /*for rx path selection */ | ||
1298 | }; | ||
1299 | |||
1300 | struct rt_link_detect { | ||
1301 | u32 num_tx_in4period[4]; | ||
1302 | u32 num_rx_in4period[4]; | ||
1303 | |||
1304 | u32 num_tx_inperiod; | ||
1305 | u32 num_rx_inperiod; | ||
1306 | |||
1307 | bool busytraffic; | ||
1308 | bool higher_busytraffic; | ||
1309 | bool higher_busyrxtraffic; | ||
1310 | |||
1311 | u32 tidtx_in4period[MAX_TID_COUNT][4]; | ||
1312 | u32 tidtx_inperiod[MAX_TID_COUNT]; | ||
1313 | bool higher_busytxtraffic[MAX_TID_COUNT]; | ||
1314 | }; | ||
1315 | |||
1316 | struct rtl_tcb_desc { | ||
1317 | u8 packet_bw:1; | ||
1318 | u8 multicast:1; | ||
1319 | u8 broadcast:1; | ||
1320 | |||
1321 | u8 rts_stbc:1; | ||
1322 | u8 rts_enable:1; | ||
1323 | u8 cts_enable:1; | ||
1324 | u8 rts_use_shortpreamble:1; | ||
1325 | u8 rts_use_shortgi:1; | ||
1326 | u8 rts_sc:1; | ||
1327 | u8 rts_bw:1; | ||
1328 | u8 rts_rate; | ||
1329 | |||
1330 | u8 use_shortgi:1; | ||
1331 | u8 use_shortpreamble:1; | ||
1332 | u8 use_driver_rate:1; | ||
1333 | u8 disable_ratefallback:1; | ||
1334 | |||
1335 | u8 ratr_index; | ||
1336 | u8 mac_id; | ||
1337 | u8 hw_rate; | ||
1338 | |||
1339 | u8 last_inipkt:1; | ||
1340 | u8 cmd_or_init:1; | ||
1341 | u8 queue_index; | ||
1342 | |||
1343 | /* early mode */ | ||
1344 | u8 empkt_num; | ||
1345 | /* The max value by HW */ | ||
1346 | u32 empkt_len[5]; | ||
1347 | }; | ||
1348 | |||
1349 | struct rtl_hal_ops { | ||
1350 | int (*init_sw_vars) (struct ieee80211_hw *hw); | ||
1351 | void (*deinit_sw_vars) (struct ieee80211_hw *hw); | ||
1352 | void (*read_chip_version)(struct ieee80211_hw *hw); | ||
1353 | void (*read_eeprom_info) (struct ieee80211_hw *hw); | ||
1354 | void (*interrupt_recognized) (struct ieee80211_hw *hw, | ||
1355 | u32 *p_inta, u32 *p_intb); | ||
1356 | int (*hw_init) (struct ieee80211_hw *hw); | ||
1357 | void (*hw_disable) (struct ieee80211_hw *hw); | ||
1358 | void (*hw_suspend) (struct ieee80211_hw *hw); | ||
1359 | void (*hw_resume) (struct ieee80211_hw *hw); | ||
1360 | void (*enable_interrupt) (struct ieee80211_hw *hw); | ||
1361 | void (*disable_interrupt) (struct ieee80211_hw *hw); | ||
1362 | int (*set_network_type) (struct ieee80211_hw *hw, | ||
1363 | enum nl80211_iftype type); | ||
1364 | void (*set_chk_bssid)(struct ieee80211_hw *hw, | ||
1365 | bool check_bssid); | ||
1366 | void (*set_bw_mode) (struct ieee80211_hw *hw, | ||
1367 | enum nl80211_channel_type ch_type); | ||
1368 | u8(*switch_channel) (struct ieee80211_hw *hw); | ||
1369 | void (*set_qos) (struct ieee80211_hw *hw, int aci); | ||
1370 | void (*set_bcn_reg) (struct ieee80211_hw *hw); | ||
1371 | void (*set_bcn_intv) (struct ieee80211_hw *hw); | ||
1372 | void (*update_interrupt_mask) (struct ieee80211_hw *hw, | ||
1373 | u32 add_msr, u32 rm_msr); | ||
1374 | void (*get_hw_reg) (struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
1375 | void (*set_hw_reg) (struct ieee80211_hw *hw, u8 variable, u8 *val); | ||
1376 | void (*update_rate_tbl) (struct ieee80211_hw *hw, | ||
1377 | struct ieee80211_sta *sta, u8 rssi_level); | ||
1378 | void (*update_rate_mask) (struct ieee80211_hw *hw, u8 rssi_level); | ||
1379 | void (*fill_tx_desc) (struct ieee80211_hw *hw, | ||
1380 | struct ieee80211_hdr *hdr, u8 *pdesc_tx, | ||
1381 | struct ieee80211_tx_info *info, | ||
1382 | struct sk_buff *skb, u8 hw_queue, | ||
1383 | struct rtl_tcb_desc *ptcb_desc); | ||
1384 | void (*fill_fake_txdesc) (struct ieee80211_hw *hw, u8 *pDesc, | ||
1385 | u32 buffer_len, bool bIsPsPoll); | ||
1386 | void (*fill_tx_cmddesc) (struct ieee80211_hw *hw, u8 *pdesc, | ||
1387 | bool firstseg, bool lastseg, | ||
1388 | struct sk_buff *skb); | ||
1389 | bool (*cmd_send_packet)(struct ieee80211_hw *hw, struct sk_buff *skb); | ||
1390 | bool (*query_rx_desc) (struct ieee80211_hw *hw, | ||
1391 | struct rtl_stats *stats, | ||
1392 | struct ieee80211_rx_status *rx_status, | ||
1393 | u8 *pdesc, struct sk_buff *skb); | ||
1394 | void (*set_channel_access) (struct ieee80211_hw *hw); | ||
1395 | bool (*radio_onoff_checking) (struct ieee80211_hw *hw, u8 *valid); | ||
1396 | void (*dm_watchdog) (struct ieee80211_hw *hw); | ||
1397 | void (*scan_operation_backup) (struct ieee80211_hw *hw, u8 operation); | ||
1398 | bool (*set_rf_power_state) (struct ieee80211_hw *hw, | ||
1399 | enum rf_pwrstate rfpwr_state); | ||
1400 | void (*led_control) (struct ieee80211_hw *hw, | ||
1401 | enum led_ctl_mode ledaction); | ||
1402 | void (*set_desc) (u8 *pdesc, bool istx, u8 desc_name, u8 *val); | ||
1403 | u32 (*get_desc) (u8 *pdesc, bool istx, u8 desc_name); | ||
1404 | void (*tx_polling) (struct ieee80211_hw *hw, u8 hw_queue); | ||
1405 | void (*enable_hw_sec) (struct ieee80211_hw *hw); | ||
1406 | void (*set_key) (struct ieee80211_hw *hw, u32 key_index, | ||
1407 | u8 *macaddr, bool is_group, u8 enc_algo, | ||
1408 | bool is_wepkey, bool clear_all); | ||
1409 | void (*init_sw_leds) (struct ieee80211_hw *hw); | ||
1410 | void (*deinit_sw_leds) (struct ieee80211_hw *hw); | ||
1411 | u32 (*get_bbreg) (struct ieee80211_hw *hw, u32 regaddr, u32 bitmask); | ||
1412 | void (*set_bbreg) (struct ieee80211_hw *hw, u32 regaddr, u32 bitmask, | ||
1413 | u32 data); | ||
1414 | u32 (*get_rfreg) (struct ieee80211_hw *hw, enum radio_path rfpath, | ||
1415 | u32 regaddr, u32 bitmask); | ||
1416 | void (*set_rfreg) (struct ieee80211_hw *hw, enum radio_path rfpath, | ||
1417 | u32 regaddr, u32 bitmask, u32 data); | ||
1418 | void (*linked_set_reg) (struct ieee80211_hw *hw); | ||
1419 | bool (*phy_rf6052_config) (struct ieee80211_hw *hw); | ||
1420 | void (*phy_rf6052_set_cck_txpower) (struct ieee80211_hw *hw, | ||
1421 | u8 *powerlevel); | ||
1422 | void (*phy_rf6052_set_ofdm_txpower) (struct ieee80211_hw *hw, | ||
1423 | u8 *ppowerlevel, u8 channel); | ||
1424 | bool (*config_bb_with_headerfile) (struct ieee80211_hw *hw, | ||
1425 | u8 configtype); | ||
1426 | bool (*config_bb_with_pgheaderfile) (struct ieee80211_hw *hw, | ||
1427 | u8 configtype); | ||
1428 | void (*phy_lc_calibrate) (struct ieee80211_hw *hw, bool is2t); | ||
1429 | void (*phy_set_bw_mode_callback) (struct ieee80211_hw *hw); | ||
1430 | void (*dm_dynamic_txpower) (struct ieee80211_hw *hw); | ||
1431 | }; | ||
1432 | |||
1433 | struct rtl_intf_ops { | ||
1434 | /*com */ | ||
1435 | void (*read_efuse_byte)(struct ieee80211_hw *hw, u16 _offset, u8 *pbuf); | ||
1436 | int (*adapter_start) (struct ieee80211_hw *hw); | ||
1437 | void (*adapter_stop) (struct ieee80211_hw *hw); | ||
1438 | |||
1439 | int (*adapter_tx) (struct ieee80211_hw *hw, struct sk_buff *skb, | ||
1440 | struct rtl_tcb_desc *ptcb_desc); | ||
1441 | void (*flush)(struct ieee80211_hw *hw, bool drop); | ||
1442 | int (*reset_trx_ring) (struct ieee80211_hw *hw); | ||
1443 | bool (*waitq_insert) (struct ieee80211_hw *hw, struct sk_buff *skb); | ||
1444 | |||
1445 | /*pci */ | ||
1446 | void (*disable_aspm) (struct ieee80211_hw *hw); | ||
1447 | void (*enable_aspm) (struct ieee80211_hw *hw); | ||
1448 | |||
1449 | /*usb */ | ||
1450 | }; | ||
1451 | |||
1452 | struct rtl_mod_params { | ||
1453 | /* default: 0 = using hardware encryption */ | ||
1454 | int sw_crypto; | ||
1455 | |||
1456 | /* default: 1 = using no linked power save */ | ||
1457 | bool inactiveps; | ||
1458 | |||
1459 | /* default: 1 = using linked sw power save */ | ||
1460 | bool swctrl_lps; | ||
1461 | |||
1462 | /* default: 1 = using linked fw power save */ | ||
1463 | bool fwctrl_lps; | ||
1464 | }; | ||
1465 | |||
1466 | struct rtl_hal_usbint_cfg { | ||
1467 | /* data - rx */ | ||
1468 | u32 in_ep_num; | ||
1469 | u32 rx_urb_num; | ||
1470 | u32 rx_max_size; | ||
1471 | |||
1472 | /* op - rx */ | ||
1473 | void (*usb_rx_hdl)(struct ieee80211_hw *, struct sk_buff *); | ||
1474 | void (*usb_rx_segregate_hdl)(struct ieee80211_hw *, struct sk_buff *, | ||
1475 | struct sk_buff_head *); | ||
1476 | |||
1477 | /* tx */ | ||
1478 | void (*usb_tx_cleanup)(struct ieee80211_hw *, struct sk_buff *); | ||
1479 | int (*usb_tx_post_hdl)(struct ieee80211_hw *, struct urb *, | ||
1480 | struct sk_buff *); | ||
1481 | struct sk_buff *(*usb_tx_aggregate_hdl)(struct ieee80211_hw *, | ||
1482 | struct sk_buff_head *); | ||
1483 | |||
1484 | /* endpoint mapping */ | ||
1485 | int (*usb_endpoint_mapping)(struct ieee80211_hw *hw); | ||
1486 | u16 (*usb_mq_to_hwq)(__le16 fc, u16 mac80211_queue_index); | ||
1487 | }; | ||
1488 | |||
1489 | struct rtl_hal_cfg { | ||
1490 | u8 bar_id; | ||
1491 | bool write_readback; | ||
1492 | char *name; | ||
1493 | char *fw_name; | ||
1494 | struct rtl_hal_ops *ops; | ||
1495 | struct rtl_mod_params *mod_params; | ||
1496 | struct rtl_hal_usbint_cfg *usb_interface_cfg; | ||
1497 | |||
1498 | /*this map used for some registers or vars | ||
1499 | defined int HAL but used in MAIN */ | ||
1500 | u32 maps[RTL_VAR_MAP_MAX]; | ||
1501 | |||
1502 | }; | ||
1503 | |||
1504 | struct rtl_locks { | ||
1505 | /* mutex */ | ||
1506 | struct mutex conf_mutex; | ||
1507 | |||
1508 | /*spin lock */ | ||
1509 | spinlock_t ips_lock; | ||
1510 | spinlock_t irq_th_lock; | ||
1511 | spinlock_t h2c_lock; | ||
1512 | spinlock_t rf_ps_lock; | ||
1513 | spinlock_t rf_lock; | ||
1514 | spinlock_t lps_lock; | ||
1515 | spinlock_t waitq_lock; | ||
1516 | |||
1517 | /*Dual mac*/ | ||
1518 | spinlock_t cck_and_rw_pagea_lock; | ||
1519 | }; | ||
1520 | |||
1521 | struct rtl_works { | ||
1522 | struct ieee80211_hw *hw; | ||
1523 | |||
1524 | /*timer */ | ||
1525 | struct timer_list watchdog_timer; | ||
1526 | |||
1527 | /*task */ | ||
1528 | struct tasklet_struct irq_tasklet; | ||
1529 | struct tasklet_struct irq_prepare_bcn_tasklet; | ||
1530 | |||
1531 | /*work queue */ | ||
1532 | struct workqueue_struct *rtl_wq; | ||
1533 | struct delayed_work watchdog_wq; | ||
1534 | struct delayed_work ips_nic_off_wq; | ||
1535 | |||
1536 | /* For SW LPS */ | ||
1537 | struct delayed_work ps_work; | ||
1538 | struct delayed_work ps_rfon_wq; | ||
1539 | }; | ||
1540 | |||
1541 | struct rtl_debug { | ||
1542 | u32 dbgp_type[DBGP_TYPE_MAX]; | ||
1543 | u32 global_debuglevel; | ||
1544 | u64 global_debugcomponents; | ||
1545 | |||
1546 | /* add for proc debug */ | ||
1547 | struct proc_dir_entry *proc_dir; | ||
1548 | char proc_name[20]; | ||
1549 | }; | ||
1550 | |||
1551 | struct rtl_priv { | ||
1552 | struct rtl_locks locks; | ||
1553 | struct rtl_works works; | ||
1554 | struct rtl_mac mac80211; | ||
1555 | struct rtl_hal rtlhal; | ||
1556 | struct rtl_regulatory regd; | ||
1557 | struct rtl_rfkill rfkill; | ||
1558 | struct rtl_io io; | ||
1559 | struct rtl_phy phy; | ||
1560 | struct rtl_dm dm; | ||
1561 | struct rtl_security sec; | ||
1562 | struct rtl_efuse efuse; | ||
1563 | |||
1564 | struct rtl_ps_ctl psc; | ||
1565 | struct rate_adaptive ra; | ||
1566 | struct wireless_stats stats; | ||
1567 | struct rt_link_detect link_info; | ||
1568 | struct false_alarm_statistics falsealm_cnt; | ||
1569 | |||
1570 | struct rtl_rate_priv *rate_priv; | ||
1571 | |||
1572 | struct rtl_debug dbg; | ||
1573 | |||
1574 | /* | ||
1575 | *hal_cfg : for diff cards | ||
1576 | *intf_ops : for diff interrface usb/pcie | ||
1577 | */ | ||
1578 | struct rtl_hal_cfg *cfg; | ||
1579 | struct rtl_intf_ops *intf_ops; | ||
1580 | |||
1581 | /*this var will be set by set_bit, | ||
1582 | and was used to indicate status of | ||
1583 | interface or hardware */ | ||
1584 | unsigned long status; | ||
1585 | |||
1586 | /*This must be the last item so | ||
1587 | that it points to the data allocated | ||
1588 | beyond this structure like: | ||
1589 | rtl_pci_priv or rtl_usb_priv */ | ||
1590 | u8 priv[0]; | ||
1591 | }; | ||
1592 | |||
1593 | #define rtl_priv(hw) (((struct rtl_priv *)(hw)->priv)) | ||
1594 | #define rtl_mac(rtlpriv) (&((rtlpriv)->mac80211)) | ||
1595 | #define rtl_hal(rtlpriv) (&((rtlpriv)->rtlhal)) | ||
1596 | #define rtl_efuse(rtlpriv) (&((rtlpriv)->efuse)) | ||
1597 | #define rtl_psc(rtlpriv) (&((rtlpriv)->psc)) | ||
1598 | |||
1599 | |||
1600 | /*************************************** | ||
1601 | Bluetooth Co-existence Related | ||
1602 | ****************************************/ | ||
1603 | |||
1604 | enum bt_ant_num { | ||
1605 | ANT_X2 = 0, | ||
1606 | ANT_X1 = 1, | ||
1607 | }; | ||
1608 | |||
1609 | enum bt_co_type { | ||
1610 | BT_2WIRE = 0, | ||
1611 | BT_ISSC_3WIRE = 1, | ||
1612 | BT_ACCEL = 2, | ||
1613 | BT_CSR_BC4 = 3, | ||
1614 | BT_CSR_BC8 = 4, | ||
1615 | BT_RTL8756 = 5, | ||
1616 | }; | ||
1617 | |||
1618 | enum bt_cur_state { | ||
1619 | BT_OFF = 0, | ||
1620 | BT_ON = 1, | ||
1621 | }; | ||
1622 | |||
1623 | enum bt_service_type { | ||
1624 | BT_SCO = 0, | ||
1625 | BT_A2DP = 1, | ||
1626 | BT_HID = 2, | ||
1627 | BT_HID_IDLE = 3, | ||
1628 | BT_SCAN = 4, | ||
1629 | BT_IDLE = 5, | ||
1630 | BT_OTHER_ACTION = 6, | ||
1631 | BT_BUSY = 7, | ||
1632 | BT_OTHERBUSY = 8, | ||
1633 | BT_PAN = 9, | ||
1634 | }; | ||
1635 | |||
1636 | enum bt_radio_shared { | ||
1637 | BT_RADIO_SHARED = 0, | ||
1638 | BT_RADIO_INDIVIDUAL = 1, | ||
1639 | }; | ||
1640 | |||
1641 | struct bt_coexist_info { | ||
1642 | |||
1643 | /* EEPROM BT info. */ | ||
1644 | u8 eeprom_bt_coexist; | ||
1645 | u8 eeprom_bt_type; | ||
1646 | u8 eeprom_bt_ant_num; | ||
1647 | u8 eeprom_bt_ant_isolation; | ||
1648 | u8 eeprom_bt_radio_shared; | ||
1649 | |||
1650 | u8 bt_coexistence; | ||
1651 | u8 bt_ant_num; | ||
1652 | u8 bt_coexist_type; | ||
1653 | u8 bt_state; | ||
1654 | u8 bt_cur_state; /* 0:on, 1:off */ | ||
1655 | u8 bt_ant_isolation; /* 0:good, 1:bad */ | ||
1656 | u8 bt_pape_ctrl; /* 0:SW, 1:SW/HW dynamic */ | ||
1657 | u8 bt_service; | ||
1658 | u8 bt_radio_shared_type; | ||
1659 | u8 bt_rfreg_origin_1e; | ||
1660 | u8 bt_rfreg_origin_1f; | ||
1661 | u8 bt_rssi_state; | ||
1662 | u32 ratio_tx; | ||
1663 | u32 ratio_pri; | ||
1664 | u32 bt_edca_ul; | ||
1665 | u32 bt_edca_dl; | ||
1666 | |||
1667 | bool init_set; | ||
1668 | bool bt_busy_traffic; | ||
1669 | bool bt_traffic_mode_set; | ||
1670 | bool bt_non_traffic_mode_set; | ||
1671 | |||
1672 | bool fw_coexist_all_off; | ||
1673 | bool sw_coexist_all_off; | ||
1674 | u32 current_state; | ||
1675 | u32 previous_state; | ||
1676 | u8 bt_pre_rssi_state; | ||
1677 | |||
1678 | u8 reg_bt_iso; | ||
1679 | u8 reg_bt_sco; | ||
1680 | |||
1681 | }; | ||
1682 | |||
1683 | |||
1684 | /**************************************** | ||
1685 | mem access macro define start | ||
1686 | Call endian free function when | ||
1687 | 1. Read/write packet content. | ||
1688 | 2. Before write integer to IO. | ||
1689 | 3. After read integer from IO. | ||
1690 | ****************************************/ | ||
1691 | /* Convert little data endian to host ordering */ | ||
1692 | #define EF1BYTE(_val) \ | ||
1693 | ((u8)(_val)) | ||
1694 | #define EF2BYTE(_val) \ | ||
1695 | (le16_to_cpu(_val)) | ||
1696 | #define EF4BYTE(_val) \ | ||
1697 | (le32_to_cpu(_val)) | ||
1698 | |||
1699 | /* Read data from memory */ | ||
1700 | #define READEF1BYTE(_ptr) \ | ||
1701 | EF1BYTE(*((u8 *)(_ptr))) | ||
1702 | /* Read le16 data from memory and convert to host ordering */ | ||
1703 | #define READEF2BYTE(_ptr) \ | ||
1704 | EF2BYTE(*((u16 *)(_ptr))) | ||
1705 | #define READEF4BYTE(_ptr) \ | ||
1706 | EF4BYTE(*((u32 *)(_ptr))) | ||
1707 | |||
1708 | /* Write data to memory */ | ||
1709 | #define WRITEEF1BYTE(_ptr, _val) \ | ||
1710 | (*((u8 *)(_ptr))) = EF1BYTE(_val) | ||
1711 | /* Write le16 data to memory in host ordering */ | ||
1712 | #define WRITEEF2BYTE(_ptr, _val) \ | ||
1713 | (*((u16 *)(_ptr))) = EF2BYTE(_val) | ||
1714 | #define WRITEEF4BYTE(_ptr, _val) \ | ||
1715 | (*((u16 *)(_ptr))) = EF2BYTE(_val) | ||
1716 | |||
1717 | /* Create a bit mask | ||
1718 | * Examples: | ||
1719 | * BIT_LEN_MASK_32(0) => 0x00000000 | ||
1720 | * BIT_LEN_MASK_32(1) => 0x00000001 | ||
1721 | * BIT_LEN_MASK_32(2) => 0x00000003 | ||
1722 | * BIT_LEN_MASK_32(32) => 0xFFFFFFFF | ||
1723 | */ | ||
1724 | #define BIT_LEN_MASK_32(__bitlen) \ | ||
1725 | (0xFFFFFFFF >> (32 - (__bitlen))) | ||
1726 | #define BIT_LEN_MASK_16(__bitlen) \ | ||
1727 | (0xFFFF >> (16 - (__bitlen))) | ||
1728 | #define BIT_LEN_MASK_8(__bitlen) \ | ||
1729 | (0xFF >> (8 - (__bitlen))) | ||
1730 | |||
1731 | /* Create an offset bit mask | ||
1732 | * Examples: | ||
1733 | * BIT_OFFSET_LEN_MASK_32(0, 2) => 0x00000003 | ||
1734 | * BIT_OFFSET_LEN_MASK_32(16, 2) => 0x00030000 | ||
1735 | */ | ||
1736 | #define BIT_OFFSET_LEN_MASK_32(__bitoffset, __bitlen) \ | ||
1737 | (BIT_LEN_MASK_32(__bitlen) << (__bitoffset)) | ||
1738 | #define BIT_OFFSET_LEN_MASK_16(__bitoffset, __bitlen) \ | ||
1739 | (BIT_LEN_MASK_16(__bitlen) << (__bitoffset)) | ||
1740 | #define BIT_OFFSET_LEN_MASK_8(__bitoffset, __bitlen) \ | ||
1741 | (BIT_LEN_MASK_8(__bitlen) << (__bitoffset)) | ||
1742 | |||
1743 | /*Description: | ||
1744 | * Return 4-byte value in host byte ordering from | ||
1745 | * 4-byte pointer in little-endian system. | ||
1746 | */ | ||
1747 | #define LE_P4BYTE_TO_HOST_4BYTE(__pstart) \ | ||
1748 | (EF4BYTE(*((u32 *)(__pstart)))) | ||
1749 | #define LE_P2BYTE_TO_HOST_2BYTE(__pstart) \ | ||
1750 | (EF2BYTE(*((u16 *)(__pstart)))) | ||
1751 | #define LE_P1BYTE_TO_HOST_1BYTE(__pstart) \ | ||
1752 | (EF1BYTE(*((u8 *)(__pstart)))) | ||
1753 | |||
1754 | /*Description: | ||
1755 | Translate subfield (continuous bits in little-endian) of 4-byte | ||
1756 | value to host byte ordering.*/ | ||
1757 | #define LE_BITS_TO_4BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1758 | ( \ | ||
1759 | (LE_P4BYTE_TO_HOST_4BYTE(__pstart) >> (__bitoffset)) & \ | ||
1760 | BIT_LEN_MASK_32(__bitlen) \ | ||
1761 | ) | ||
1762 | #define LE_BITS_TO_2BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1763 | ( \ | ||
1764 | (LE_P2BYTE_TO_HOST_2BYTE(__pstart) >> (__bitoffset)) & \ | ||
1765 | BIT_LEN_MASK_16(__bitlen) \ | ||
1766 | ) | ||
1767 | #define LE_BITS_TO_1BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1768 | ( \ | ||
1769 | (LE_P1BYTE_TO_HOST_1BYTE(__pstart) >> (__bitoffset)) & \ | ||
1770 | BIT_LEN_MASK_8(__bitlen) \ | ||
1771 | ) | ||
1772 | |||
1773 | /* Description: | ||
1774 | * Mask subfield (continuous bits in little-endian) of 4-byte value | ||
1775 | * and return the result in 4-byte value in host byte ordering. | ||
1776 | */ | ||
1777 | #define LE_BITS_CLEARED_TO_4BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1778 | ( \ | ||
1779 | LE_P4BYTE_TO_HOST_4BYTE(__pstart) & \ | ||
1780 | (~BIT_OFFSET_LEN_MASK_32(__bitoffset, __bitlen)) \ | ||
1781 | ) | ||
1782 | #define LE_BITS_CLEARED_TO_2BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1783 | ( \ | ||
1784 | LE_P2BYTE_TO_HOST_2BYTE(__pstart) & \ | ||
1785 | (~BIT_OFFSET_LEN_MASK_16(__bitoffset, __bitlen)) \ | ||
1786 | ) | ||
1787 | #define LE_BITS_CLEARED_TO_1BYTE(__pstart, __bitoffset, __bitlen) \ | ||
1788 | ( \ | ||
1789 | LE_P1BYTE_TO_HOST_1BYTE(__pstart) & \ | ||
1790 | (~BIT_OFFSET_LEN_MASK_8(__bitoffset, __bitlen)) \ | ||
1791 | ) | ||
1792 | |||
1793 | /* Description: | ||
1794 | * Set subfield of little-endian 4-byte value to specified value. | ||
1795 | */ | ||
1796 | #define SET_BITS_TO_LE_4BYTE(__pstart, __bitoffset, __bitlen, __val) \ | ||
1797 | *((u32 *)(__pstart)) = EF4BYTE \ | ||
1798 | ( \ | ||
1799 | LE_BITS_CLEARED_TO_4BYTE(__pstart, __bitoffset, __bitlen) | \ | ||
1800 | ((((u32)__val) & BIT_LEN_MASK_32(__bitlen)) << (__bitoffset)) \ | ||
1801 | ); | ||
1802 | #define SET_BITS_TO_LE_2BYTE(__pstart, __bitoffset, __bitlen, __val) \ | ||
1803 | *((u16 *)(__pstart)) = EF2BYTE \ | ||
1804 | ( \ | ||
1805 | LE_BITS_CLEARED_TO_2BYTE(__pstart, __bitoffset, __bitlen) | \ | ||
1806 | ((((u16)__val) & BIT_LEN_MASK_16(__bitlen)) << (__bitoffset)) \ | ||
1807 | ); | ||
1808 | #define SET_BITS_TO_LE_1BYTE(__pstart, __bitoffset, __bitlen, __val) \ | ||
1809 | *((u8 *)(__pstart)) = EF1BYTE \ | ||
1810 | ( \ | ||
1811 | LE_BITS_CLEARED_TO_1BYTE(__pstart, __bitoffset, __bitlen) | \ | ||
1812 | ((((u8)__val) & BIT_LEN_MASK_8(__bitlen)) << (__bitoffset)) \ | ||
1813 | ); | ||
1814 | |||
1815 | #define N_BYTE_ALIGMENT(__value, __aligment) ((__aligment == 1) ? \ | ||
1816 | (__value) : (((__value + __aligment - 1) / __aligment) * __aligment)) | ||
1817 | |||
1818 | /**************************************** | ||
1819 | mem access macro define end | ||
1820 | ****************************************/ | ||
1821 | |||
1822 | #define byte(x, n) ((x >> (8 * n)) & 0xff) | ||
1823 | |||
1824 | #define packet_get_type(_packet) (EF1BYTE((_packet).octet[0]) & 0xFC) | ||
1825 | #define RTL_WATCH_DOG_TIME 2000 | ||
1826 | #define MSECS(t) msecs_to_jiffies(t) | ||
1827 | #define WLAN_FC_GET_VERS(fc) (le16_to_cpu(fc) & IEEE80211_FCTL_VERS) | ||
1828 | #define WLAN_FC_GET_TYPE(fc) (le16_to_cpu(fc) & IEEE80211_FCTL_FTYPE) | ||
1829 | #define WLAN_FC_GET_STYPE(fc) (le16_to_cpu(fc) & IEEE80211_FCTL_STYPE) | ||
1830 | #define WLAN_FC_MORE_DATA(fc) (le16_to_cpu(fc) & IEEE80211_FCTL_MOREDATA) | ||
1831 | #define SEQ_TO_SN(seq) (((seq) & IEEE80211_SCTL_SEQ) >> 4) | ||
1832 | #define SN_TO_SEQ(ssn) (((ssn) << 4) & IEEE80211_SCTL_SEQ) | ||
1833 | #define MAX_SN ((IEEE80211_SCTL_SEQ) >> 4) | ||
1834 | |||
1835 | #define RT_RF_OFF_LEVL_ASPM BIT(0) /*PCI ASPM */ | ||
1836 | #define RT_RF_OFF_LEVL_CLK_REQ BIT(1) /*PCI clock request */ | ||
1837 | #define RT_RF_OFF_LEVL_PCI_D3 BIT(2) /*PCI D3 mode */ | ||
1838 | /*NIC halt, re-initialize hw parameters*/ | ||
1839 | #define RT_RF_OFF_LEVL_HALT_NIC BIT(3) | ||
1840 | #define RT_RF_OFF_LEVL_FREE_FW BIT(4) /*FW free, re-download the FW */ | ||
1841 | #define RT_RF_OFF_LEVL_FW_32K BIT(5) /*FW in 32k */ | ||
1842 | /*Always enable ASPM and Clock Req in initialization.*/ | ||
1843 | #define RT_RF_PS_LEVEL_ALWAYS_ASPM BIT(6) | ||
1844 | /* no matter RFOFF or SLEEP we set PS_ASPM_LEVL*/ | ||
1845 | #define RT_PS_LEVEL_ASPM BIT(7) | ||
1846 | /*When LPS is on, disable 2R if no packet is received or transmittd.*/ | ||
1847 | #define RT_RF_LPS_DISALBE_2R BIT(30) | ||
1848 | #define RT_RF_LPS_LEVEL_ASPM BIT(31) /*LPS with ASPM */ | ||
1849 | #define RT_IN_PS_LEVEL(ppsc, _ps_flg) \ | ||
1850 | ((ppsc->cur_ps_level & _ps_flg) ? true : false) | ||
1851 | #define RT_CLEAR_PS_LEVEL(ppsc, _ps_flg) \ | ||
1852 | (ppsc->cur_ps_level &= (~(_ps_flg))) | ||
1853 | #define RT_SET_PS_LEVEL(ppsc, _ps_flg) \ | ||
1854 | (ppsc->cur_ps_level |= _ps_flg) | ||
1855 | |||
1856 | #define container_of_dwork_rtl(x, y, z) \ | ||
1857 | container_of(container_of(x, struct delayed_work, work), y, z) | ||
1858 | |||
1859 | #define FILL_OCTET_STRING(_os, _octet, _len) \ | ||
1860 | (_os).octet = (u8 *)(_octet); \ | ||
1861 | (_os).length = (_len); | ||
1862 | |||
1863 | #define CP_MACADDR(des, src) \ | ||
1864 | ((des)[0] = (src)[0], (des)[1] = (src)[1],\ | ||
1865 | (des)[2] = (src)[2], (des)[3] = (src)[3],\ | ||
1866 | (des)[4] = (src)[4], (des)[5] = (src)[5]) | ||
1867 | |||
1868 | static inline u8 rtl_read_byte(struct rtl_priv *rtlpriv, u32 addr) | ||
1869 | { | ||
1870 | return rtlpriv->io.read8_sync(rtlpriv, addr); | ||
1871 | } | ||
1872 | |||
1873 | static inline u16 rtl_read_word(struct rtl_priv *rtlpriv, u32 addr) | ||
1874 | { | ||
1875 | return rtlpriv->io.read16_sync(rtlpriv, addr); | ||
1876 | } | ||
1877 | |||
1878 | static inline u32 rtl_read_dword(struct rtl_priv *rtlpriv, u32 addr) | ||
1879 | { | ||
1880 | return rtlpriv->io.read32_sync(rtlpriv, addr); | ||
1881 | } | ||
1882 | |||
1883 | static inline void rtl_write_byte(struct rtl_priv *rtlpriv, u32 addr, u8 val8) | ||
1884 | { | ||
1885 | rtlpriv->io.write8_async(rtlpriv, addr, val8); | ||
1886 | |||
1887 | if (rtlpriv->cfg->write_readback) | ||
1888 | rtlpriv->io.read8_sync(rtlpriv, addr); | ||
1889 | } | ||
1890 | |||
1891 | static inline void rtl_write_word(struct rtl_priv *rtlpriv, u32 addr, u16 val16) | ||
1892 | { | ||
1893 | rtlpriv->io.write16_async(rtlpriv, addr, val16); | ||
1894 | |||
1895 | if (rtlpriv->cfg->write_readback) | ||
1896 | rtlpriv->io.read16_sync(rtlpriv, addr); | ||
1897 | } | ||
1898 | |||
1899 | static inline void rtl_write_dword(struct rtl_priv *rtlpriv, | ||
1900 | u32 addr, u32 val32) | ||
1901 | { | ||
1902 | rtlpriv->io.write32_async(rtlpriv, addr, val32); | ||
1903 | |||
1904 | if (rtlpriv->cfg->write_readback) | ||
1905 | rtlpriv->io.read32_sync(rtlpriv, addr); | ||
1906 | } | ||
1907 | |||
1908 | static inline u32 rtl_get_bbreg(struct ieee80211_hw *hw, | ||
1909 | u32 regaddr, u32 bitmask) | ||
1910 | { | ||
1911 | return ((struct rtl_priv *)(hw)->priv)->cfg->ops->get_bbreg(hw, | ||
1912 | regaddr, | ||
1913 | bitmask); | ||
1914 | } | ||
1915 | |||
1916 | static inline void rtl_set_bbreg(struct ieee80211_hw *hw, u32 regaddr, | ||
1917 | u32 bitmask, u32 data) | ||
1918 | { | ||
1919 | ((struct rtl_priv *)(hw)->priv)->cfg->ops->set_bbreg(hw, | ||
1920 | regaddr, bitmask, | ||
1921 | data); | ||
1922 | |||
1923 | } | ||
1924 | |||
1925 | static inline u32 rtl_get_rfreg(struct ieee80211_hw *hw, | ||
1926 | enum radio_path rfpath, u32 regaddr, | ||
1927 | u32 bitmask) | ||
1928 | { | ||
1929 | return ((struct rtl_priv *)(hw)->priv)->cfg->ops->get_rfreg(hw, | ||
1930 | rfpath, | ||
1931 | regaddr, | ||
1932 | bitmask); | ||
1933 | } | ||
1934 | |||
1935 | static inline void rtl_set_rfreg(struct ieee80211_hw *hw, | ||
1936 | enum radio_path rfpath, u32 regaddr, | ||
1937 | u32 bitmask, u32 data) | ||
1938 | { | ||
1939 | ((struct rtl_priv *)(hw)->priv)->cfg->ops->set_rfreg(hw, | ||
1940 | rfpath, regaddr, | ||
1941 | bitmask, data); | ||
1942 | } | ||
1943 | |||
1944 | static inline bool is_hal_stop(struct rtl_hal *rtlhal) | ||
1945 | { | ||
1946 | return (_HAL_STATE_STOP == rtlhal->state); | ||
1947 | } | ||
1948 | |||
1949 | static inline void set_hal_start(struct rtl_hal *rtlhal) | ||
1950 | { | ||
1951 | rtlhal->state = _HAL_STATE_START; | ||
1952 | } | ||
1953 | |||
1954 | static inline void set_hal_stop(struct rtl_hal *rtlhal) | ||
1955 | { | ||
1956 | rtlhal->state = _HAL_STATE_STOP; | ||
1957 | } | ||
1958 | |||
1959 | static inline u8 get_rf_type(struct rtl_phy *rtlphy) | ||
1960 | { | ||
1961 | return rtlphy->rf_type; | ||
1962 | } | ||
1963 | |||
1964 | static inline struct ieee80211_hdr *rtl_get_hdr(struct sk_buff *skb) | ||
1965 | { | ||
1966 | return (struct ieee80211_hdr *)(skb->data); | ||
1967 | } | ||
1968 | |||
1969 | static inline __le16 rtl_get_fc(struct sk_buff *skb) | ||
1970 | { | ||
1971 | return rtl_get_hdr(skb)->frame_control; | ||
1972 | } | ||
1973 | |||
1974 | static inline u16 rtl_get_tid_h(struct ieee80211_hdr *hdr) | ||
1975 | { | ||
1976 | return (ieee80211_get_qos_ctl(hdr))[0] & IEEE80211_QOS_CTL_TID_MASK; | ||
1977 | } | ||
1978 | |||
1979 | static inline u16 rtl_get_tid(struct sk_buff *skb) | ||
1980 | { | ||
1981 | return rtl_get_tid_h(rtl_get_hdr(skb)); | ||
1982 | } | ||
1983 | |||
1984 | static inline struct ieee80211_sta *get_sta(struct ieee80211_hw *hw, | ||
1985 | struct ieee80211_vif *vif, | ||
1986 | u8 *bssid) | ||
1987 | { | ||
1988 | return ieee80211_find_sta(vif, bssid); | ||
1989 | } | ||
1990 | |||
1991 | #endif | ||