blob: 50793c9df1b3711f54b435336ff8fef8e5d53473 [file] [log] [blame]
Larry Fingera2c60d42013-08-21 22:33:58 -05001/******************************************************************************
2 *
3 * Copyright(c) 2007 - 2012 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 *
Larry Fingera2c60d42013-08-21 22:33:58 -050014 ******************************************************************************/
15#define _IOCTL_LINUX_C_
16
Paul Gortmaker04fbf972015-04-27 01:25:34 -040017#include <linux/ieee80211.h>
18
Larry Fingera2c60d42013-08-21 22:33:58 -050019#include <osdep_service.h>
20#include <drv_types.h>
21#include <wlan_bssdef.h>
22#include <rtw_debug.h>
23#include <wifi.h>
24#include <rtw_mlme.h>
25#include <rtw_mlme_ext.h>
26#include <rtw_ioctl.h>
27#include <rtw_ioctl_set.h>
Larry Fingera2c60d42013-08-21 22:33:58 -050028#include <rtl8188e_hal.h>
29
Larry Fingera2c60d42013-08-21 22:33:58 -050030#include <rtw_iol.h>
Larry Fingerd249db9e52014-02-07 18:38:50 -060031#include <linux/vmalloc.h>
Aya Mahfouz8e5d9432015-03-04 08:56:45 +020032#include <linux/etherdevice.h>
33
Jeff Oczek45c2fc82014-07-24 20:19:43 -040034#include "osdep_intf.h"
Larry Fingera2c60d42013-08-21 22:33:58 -050035
36#define RTL_IOCTL_WPA_SUPPLICANT (SIOCIWFIRSTPRIV + 30)
37
38#define SCAN_ITEM_SIZE 768
39#define MAX_CUSTOM_LEN 64
40#define RATE_COUNT 4
41
42/* combo scan */
43#define WEXT_CSCAN_AMOUNT 9
44#define WEXT_CSCAN_BUF_LEN 360
45#define WEXT_CSCAN_HEADER "CSCAN S\x01\x00\x00S\x00"
46#define WEXT_CSCAN_HEADER_SIZE 12
47#define WEXT_CSCAN_SSID_SECTION 'S'
48#define WEXT_CSCAN_CHANNEL_SECTION 'C'
49#define WEXT_CSCAN_NPROBE_SECTION 'N'
50#define WEXT_CSCAN_ACTV_DWELL_SECTION 'A'
51#define WEXT_CSCAN_PASV_DWELL_SECTION 'P'
52#define WEXT_CSCAN_HOME_DWELL_SECTION 'H'
53#define WEXT_CSCAN_TYPE_SECTION 'T'
54
Larry Fingera2c60d42013-08-21 22:33:58 -050055static u32 rtw_rates[] = {1000000, 2000000, 5500000, 11000000,
56 6000000, 9000000, 12000000, 18000000, 24000000, 36000000,
57 48000000, 54000000};
58
59static const char * const iw_operation_mode[] = {
60 "Auto", "Ad-Hoc", "Managed", "Master", "Repeater",
61 "Secondary", "Monitor"
62};
63
Larry Fingera2c60d42013-08-21 22:33:58 -050064void indicate_wx_scan_complete_event(struct adapter *padapter)
65{
66 union iwreq_data wrqu;
67
navin patidar1ce39842014-06-22 13:49:28 +053068 memset(&wrqu, 0, sizeof(union iwreq_data));
Larry Fingera2c60d42013-08-21 22:33:58 -050069 wireless_send_event(padapter->pnetdev, SIOCGIWSCAN, &wrqu, NULL);
70}
71
72void rtw_indicate_wx_assoc_event(struct adapter *padapter)
73{
74 union iwreq_data wrqu;
75 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
76
navin patidar1ce39842014-06-22 13:49:28 +053077 memset(&wrqu, 0, sizeof(union iwreq_data));
Larry Fingera2c60d42013-08-21 22:33:58 -050078
79 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
80
81 memcpy(wrqu.ap_addr.sa_data, pmlmepriv->cur_network.network.MacAddress, ETH_ALEN);
82
83 DBG_88E_LEVEL(_drv_always_, "assoc success\n");
84 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
85}
86
87void rtw_indicate_wx_disassoc_event(struct adapter *padapter)
88{
89 union iwreq_data wrqu;
90
navin patidar1ce39842014-06-22 13:49:28 +053091 memset(&wrqu, 0, sizeof(union iwreq_data));
Larry Fingera2c60d42013-08-21 22:33:58 -050092
93 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
Aya Mahfouz8e5d9432015-03-04 08:56:45 +020094 eth_zero_addr(wrqu.ap_addr.sa_data);
Larry Fingera2c60d42013-08-21 22:33:58 -050095
96 DBG_88E_LEVEL(_drv_always_, "indicate disassoc\n");
97 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
98}
99
100static char *translate_scan(struct adapter *padapter,
101 struct iw_request_info *info,
102 struct wlan_network *pnetwork,
103 char *start, char *stop)
104{
105 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
106 struct iw_event iwe;
107 u16 cap;
108 __le16 le_tmp;
109 u32 ht_ielen = 0;
110 char custom[MAX_CUSTOM_LEN];
111 char *p;
112 u16 max_rate = 0, rate, ht_cap = false;
113 u32 i = 0;
114 u8 bw_40MHz = 0, short_GI = 0;
115 u16 mcs_rate = 0;
116 u8 ss, sq;
Larry Fingera2c60d42013-08-21 22:33:58 -0500117
118 /* AP MAC address */
119 iwe.cmd = SIOCGIWAP;
120 iwe.u.ap_addr.sa_family = ARPHRD_ETHER;
121
122 memcpy(iwe.u.ap_addr.sa_data, pnetwork->network.MacAddress, ETH_ALEN);
123 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_ADDR_LEN);
124
125 /* Add the ESSID */
126 iwe.cmd = SIOCGIWESSID;
127 iwe.u.data.flags = 1;
128 iwe.u.data.length = min_t(u16, pnetwork->network.Ssid.SsidLength, 32);
129 start = iwe_stream_add_point(info, start, stop, &iwe, pnetwork->network.Ssid.Ssid);
130
131 /* parsing HT_CAP_IE */
132 p = rtw_get_ie(&pnetwork->network.IEs[12], _HT_CAPABILITY_IE_, &ht_ielen, pnetwork->network.IELength-12);
133
134 if (p && ht_ielen > 0) {
Ivan Safonovda9090a2016-08-24 15:21:14 +0700135 struct ieee80211_ht_cap *pht_capie;
Larry Fingera2c60d42013-08-21 22:33:58 -0500136 ht_cap = true;
Ivan Safonovda9090a2016-08-24 15:21:14 +0700137 pht_capie = (struct ieee80211_ht_cap *)(p + 2);
138 memcpy(&mcs_rate, pht_capie->mcs.rx_mask, 2);
139 bw_40MHz = !!(le16_to_cpu(pht_capie->cap_info) &
140 IEEE80211_HT_CAP_SUP_WIDTH);
141 short_GI = !!(le16_to_cpu(pht_capie->cap_info) &
142 (IEEE80211_HT_CAP_SGI_20 |
143 IEEE80211_HT_CAP_SGI_40));
Larry Fingera2c60d42013-08-21 22:33:58 -0500144 }
145
146 /* Add the protocol name */
147 iwe.cmd = SIOCGIWNAME;
148 if ((rtw_is_cckratesonly_included((u8 *)&pnetwork->network.SupportedRates))) {
149 if (ht_cap)
150 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bn");
151 else
152 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11b");
153 } else if ((rtw_is_cckrates_included((u8 *)&pnetwork->network.SupportedRates))) {
154 if (ht_cap)
155 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bgn");
156 else
157 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bg");
158 } else {
159 if (pnetwork->network.Configuration.DSConfig > 14) {
160 if (ht_cap)
161 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11an");
162 else
163 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11a");
164 } else {
165 if (ht_cap)
166 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11gn");
167 else
168 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11g");
169 }
170 }
171
172 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_CHAR_LEN);
173
174 /* Add mode */
175 iwe.cmd = SIOCGIWMODE;
176 memcpy(&le_tmp, rtw_get_capability_from_ie(pnetwork->network.IEs), 2);
177
178 cap = le16_to_cpu(le_tmp);
179
Jakub Sitnicki027d3ef2015-06-26 07:50:34 +0200180 if (!WLAN_CAPABILITY_IS_STA_BSS(cap)) {
181 if (cap & WLAN_CAPABILITY_ESS)
Larry Fingera2c60d42013-08-21 22:33:58 -0500182 iwe.u.mode = IW_MODE_MASTER;
183 else
184 iwe.u.mode = IW_MODE_ADHOC;
185
186 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_UINT_LEN);
187 }
188
189 if (pnetwork->network.Configuration.DSConfig < 1)
190 pnetwork->network.Configuration.DSConfig = 1;
191
192 /* Add frequency/channel */
193 iwe.cmd = SIOCGIWFREQ;
194 iwe.u.freq.m = rtw_ch2freq(pnetwork->network.Configuration.DSConfig) * 100000;
195 iwe.u.freq.e = 1;
196 iwe.u.freq.i = pnetwork->network.Configuration.DSConfig;
197 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_FREQ_LEN);
198
199 /* Add encryption capability */
200 iwe.cmd = SIOCGIWENCODE;
201 if (cap & WLAN_CAPABILITY_PRIVACY)
202 iwe.u.data.flags = IW_ENCODE_ENABLED | IW_ENCODE_NOKEY;
203 else
204 iwe.u.data.flags = IW_ENCODE_DISABLED;
205 iwe.u.data.length = 0;
206 start = iwe_stream_add_point(info, start, stop, &iwe, pnetwork->network.Ssid.Ssid);
207
208 /*Add basic and extended rates */
209 max_rate = 0;
210 p = custom;
211 p += snprintf(p, MAX_CUSTOM_LEN - (p - custom), " Rates (Mb/s): ");
212 while (pnetwork->network.SupportedRates[i] != 0) {
213 rate = pnetwork->network.SupportedRates[i]&0x7F;
214 if (rate > max_rate)
215 max_rate = rate;
216 p += snprintf(p, MAX_CUSTOM_LEN - (p - custom),
217 "%d%s ", rate >> 1, (rate & 1) ? ".5" : "");
218 i++;
219 }
220
221 if (ht_cap) {
222 if (mcs_rate&0x8000)/* MCS15 */
223 max_rate = (bw_40MHz) ? ((short_GI) ? 300 : 270) : ((short_GI) ? 144 : 130);
224 else if (mcs_rate&0x0080)/* MCS7 */
225 ;
226 else/* default MCS7 */
227 max_rate = (bw_40MHz) ? ((short_GI) ? 150 : 135) : ((short_GI) ? 72 : 65);
228
229 max_rate = max_rate*2;/* Mbps/2; */
230 }
231
232 iwe.cmd = SIOCGIWRATE;
233 iwe.u.bitrate.fixed = 0;
234 iwe.u.bitrate.disabled = 0;
235 iwe.u.bitrate.value = max_rate * 500000;
236 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_PARAM_LEN);
237
238 /* parsing WPA/WPA2 IE */
239 {
240 u8 buf[MAX_WPA_IE_LEN];
241 u8 wpa_ie[255], rsn_ie[255];
242 u16 wpa_len = 0, rsn_len = 0;
243 u8 *p;
244
245 rtw_get_sec_ie(pnetwork->network.IEs, pnetwork->network.IELength, rsn_ie, &rsn_len, wpa_ie, &wpa_len);
246 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan: ssid =%s\n", pnetwork->network.Ssid.Ssid));
247 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan: wpa_len =%d rsn_len =%d\n", wpa_len, rsn_len));
248
249 if (wpa_len > 0) {
250 p = buf;
navin patidar1ce39842014-06-22 13:49:28 +0530251 memset(buf, 0, MAX_WPA_IE_LEN);
Wilfried Klaebeec03ab72014-03-25 17:59:39 +0000252 p += sprintf(p, "wpa_ie=");
Larry Fingera2c60d42013-08-21 22:33:58 -0500253 for (i = 0; i < wpa_len; i++)
254 p += sprintf(p, "%02x", wpa_ie[i]);
255
navin patidar1ce39842014-06-22 13:49:28 +0530256 memset(&iwe, 0, sizeof(iwe));
Larry Fingera2c60d42013-08-21 22:33:58 -0500257 iwe.cmd = IWEVCUSTOM;
258 iwe.u.data.length = strlen(buf);
259 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
260
navin patidar1ce39842014-06-22 13:49:28 +0530261 memset(&iwe, 0, sizeof(iwe));
Larry Fingera2c60d42013-08-21 22:33:58 -0500262 iwe.cmd = IWEVGENIE;
263 iwe.u.data.length = wpa_len;
264 start = iwe_stream_add_point(info, start, stop, &iwe, wpa_ie);
265 }
266 if (rsn_len > 0) {
267 p = buf;
navin patidar1ce39842014-06-22 13:49:28 +0530268 memset(buf, 0, MAX_WPA_IE_LEN);
Wilfried Klaebeec03ab72014-03-25 17:59:39 +0000269 p += sprintf(p, "rsn_ie=");
Larry Fingera2c60d42013-08-21 22:33:58 -0500270 for (i = 0; i < rsn_len; i++)
271 p += sprintf(p, "%02x", rsn_ie[i]);
navin patidar1ce39842014-06-22 13:49:28 +0530272 memset(&iwe, 0, sizeof(iwe));
Larry Fingera2c60d42013-08-21 22:33:58 -0500273 iwe.cmd = IWEVCUSTOM;
274 iwe.u.data.length = strlen(buf);
275 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
276
navin patidar1ce39842014-06-22 13:49:28 +0530277 memset(&iwe, 0, sizeof(iwe));
Larry Fingera2c60d42013-08-21 22:33:58 -0500278 iwe.cmd = IWEVGENIE;
279 iwe.u.data.length = rsn_len;
280 start = iwe_stream_add_point(info, start, stop, &iwe, rsn_ie);
281 }
282 }
283
284 {/* parsing WPS IE */
285 uint cnt = 0, total_ielen;
286 u8 *wpsie_ptr = NULL;
287 uint wps_ielen = 0;
288
289 u8 *ie_ptr = pnetwork->network.IEs + _FIXED_IE_LENGTH_;
290 total_ielen = pnetwork->network.IELength - _FIXED_IE_LENGTH_;
291
292 while (cnt < total_ielen) {
293 if (rtw_is_wps_ie(&ie_ptr[cnt], &wps_ielen) && (wps_ielen > 2)) {
294 wpsie_ptr = &ie_ptr[cnt];
295 iwe.cmd = IWEVGENIE;
296 iwe.u.data.length = (u16)wps_ielen;
297 start = iwe_stream_add_point(info, start, stop, &iwe, wpsie_ptr);
298 }
299 cnt += ie_ptr[cnt+1]+2; /* goto next */
300 }
301 }
302
303 /* Add quality statistics */
304 iwe.cmd = IWEVQUAL;
305 iwe.u.qual.updated = IW_QUAL_QUAL_UPDATED | IW_QUAL_LEVEL_UPDATED | IW_QUAL_NOISE_INVALID;
306
307 if (check_fwstate(pmlmepriv, _FW_LINKED) == true &&
308 is_same_network(&pmlmepriv->cur_network.network, &pnetwork->network)) {
309 ss = padapter->recvpriv.signal_strength;
310 sq = padapter->recvpriv.signal_qual;
311 } else {
312 ss = pnetwork->network.PhyInfo.SignalStrength;
313 sq = pnetwork->network.PhyInfo.SignalQuality;
314 }
315
316 iwe.u.qual.level = (u8)ss;
317 iwe.u.qual.qual = (u8)sq; /* signal quality */
318 iwe.u.qual.noise = 0; /* noise level */
319 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_QUAL_LEN);
320 return start;
321}
322
323static int wpa_set_auth_algs(struct net_device *dev, u32 value)
324{
325 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
326 int ret = 0;
327
328 if ((value & AUTH_ALG_SHARED_KEY) && (value & AUTH_ALG_OPEN_SYSTEM)) {
329 DBG_88E("wpa_set_auth_algs, AUTH_ALG_SHARED_KEY and AUTH_ALG_OPEN_SYSTEM [value:0x%x]\n", value);
330 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
331 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeAutoSwitch;
332 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Auto;
333 } else if (value & AUTH_ALG_SHARED_KEY) {
334 DBG_88E("wpa_set_auth_algs, AUTH_ALG_SHARED_KEY [value:0x%x]\n", value);
335 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
336
337 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeShared;
338 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Shared;
339 } else if (value & AUTH_ALG_OPEN_SYSTEM) {
340 DBG_88E("wpa_set_auth_algs, AUTH_ALG_OPEN_SYSTEM\n");
341 if (padapter->securitypriv.ndisauthtype < Ndis802_11AuthModeWPAPSK) {
342 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeOpen;
343 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
344 }
345 } else if (value & AUTH_ALG_LEAP) {
346 DBG_88E("wpa_set_auth_algs, AUTH_ALG_LEAP\n");
347 } else {
348 DBG_88E("wpa_set_auth_algs, error!\n");
349 ret = -EINVAL;
350 }
351 return ret;
352}
353
354static int wpa_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
355{
356 int ret = 0;
357 u32 wep_key_idx, wep_key_len, wep_total_len;
358 struct ndis_802_11_wep *pwep = NULL;
359 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
360 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
361 struct security_priv *psecuritypriv = &padapter->securitypriv;
Larry Fingera2c60d42013-08-21 22:33:58 -0500362
Larry Fingera2c60d42013-08-21 22:33:58 -0500363 param->u.crypt.err = 0;
364 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
365
Jia He7be921a22014-11-04 09:39:58 +0800366 if (param_len < (u32)((u8 *)param->u.crypt.key - (u8 *)param) + param->u.crypt.key_len) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500367 ret = -EINVAL;
368 goto exit;
369 }
370
371 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
372 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
373 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff) {
374 if (param->u.crypt.idx >= WEP_KEYS) {
375 ret = -EINVAL;
376 goto exit;
377 }
378 } else {
379 ret = -EINVAL;
380 goto exit;
381 }
382
383 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
384 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("wpa_set_encryption, crypt.alg = WEP\n"));
385 DBG_88E("wpa_set_encryption, crypt.alg = WEP\n");
386
387 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
388 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
389 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
390
391 wep_key_idx = param->u.crypt.idx;
392 wep_key_len = param->u.crypt.key_len;
393
394 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("(1)wep_key_idx =%d\n", wep_key_idx));
395 DBG_88E("(1)wep_key_idx =%d\n", wep_key_idx);
396
397 if (wep_key_idx > WEP_KEYS)
398 return -EINVAL;
399
400 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("(2)wep_key_idx =%d\n", wep_key_idx));
401
402 if (wep_key_len > 0) {
403 wep_key_len = wep_key_len <= 5 ? 5 : 13;
Ivan Safonov5e3027b2016-03-02 14:59:01 +0700404 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
Larry Fingera2c60d42013-08-21 22:33:58 -0500405 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530406 if (!pwep) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500407 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, (" wpa_set_encryption: pwep allocate fail !!!\n"));
408 goto exit;
409 }
navin patidar1ce39842014-06-22 13:49:28 +0530410 memset(pwep, 0, wep_total_len);
Larry Fingera2c60d42013-08-21 22:33:58 -0500411 pwep->KeyLength = wep_key_len;
412 pwep->Length = wep_total_len;
413 if (wep_key_len == 13) {
414 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
415 padapter->securitypriv.dot118021XGrpPrivacy = _WEP104_;
416 }
417 } else {
418 ret = -EINVAL;
419 goto exit;
420 }
421 pwep->KeyIndex = wep_key_idx;
422 pwep->KeyIndex |= 0x80000000;
423 memcpy(pwep->KeyMaterial, param->u.crypt.key, pwep->KeyLength);
424 if (param->u.crypt.set_tx) {
425 DBG_88E("wep, set_tx = 1\n");
426 if (rtw_set_802_11_add_wep(padapter, pwep) == (u8)_FAIL)
427 ret = -EOPNOTSUPP;
428 } else {
429 DBG_88E("wep, set_tx = 0\n");
430 if (wep_key_idx >= WEP_KEYS) {
431 ret = -EOPNOTSUPP;
432 goto exit;
433 }
434 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
435 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
436 rtw_set_key(padapter, psecuritypriv, wep_key_idx, 0);
437 }
438 goto exit;
439 }
440
441 if (padapter->securitypriv.dot11AuthAlgrthm == dot11AuthAlgrthm_8021X) { /* 802_1x */
442 struct sta_info *psta, *pbcmc_sta;
443 struct sta_priv *pstapriv = &padapter->stapriv;
444
445 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE | WIFI_MP_STATE)) { /* sta mode */
446 psta = rtw_get_stainfo(pstapriv, get_bssid(pmlmepriv));
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530447 if (!psta) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500448 ;
449 } else {
450 if (strcmp(param->u.crypt.alg, "none") != 0)
451 psta->ieee8021x_blocked = false;
452
453 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
454 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
455 psta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
456
457 if (param->u.crypt.set_tx == 1) { /* pairwise key */
Ivan Safonovad8d8cd2015-10-27 22:18:10 +0700458 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -0500459
460 if (strcmp(param->u.crypt.alg, "TKIP") == 0) { /* set mic key */
461 memcpy(psta->dot11tkiptxmickey.skey, &(param->u.crypt.key[16]), 8);
462 memcpy(psta->dot11tkiprxmickey.skey, &(param->u.crypt.key[24]), 8);
463 padapter->securitypriv.busetkipkey = false;
464 }
465
466 DBG_88E(" ~~~~set sta key:unicastkey\n");
467
468 rtw_setstakey_cmd(padapter, (unsigned char *)psta, true);
469 } else { /* group key */
Ivan Safonovad8d8cd2015-10-27 22:18:10 +0700470 memcpy(padapter->securitypriv.dot118021XGrpKey[param->u.crypt.idx].skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16 ));
Larry Fingera2c60d42013-08-21 22:33:58 -0500471 memcpy(padapter->securitypriv.dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
472 memcpy(padapter->securitypriv.dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
473 padapter->securitypriv.binstallGrpkey = true;
474 DBG_88E(" ~~~~set sta key:groupkey\n");
475
476 padapter->securitypriv.dot118021XGrpKeyid = param->u.crypt.idx;
477
478 rtw_set_key(padapter, &padapter->securitypriv, param->u.crypt.idx, 1);
Larry Fingera2c60d42013-08-21 22:33:58 -0500479 }
480 }
481 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530482 if (!pbcmc_sta) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500483 ;
484 } else {
485 /* Jeff: don't disable ieee8021x_blocked while clearing key */
486 if (strcmp(param->u.crypt.alg, "none") != 0)
487 pbcmc_sta->ieee8021x_blocked = false;
488
489 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
490 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
491 pbcmc_sta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
492 }
493 }
494 }
495
496exit:
497
498 kfree(pwep);
Larry Fingera2c60d42013-08-21 22:33:58 -0500499 return ret;
500}
501
502static int rtw_set_wpa_ie(struct adapter *padapter, char *pie, unsigned short ielen)
503{
504 u8 *buf = NULL;
505 int group_cipher = 0, pairwise_cipher = 0;
506 int ret = 0;
Larry Fingera2c60d42013-08-21 22:33:58 -0500507
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530508 if ((ielen > MAX_WPA_IE_LEN) || (!pie)) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500509 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530510 if (!pie)
Larry Fingera2c60d42013-08-21 22:33:58 -0500511 return ret;
512 else
513 return -EINVAL;
514 }
515
516 if (ielen) {
Tapasweni Pathak705515c2014-10-21 09:48:20 +0530517 buf = kmemdup(pie, ielen, GFP_KERNEL);
Amit Ghadge4fe514d2016-09-04 22:53:35 +0530518 if (!buf) {
Larry Fingera2c60d42013-08-21 22:33:58 -0500519 ret = -ENOMEM;
520 goto exit;
521 }
522
Larry Fingera2c60d42013-08-21 22:33:58 -0500523 /* dump */
524 {
525 int i;
526 DBG_88E("\n wpa_ie(length:%d):\n", ielen);
527 for (i = 0; i < ielen; i += 8)
528 DBG_88E("0x%.2x 0x%.2x 0x%.2x 0x%.2x 0x%.2x 0x%.2x 0x%.2x 0x%.2x\n", buf[i], buf[i+1], buf[i+2], buf[i+3], buf[i+4], buf[i+5], buf[i+6], buf[i+7]);
529 }
530
531 if (ielen < RSN_HEADER_LEN) {
532 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("Ie len too short %d\n", ielen));
533 ret = -1;
534 goto exit;
535 }
536
537 if (rtw_parse_wpa_ie(buf, ielen, &group_cipher, &pairwise_cipher, NULL) == _SUCCESS) {
538 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X;
539 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPAPSK;
540 memcpy(padapter->securitypriv.supplicant_ie, &buf[0], ielen);
541 }
542
543 if (rtw_parse_wpa2_ie(buf, ielen, &group_cipher, &pairwise_cipher, NULL) == _SUCCESS) {
544 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X;
545 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPA2PSK;
546 memcpy(padapter->securitypriv.supplicant_ie, &buf[0], ielen);
547 }
548
549 switch (group_cipher) {
550 case WPA_CIPHER_NONE:
551 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
552 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
553 break;
554 case WPA_CIPHER_WEP40:
555 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
556 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
557 break;
558 case WPA_CIPHER_TKIP:
559 padapter->securitypriv.dot118021XGrpPrivacy = _TKIP_;
560 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
561 break;
562 case WPA_CIPHER_CCMP:
563 padapter->securitypriv.dot118021XGrpPrivacy = _AES_;
564 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
565 break;
566 case WPA_CIPHER_WEP104:
567 padapter->securitypriv.dot118021XGrpPrivacy = _WEP104_;
568 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
569 break;
570 }
571
572 switch (pairwise_cipher) {
573 case WPA_CIPHER_NONE:
574 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
575 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
576 break;
577 case WPA_CIPHER_WEP40:
578 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
579 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
580 break;
581 case WPA_CIPHER_TKIP:
582 padapter->securitypriv.dot11PrivacyAlgrthm = _TKIP_;
583 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
584 break;
585 case WPA_CIPHER_CCMP:
586 padapter->securitypriv.dot11PrivacyAlgrthm = _AES_;
587 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
588 break;
589 case WPA_CIPHER_WEP104:
590 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
591 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
592 break;
593 }
594
595 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
596 {/* set wps_ie */
597 u16 cnt = 0;
598 u8 eid, wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
599
600 while (cnt < ielen) {
601 eid = buf[cnt];
602 if ((eid == _VENDOR_SPECIFIC_IE_) && (!memcmp(&buf[cnt+2], wps_oui, 4))) {
603 DBG_88E("SET WPS_IE\n");
604
Ivan Safonov20e76532015-10-27 22:16:25 +0700605 padapter->securitypriv.wps_ie_len = min(buf[cnt + 1] + 2, MAX_WPA_IE_LEN << 2);
Larry Fingera2c60d42013-08-21 22:33:58 -0500606
607 memcpy(padapter->securitypriv.wps_ie, &buf[cnt], padapter->securitypriv.wps_ie_len);
608
609 set_fwstate(&padapter->mlmepriv, WIFI_UNDER_WPS);
Larry Fingera2c60d42013-08-21 22:33:58 -0500610 cnt += buf[cnt+1]+2;
611 break;
612 } else {
613 cnt += buf[cnt+1]+2; /* goto next */
614 }
615 }
616 }
617 }
618
619 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
620 ("rtw_set_wpa_ie: pairwise_cipher = 0x%08x padapter->securitypriv.ndisencryptstatus =%d padapter->securitypriv.ndisauthtype =%d\n",
621 pairwise_cipher, padapter->securitypriv.ndisencryptstatus, padapter->securitypriv.ndisauthtype));
622exit:
623 kfree(buf);
624 return ret;
625}
626
627typedef unsigned char NDIS_802_11_RATES_EX[NDIS_802_11_LENGTH_RATES_EX];
628
629static int rtw_wx_get_name(struct net_device *dev,
630 struct iw_request_info *info,
631 union iwreq_data *wrqu, char *extra)
632{
633 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
634 u32 ht_ielen = 0;
635 char *p;
636 u8 ht_cap = false;
637 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
638 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
639 NDIS_802_11_RATES_EX *prates = NULL;
640
641 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("cmd_code =%x\n", info->cmd));
642
Larry Fingera2c60d42013-08-21 22:33:58 -0500643 if (check_fwstate(pmlmepriv, _FW_LINKED|WIFI_ADHOC_MASTER_STATE) == true) {
644 /* parsing HT_CAP_IE */
645 p = rtw_get_ie(&pcur_bss->IEs[12], _HT_CAPABILITY_IE_, &ht_ielen, pcur_bss->IELength-12);
646 if (p && ht_ielen > 0)
647 ht_cap = true;
648
649 prates = &pcur_bss->SupportedRates;
650
651 if (rtw_is_cckratesonly_included((u8 *)prates) == true) {
652 if (ht_cap)
653 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bn");
654 else
655 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11b");
656 } else if ((rtw_is_cckrates_included((u8 *)prates)) == true) {
657 if (ht_cap)
658 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bgn");
659 else
660 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bg");
661 } else {
662 if (pcur_bss->Configuration.DSConfig > 14) {
663 if (ht_cap)
664 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11an");
665 else
666 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11a");
667 } else {
668 if (ht_cap)
669 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11gn");
670 else
671 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11g");
672 }
673 }
674 } else {
675 snprintf(wrqu->name, IFNAMSIZ, "unassociated");
676 }
Larry Fingera2c60d42013-08-21 22:33:58 -0500677 return 0;
678}
679
680static int rtw_wx_set_freq(struct net_device *dev,
681 struct iw_request_info *info,
682 union iwreq_data *wrqu, char *extra)
683{
Larry Fingera2c60d42013-08-21 22:33:58 -0500684 RT_TRACE(_module_rtl871x_mlme_c_, _drv_notice_, ("+rtw_wx_set_freq\n"));
Larry Fingera2c60d42013-08-21 22:33:58 -0500685 return 0;
686}
687
688static int rtw_wx_get_freq(struct net_device *dev,
689 struct iw_request_info *info,
690 union iwreq_data *wrqu, char *extra)
691{
692 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
693 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
694 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
695
696 if (check_fwstate(pmlmepriv, _FW_LINKED)) {
697 /* wrqu->freq.m = ieee80211_wlan_frequencies[pcur_bss->Configuration.DSConfig-1] * 100000; */
698 wrqu->freq.m = rtw_ch2freq(pcur_bss->Configuration.DSConfig) * 100000;
699 wrqu->freq.e = 1;
700 wrqu->freq.i = pcur_bss->Configuration.DSConfig;
701 } else {
702 wrqu->freq.m = rtw_ch2freq(padapter->mlmeextpriv.cur_channel) * 100000;
703 wrqu->freq.e = 1;
704 wrqu->freq.i = padapter->mlmeextpriv.cur_channel;
705 }
706
707 return 0;
708}
709
710static int rtw_wx_set_mode(struct net_device *dev, struct iw_request_info *a,
711 union iwreq_data *wrqu, char *b)
712{
713 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
714 enum ndis_802_11_network_infra networkType;
715 int ret = 0;
716
Larry Fingera2c60d42013-08-21 22:33:58 -0500717 if (_FAIL == rtw_pwr_wakeup(padapter)) {
718 ret = -EPERM;
719 goto exit;
720 }
721
722 if (!padapter->hw_init_completed) {
723 ret = -EPERM;
724 goto exit;
725 }
726
727 switch (wrqu->mode) {
728 case IW_MODE_AUTO:
729 networkType = Ndis802_11AutoUnknown;
730 DBG_88E("set_mode = IW_MODE_AUTO\n");
731 break;
732 case IW_MODE_ADHOC:
733 networkType = Ndis802_11IBSS;
734 DBG_88E("set_mode = IW_MODE_ADHOC\n");
735 break;
736 case IW_MODE_MASTER:
737 networkType = Ndis802_11APMode;
738 DBG_88E("set_mode = IW_MODE_MASTER\n");
739 break;
740 case IW_MODE_INFRA:
741 networkType = Ndis802_11Infrastructure;
742 DBG_88E("set_mode = IW_MODE_INFRA\n");
743 break;
744 default:
745 ret = -EINVAL;
746 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("\n Mode: %s is not supported\n", iw_operation_mode[wrqu->mode]));
747 goto exit;
748 }
749 if (rtw_set_802_11_infrastructure_mode(padapter, networkType) == false) {
750 ret = -EPERM;
751 goto exit;
752 }
753 rtw_setopmode_cmd(padapter, networkType);
754exit:
Larry Fingera2c60d42013-08-21 22:33:58 -0500755 return ret;
756}
757
758static int rtw_wx_get_mode(struct net_device *dev, struct iw_request_info *a,
759 union iwreq_data *wrqu, char *b)
760{
761 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
762 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
763
764 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_get_mode\n"));
765
Larry Fingera2c60d42013-08-21 22:33:58 -0500766 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE))
767 wrqu->mode = IW_MODE_INFRA;
768 else if ((check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) ||
769 (check_fwstate(pmlmepriv, WIFI_ADHOC_STATE)))
770 wrqu->mode = IW_MODE_ADHOC;
771 else if (check_fwstate(pmlmepriv, WIFI_AP_STATE))
772 wrqu->mode = IW_MODE_MASTER;
773 else
774 wrqu->mode = IW_MODE_AUTO;
775
Larry Fingera2c60d42013-08-21 22:33:58 -0500776 return 0;
777}
778
779static int rtw_wx_set_pmkid(struct net_device *dev,
780 struct iw_request_info *a,
781 union iwreq_data *wrqu, char *extra)
782{
783 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
784 u8 j, blInserted = false;
785 int ret = false;
786 struct security_priv *psecuritypriv = &padapter->securitypriv;
787 struct iw_pmksa *pPMK = (struct iw_pmksa *)extra;
788 u8 strZeroMacAddress[ETH_ALEN] = {0x00};
789 u8 strIssueBssid[ETH_ALEN] = {0x00};
790
791 memcpy(strIssueBssid, pPMK->bssid.sa_data, ETH_ALEN);
792 if (pPMK->cmd == IW_PMKSA_ADD) {
793 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_ADD!\n");
Larry Fingerdc14a622013-09-04 15:56:02 -0500794 if (!memcmp(strIssueBssid, strZeroMacAddress, ETH_ALEN))
Larry Fingera2c60d42013-08-21 22:33:58 -0500795 return ret;
796 else
797 ret = true;
798 blInserted = false;
799
800 /* overwrite PMKID */
801 for (j = 0; j < NUM_PMKID_CACHE; j++) {
802 if (!memcmp(psecuritypriv->PMKIDList[j].Bssid, strIssueBssid, ETH_ALEN)) {
803 /* BSSID is matched, the same AP => rewrite with new PMKID. */
804 DBG_88E("[rtw_wx_set_pmkid] BSSID exists in the PMKList.\n");
805 memcpy(psecuritypriv->PMKIDList[j].PMKID, pPMK->pmkid, IW_PMKID_LEN);
806 psecuritypriv->PMKIDList[j].bUsed = true;
807 psecuritypriv->PMKIDIndex = j+1;
808 blInserted = true;
809 break;
810 }
811 }
812
813 if (!blInserted) {
814 /* Find a new entry */
815 DBG_88E("[rtw_wx_set_pmkid] Use the new entry index = %d for this PMKID.\n",
816 psecuritypriv->PMKIDIndex);
817
818 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].Bssid, strIssueBssid, ETH_ALEN);
819 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].PMKID, pPMK->pmkid, IW_PMKID_LEN);
820
821 psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].bUsed = true;
822 psecuritypriv->PMKIDIndex++;
823 if (psecuritypriv->PMKIDIndex == 16)
824 psecuritypriv->PMKIDIndex = 0;
825 }
826 } else if (pPMK->cmd == IW_PMKSA_REMOVE) {
827 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_REMOVE!\n");
828 ret = true;
829 for (j = 0; j < NUM_PMKID_CACHE; j++) {
830 if (!memcmp(psecuritypriv->PMKIDList[j].Bssid, strIssueBssid, ETH_ALEN)) {
831 /* BSSID is matched, the same AP => Remove this PMKID information and reset it. */
Aya Mahfouz8e5d9432015-03-04 08:56:45 +0200832 eth_zero_addr(psecuritypriv->PMKIDList[j].Bssid);
Larry Fingera2c60d42013-08-21 22:33:58 -0500833 psecuritypriv->PMKIDList[j].bUsed = false;
834 break;
835 }
836 }
837 } else if (pPMK->cmd == IW_PMKSA_FLUSH) {
838 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_FLUSH!\n");
navin patidar1ce39842014-06-22 13:49:28 +0530839 memset(&psecuritypriv->PMKIDList[0], 0x00, sizeof(struct rt_pmkid_list) * NUM_PMKID_CACHE);
Larry Fingera2c60d42013-08-21 22:33:58 -0500840 psecuritypriv->PMKIDIndex = 0;
841 ret = true;
842 }
843 return ret;
844}
845
846static int rtw_wx_get_sens(struct net_device *dev,
847 struct iw_request_info *info,
848 union iwreq_data *wrqu, char *extra)
849{
850 wrqu->sens.value = 0;
851 wrqu->sens.fixed = 0; /* no auto select */
852 wrqu->sens.disabled = 1;
853 return 0;
854}
855
856static int rtw_wx_get_range(struct net_device *dev,
857 struct iw_request_info *info,
858 union iwreq_data *wrqu, char *extra)
859{
860 struct iw_range *range = (struct iw_range *)extra;
861 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
862 struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
863
864 u16 val;
865 int i;
866
Larry Fingera2c60d42013-08-21 22:33:58 -0500867 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_range. cmd_code =%x\n", info->cmd));
868
869 wrqu->data.length = sizeof(*range);
navin patidar1ce39842014-06-22 13:49:28 +0530870 memset(range, 0, sizeof(*range));
Larry Fingera2c60d42013-08-21 22:33:58 -0500871
872 /* Let's try to keep this struct in the same order as in
873 * linux/include/wireless.h
874 */
875
876 /* TODO: See what values we can set, and remove the ones we can't
877 * set, or fill them with some default data.
878 */
879
880 /* ~5 Mb/s real (802.11b) */
881 range->throughput = 5 * 1000 * 1000;
882
883 /* signal level threshold range */
884
885 /* percent values between 0 and 100. */
886 range->max_qual.qual = 100;
887 range->max_qual.level = 100;
888 range->max_qual.noise = 100;
889 range->max_qual.updated = 7; /* Updated all three */
890
891 range->avg_qual.qual = 92; /* > 8% missed beacons is 'bad' */
892 /* TODO: Find real 'good' to 'bad' threshol value for RSSI */
Larry Fingerdc14a622013-09-04 15:56:02 -0500893 range->avg_qual.level = 178; /* -78 dBm */
Larry Fingera2c60d42013-08-21 22:33:58 -0500894 range->avg_qual.noise = 0;
895 range->avg_qual.updated = 7; /* Updated all three */
896
897 range->num_bitrates = RATE_COUNT;
898
899 for (i = 0; i < RATE_COUNT && i < IW_MAX_BITRATES; i++)
900 range->bitrate[i] = rtw_rates[i];
901
902 range->min_frag = MIN_FRAG_THRESHOLD;
903 range->max_frag = MAX_FRAG_THRESHOLD;
904
905 range->pm_capa = 0;
906
907 range->we_version_compiled = WIRELESS_EXT;
908 range->we_version_source = 16;
909
910 for (i = 0, val = 0; i < MAX_CHANNEL_NUM; i++) {
911 /* Include only legal frequencies for some countries */
912 if (pmlmeext->channel_set[i].ChannelNum != 0) {
913 range->freq[val].i = pmlmeext->channel_set[i].ChannelNum;
914 range->freq[val].m = rtw_ch2freq(pmlmeext->channel_set[i].ChannelNum) * 100000;
915 range->freq[val].e = 1;
916 val++;
917 }
918
919 if (val == IW_MAX_FREQUENCIES)
920 break;
921 }
922
923 range->num_channels = val;
924 range->num_frequency = val;
925
926/* The following code will proivde the security capability to network manager. */
927/* If the driver doesn't provide this capability to network manager, */
Masanari Iida7efc02c2013-09-27 00:11:46 +0900928/* the WPA/WPA2 routers can't be chosen in the network manager. */
Larry Fingera2c60d42013-08-21 22:33:58 -0500929
930/*
931#define IW_SCAN_CAPA_NONE 0x00
932#define IW_SCAN_CAPA_ESSID 0x01
933#define IW_SCAN_CAPA_BSSID 0x02
934#define IW_SCAN_CAPA_CHANNEL 0x04
935#define IW_SCAN_CAPA_MODE 0x08
936#define IW_SCAN_CAPA_RATE 0x10
937#define IW_SCAN_CAPA_TYPE 0x20
938#define IW_SCAN_CAPA_TIME 0x40
939*/
940
941 range->enc_capa = IW_ENC_CAPA_WPA | IW_ENC_CAPA_WPA2 |
942 IW_ENC_CAPA_CIPHER_TKIP | IW_ENC_CAPA_CIPHER_CCMP;
943
944 range->scan_capa = IW_SCAN_CAPA_ESSID | IW_SCAN_CAPA_TYPE |
945 IW_SCAN_CAPA_BSSID | IW_SCAN_CAPA_CHANNEL |
946 IW_SCAN_CAPA_MODE | IW_SCAN_CAPA_RATE;
Larry Fingera2c60d42013-08-21 22:33:58 -0500947 return 0;
948}
949
950/* set bssid flow */
951/* s1. rtw_set_802_11_infrastructure_mode() */
952/* s2. rtw_set_802_11_authentication_mode() */
953/* s3. set_802_11_encryption_mode() */
954/* s4. rtw_set_802_11_bssid() */
955static int rtw_wx_set_wap(struct net_device *dev,
956 struct iw_request_info *info,
957 union iwreq_data *awrq,
958 char *extra)
959{
Larry Fingera2c60d42013-08-21 22:33:58 -0500960 uint ret = 0;
961 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
962 struct sockaddr *temp = (struct sockaddr *)awrq;
963 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
964 struct list_head *phead;
965 u8 *dst_bssid, *src_bssid;
966 struct __queue *queue = &(pmlmepriv->scanned_queue);
967 struct wlan_network *pnetwork = NULL;
968 enum ndis_802_11_auth_mode authmode;
969
Larry Fingera2c60d42013-08-21 22:33:58 -0500970 if (_FAIL == rtw_pwr_wakeup(padapter)) {
971 ret = -1;
972 goto exit;
973 }
974
975 if (!padapter->bup) {
976 ret = -1;
977 goto exit;
978 }
979
980 if (temp->sa_family != ARPHRD_ETHER) {
981 ret = -EINVAL;
982 goto exit;
983 }
984
985 authmode = padapter->securitypriv.ndisauthtype;
Larry Finger7057dcb2013-12-19 22:38:34 -0600986 spin_lock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -0500987 phead = get_list_head(queue);
Larry Fingerc44e5e32014-02-09 15:15:58 -0600988 pmlmepriv->pscanned = phead->next;
Larry Fingera2c60d42013-08-21 22:33:58 -0500989
navin patidar84660702014-06-22 13:49:32 +0530990 while (phead != pmlmepriv->pscanned) {
Larry Fingerbea88102014-02-09 15:15:57 -0600991 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
Larry Fingera2c60d42013-08-21 22:33:58 -0500992
Larry Fingerc44e5e32014-02-09 15:15:58 -0600993 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
Larry Fingera2c60d42013-08-21 22:33:58 -0500994
995 dst_bssid = pnetwork->network.MacAddress;
996
997 src_bssid = temp->sa_data;
998
999 if ((!memcmp(dst_bssid, src_bssid, ETH_ALEN))) {
1000 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
1001 ret = -1;
Larry Fingere02bcf62013-12-19 22:38:35 -06001002 spin_unlock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001003 goto exit;
1004 }
1005
1006 break;
1007 }
1008 }
Larry Fingere02bcf62013-12-19 22:38:35 -06001009 spin_unlock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001010
1011 rtw_set_802_11_authentication_mode(padapter, authmode);
1012 /* set_802_11_encryption_mode(padapter, padapter->securitypriv.ndisencryptstatus); */
1013 if (rtw_set_802_11_bssid(padapter, temp->sa_data) == false) {
1014 ret = -1;
1015 goto exit;
1016 }
1017
1018exit:
1019
Larry Fingera2c60d42013-08-21 22:33:58 -05001020 return ret;
1021}
1022
1023static int rtw_wx_get_wap(struct net_device *dev,
1024 struct iw_request_info *info,
1025 union iwreq_data *wrqu, char *extra)
1026{
1027 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1028 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1029 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
1030
1031 wrqu->ap_addr.sa_family = ARPHRD_ETHER;
1032
Aya Mahfouz8e5d9432015-03-04 08:56:45 +02001033 eth_zero_addr(wrqu->ap_addr.sa_data);
Larry Fingera2c60d42013-08-21 22:33:58 -05001034
1035 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_wap\n"));
1036
Larry Fingera2c60d42013-08-21 22:33:58 -05001037 if (((check_fwstate(pmlmepriv, _FW_LINKED)) == true) ||
1038 ((check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) == true) ||
1039 ((check_fwstate(pmlmepriv, WIFI_AP_STATE)) == true))
1040 memcpy(wrqu->ap_addr.sa_data, pcur_bss->MacAddress, ETH_ALEN);
1041 else
Aya Mahfouz8e5d9432015-03-04 08:56:45 +02001042 eth_zero_addr(wrqu->ap_addr.sa_data);
Larry Fingera2c60d42013-08-21 22:33:58 -05001043 return 0;
1044}
1045
1046static int rtw_wx_set_mlme(struct net_device *dev,
1047 struct iw_request_info *info,
1048 union iwreq_data *wrqu, char *extra)
1049{
1050 int ret = 0;
1051 u16 reason;
1052 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1053 struct iw_mlme *mlme = (struct iw_mlme *)extra;
1054
Amit Ghadge4fe514d2016-09-04 22:53:35 +05301055 if (!mlme)
Larry Fingera2c60d42013-08-21 22:33:58 -05001056 return -1;
1057
1058 DBG_88E("%s\n", __func__);
1059
1060 reason = mlme->reason_code;
1061
1062 DBG_88E("%s, cmd =%d, reason =%d\n", __func__, mlme->cmd, reason);
1063
1064 switch (mlme->cmd) {
1065 case IW_MLME_DEAUTH:
1066 if (!rtw_set_802_11_disassociate(padapter))
1067 ret = -1;
1068 break;
1069 case IW_MLME_DISASSOC:
1070 if (!rtw_set_802_11_disassociate(padapter))
1071 ret = -1;
1072 break;
1073 default:
1074 return -EOPNOTSUPP;
1075 }
1076 return ret;
1077}
1078
1079static int rtw_wx_set_scan(struct net_device *dev, struct iw_request_info *a,
1080 union iwreq_data *wrqu, char *extra)
1081{
1082 u8 _status = false;
1083 int ret = 0;
1084 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1085 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
1086 struct ndis_802_11_ssid ssid[RTW_SSID_SCAN_AMOUNT];
Larry Fingera2c60d42013-08-21 22:33:58 -05001087 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_set_scan\n"));
1088
Larry Fingera2c60d42013-08-21 22:33:58 -05001089 if (padapter->registrypriv.mp_mode == 1) {
1090 if (check_fwstate(pmlmepriv, WIFI_MP_STATE)) {
1091 ret = -1;
1092 goto exit;
1093 }
1094 }
1095 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1096 ret = -1;
1097 goto exit;
1098 }
1099
1100 if (padapter->bDriverStopped) {
1101 DBG_88E("bDriverStopped =%d\n", padapter->bDriverStopped);
1102 ret = -1;
1103 goto exit;
1104 }
1105
1106 if (!padapter->bup) {
1107 ret = -1;
1108 goto exit;
1109 }
1110
1111 if (!padapter->hw_init_completed) {
1112 ret = -1;
1113 goto exit;
1114 }
1115
1116 /* When Busy Traffic, driver do not site survey. So driver return success. */
1117 /* wpa_supplicant will not issue SIOCSIWSCAN cmd again after scan timeout. */
1118 /* modify by thomas 2011-02-22. */
1119 if (pmlmepriv->LinkDetectInfo.bBusyTraffic) {
1120 indicate_wx_scan_complete_event(padapter);
1121 goto exit;
1122 }
1123
1124 if (check_fwstate(pmlmepriv, _FW_UNDER_SURVEY|_FW_UNDER_LINKING)) {
1125 indicate_wx_scan_complete_event(padapter);
1126 goto exit;
1127 }
1128
1129/* For the DMP WiFi Display project, the driver won't to scan because */
1130/* the pmlmepriv->scan_interval is always equal to 3. */
1131/* So, the wpa_supplicant won't find out the WPS SoftAP. */
1132
navin patidar1ce39842014-06-22 13:49:28 +05301133 memset(ssid, 0, sizeof(struct ndis_802_11_ssid)*RTW_SSID_SCAN_AMOUNT);
Larry Fingera2c60d42013-08-21 22:33:58 -05001134
1135 if (wrqu->data.length == sizeof(struct iw_scan_req)) {
1136 struct iw_scan_req *req = (struct iw_scan_req *)extra;
1137
1138 if (wrqu->data.flags & IW_SCAN_THIS_ESSID) {
Gulsah Kose3fd511d2014-10-30 03:03:46 +02001139 int len = min_t(int, req->essid_len,
1140 IW_ESSID_MAX_SIZE);
Larry Fingera2c60d42013-08-21 22:33:58 -05001141
1142 memcpy(ssid[0].Ssid, req->essid, len);
1143 ssid[0].SsidLength = len;
1144
1145 DBG_88E("IW_SCAN_THIS_ESSID, ssid =%s, len =%d\n", req->essid, req->essid_len);
1146
Larry Finger7057dcb2013-12-19 22:38:34 -06001147 spin_lock_bh(&pmlmepriv->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001148
1149 _status = rtw_sitesurvey_cmd(padapter, ssid, 1, NULL, 0);
1150
Larry Fingere02bcf62013-12-19 22:38:35 -06001151 spin_unlock_bh(&pmlmepriv->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001152 } else if (req->scan_type == IW_SCAN_TYPE_PASSIVE) {
1153 DBG_88E("rtw_wx_set_scan, req->scan_type == IW_SCAN_TYPE_PASSIVE\n");
1154 }
1155 } else {
1156 if (wrqu->data.length >= WEXT_CSCAN_HEADER_SIZE &&
1157 !memcmp(extra, WEXT_CSCAN_HEADER, WEXT_CSCAN_HEADER_SIZE)) {
1158 int len = wrqu->data.length - WEXT_CSCAN_HEADER_SIZE;
1159 char *pos = extra+WEXT_CSCAN_HEADER_SIZE;
1160 char section;
1161 char sec_len;
1162 int ssid_index = 0;
1163
1164 while (len >= 1) {
1165 section = *(pos++);
1166 len -= 1;
1167
1168 switch (section) {
1169 case WEXT_CSCAN_SSID_SECTION:
1170 if (len < 1) {
1171 len = 0;
1172 break;
1173 }
1174 sec_len = *(pos++); len -= 1;
1175 if (sec_len > 0 && sec_len <= len) {
1176 ssid[ssid_index].SsidLength = sec_len;
1177 memcpy(ssid[ssid_index].Ssid, pos, ssid[ssid_index].SsidLength);
1178 ssid_index++;
1179 }
1180 pos += sec_len;
1181 len -= sec_len;
1182 break;
1183 case WEXT_CSCAN_TYPE_SECTION:
1184 case WEXT_CSCAN_CHANNEL_SECTION:
1185 pos += 1;
1186 len -= 1;
1187 break;
1188 case WEXT_CSCAN_PASV_DWELL_SECTION:
1189 case WEXT_CSCAN_HOME_DWELL_SECTION:
1190 case WEXT_CSCAN_ACTV_DWELL_SECTION:
1191 pos += 2;
1192 len -= 2;
1193 break;
1194 default:
1195 len = 0; /* stop parsing */
1196 }
1197 }
1198
Masanari Iida7efc02c2013-09-27 00:11:46 +09001199 /* it has still some scan parameter to parse, we only do this now... */
Larry Fingera2c60d42013-08-21 22:33:58 -05001200 _status = rtw_set_802_11_bssid_list_scan(padapter, ssid, RTW_SSID_SCAN_AMOUNT);
1201 } else {
1202 _status = rtw_set_802_11_bssid_list_scan(padapter, NULL, 0);
1203 }
1204 }
1205
1206 if (!_status)
1207 ret = -1;
1208
1209exit:
1210
Larry Fingera2c60d42013-08-21 22:33:58 -05001211 return ret;
1212}
1213
1214static int rtw_wx_get_scan(struct net_device *dev, struct iw_request_info *a,
1215 union iwreq_data *wrqu, char *extra)
1216{
Larry Fingera2c60d42013-08-21 22:33:58 -05001217 struct list_head *plist, *phead;
1218 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1219 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1220 struct __queue *queue = &(pmlmepriv->scanned_queue);
1221 struct wlan_network *pnetwork = NULL;
1222 char *ev = extra;
1223 char *stop = ev + wrqu->data.length;
1224 u32 ret = 0;
1225 u32 cnt = 0;
1226 u32 wait_for_surveydone;
1227 int wait_status;
Larry Fingera2c60d42013-08-21 22:33:58 -05001228 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan\n"));
1229 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, (" Start of Query SIOCGIWSCAN .\n"));
1230
Larry Fingera2c60d42013-08-21 22:33:58 -05001231 if (padapter->pwrctrlpriv.brfoffbyhw && padapter->bDriverStopped) {
1232 ret = -EINVAL;
1233 goto exit;
1234 }
1235
navin patidar2454e792014-07-10 09:11:27 +05301236 wait_for_surveydone = 100;
Larry Fingera2c60d42013-08-21 22:33:58 -05001237
1238 wait_status = _FW_UNDER_SURVEY | _FW_UNDER_LINKING;
1239
1240 while (check_fwstate(pmlmepriv, wait_status)) {
Larry Finger0da46e62013-12-19 22:38:41 -06001241 msleep(30);
Larry Fingera2c60d42013-08-21 22:33:58 -05001242 cnt++;
1243 if (cnt > wait_for_surveydone)
1244 break;
1245 }
1246
Larry Finger7057dcb2013-12-19 22:38:34 -06001247 spin_lock_bh(&(pmlmepriv->scanned_queue.lock));
Larry Fingera2c60d42013-08-21 22:33:58 -05001248
1249 phead = get_list_head(queue);
Larry Fingerc44e5e32014-02-09 15:15:58 -06001250 plist = phead->next;
Larry Fingera2c60d42013-08-21 22:33:58 -05001251
navin patidar84660702014-06-22 13:49:32 +05301252 while (phead != plist) {
Larry Fingera2c60d42013-08-21 22:33:58 -05001253 if ((stop - ev) < SCAN_ITEM_SIZE) {
1254 ret = -E2BIG;
1255 break;
1256 }
1257
Larry Fingerbea88102014-02-09 15:15:57 -06001258 pnetwork = container_of(plist, struct wlan_network, list);
Larry Fingera2c60d42013-08-21 22:33:58 -05001259
1260 /* report network only if the current channel set contains the channel to which this network belongs */
1261 if (rtw_ch_set_search_ch(padapter->mlmeextpriv.channel_set, pnetwork->network.Configuration.DSConfig) >= 0)
1262 ev = translate_scan(padapter, a, pnetwork, ev, stop);
1263
Larry Fingerc44e5e32014-02-09 15:15:58 -06001264 plist = plist->next;
Larry Fingera2c60d42013-08-21 22:33:58 -05001265 }
1266
Larry Fingere02bcf62013-12-19 22:38:35 -06001267 spin_unlock_bh(&pmlmepriv->scanned_queue.lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001268
1269 wrqu->data.length = ev-extra;
1270 wrqu->data.flags = 0;
1271
1272exit:
Larry Fingera2c60d42013-08-21 22:33:58 -05001273 return ret;
1274}
1275
1276/* set ssid flow */
1277/* s1. rtw_set_802_11_infrastructure_mode() */
1278/* s2. set_802_11_authenticaion_mode() */
1279/* s3. set_802_11_encryption_mode() */
1280/* s4. rtw_set_802_11_ssid() */
1281static int rtw_wx_set_essid(struct net_device *dev,
1282 struct iw_request_info *a,
1283 union iwreq_data *wrqu, char *extra)
1284{
Larry Fingera2c60d42013-08-21 22:33:58 -05001285 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1286 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
1287 struct __queue *queue = &pmlmepriv->scanned_queue;
1288 struct list_head *phead;
1289 struct wlan_network *pnetwork = NULL;
1290 enum ndis_802_11_auth_mode authmode;
1291 struct ndis_802_11_ssid ndis_ssid;
1292 u8 *dst_ssid, *src_ssid;
1293
1294 uint ret = 0, len;
1295
Larry Fingera2c60d42013-08-21 22:33:58 -05001296
1297 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1298 ("+rtw_wx_set_essid: fw_state = 0x%08x\n", get_fwstate(pmlmepriv)));
1299 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1300 ret = -1;
1301 goto exit;
1302 }
1303
1304 if (!padapter->bup) {
1305 ret = -1;
1306 goto exit;
1307 }
1308
1309 if (wrqu->essid.length > IW_ESSID_MAX_SIZE) {
1310 ret = -E2BIG;
1311 goto exit;
1312 }
1313
1314 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
1315 ret = -1;
1316 goto exit;
1317 }
1318
1319 authmode = padapter->securitypriv.ndisauthtype;
1320 DBG_88E("=>%s\n", __func__);
1321 if (wrqu->essid.flags && wrqu->essid.length) {
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07001322 len = min_t(uint, wrqu->essid.length, IW_ESSID_MAX_SIZE);
Larry Fingera2c60d42013-08-21 22:33:58 -05001323
1324 if (wrqu->essid.length != 33)
1325 DBG_88E("ssid =%s, len =%d\n", extra, wrqu->essid.length);
1326
navin patidar1ce39842014-06-22 13:49:28 +05301327 memset(&ndis_ssid, 0, sizeof(struct ndis_802_11_ssid));
Larry Fingera2c60d42013-08-21 22:33:58 -05001328 ndis_ssid.SsidLength = len;
1329 memcpy(ndis_ssid.Ssid, extra, len);
1330 src_ssid = ndis_ssid.Ssid;
1331
1332 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("rtw_wx_set_essid: ssid =[%s]\n", src_ssid));
Larry Finger7057dcb2013-12-19 22:38:34 -06001333 spin_lock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001334 phead = get_list_head(queue);
Larry Fingerc44e5e32014-02-09 15:15:58 -06001335 pmlmepriv->pscanned = phead->next;
Larry Fingera2c60d42013-08-21 22:33:58 -05001336
navin patidar84660702014-06-22 13:49:32 +05301337 while (phead != pmlmepriv->pscanned) {
Larry Fingerbea88102014-02-09 15:15:57 -06001338 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
Larry Fingera2c60d42013-08-21 22:33:58 -05001339
Larry Fingerc44e5e32014-02-09 15:15:58 -06001340 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
Larry Fingera2c60d42013-08-21 22:33:58 -05001341
1342 dst_ssid = pnetwork->network.Ssid.Ssid;
1343
1344 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1345 ("rtw_wx_set_essid: dst_ssid =%s\n",
1346 pnetwork->network.Ssid.Ssid));
1347
1348 if ((!memcmp(dst_ssid, src_ssid, ndis_ssid.SsidLength)) &&
1349 (pnetwork->network.Ssid.SsidLength == ndis_ssid.SsidLength)) {
1350 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1351 ("rtw_wx_set_essid: find match, set infra mode\n"));
1352
1353 if (check_fwstate(pmlmepriv, WIFI_ADHOC_STATE) == true) {
1354 if (pnetwork->network.InfrastructureMode != pmlmepriv->cur_network.network.InfrastructureMode)
1355 continue;
1356 }
1357
1358 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
1359 ret = -1;
Larry Fingere02bcf62013-12-19 22:38:35 -06001360 spin_unlock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001361 goto exit;
1362 }
1363
1364 break;
1365 }
1366 }
Larry Fingere02bcf62013-12-19 22:38:35 -06001367 spin_unlock_bh(&queue->lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05001368 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1369 ("set ssid: set_802_11_auth. mode =%d\n", authmode));
1370 rtw_set_802_11_authentication_mode(padapter, authmode);
1371 if (rtw_set_802_11_ssid(padapter, &ndis_ssid) == false) {
1372 ret = -1;
1373 goto exit;
1374 }
1375 }
1376
1377exit:
1378
1379 DBG_88E("<=%s, ret %d\n", __func__, ret);
1380
Larry Fingera2c60d42013-08-21 22:33:58 -05001381
1382 return ret;
1383}
1384
1385static int rtw_wx_get_essid(struct net_device *dev,
1386 struct iw_request_info *a,
1387 union iwreq_data *wrqu, char *extra)
1388{
1389 u32 len, ret = 0;
1390 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1391 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1392 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
1393
1394 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_essid\n"));
1395
Larry Fingera2c60d42013-08-21 22:33:58 -05001396
1397 if ((check_fwstate(pmlmepriv, _FW_LINKED)) ||
1398 (check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE))) {
1399 len = pcur_bss->Ssid.SsidLength;
Larry Fingera2c60d42013-08-21 22:33:58 -05001400 memcpy(extra, pcur_bss->Ssid.Ssid, len);
Larry Fingera2c60d42013-08-21 22:33:58 -05001401 } else {
Larry Fingerace19112017-11-25 13:32:38 -06001402 len = 0;
1403 *extra = 0;
Larry Fingera2c60d42013-08-21 22:33:58 -05001404 }
Larry Fingerace19112017-11-25 13:32:38 -06001405 wrqu->essid.length = len;
1406 wrqu->essid.flags = 1;
Larry Fingera2c60d42013-08-21 22:33:58 -05001407
1408 return ret;
1409}
1410
1411static int rtw_wx_set_rate(struct net_device *dev,
1412 struct iw_request_info *a,
1413 union iwreq_data *wrqu, char *extra)
1414{
Sudip Mukherjeec78a9642014-11-07 16:50:45 +05301415 int i;
Larry Fingera2c60d42013-08-21 22:33:58 -05001416 u8 datarates[NumRates];
1417 u32 target_rate = wrqu->bitrate.value;
1418 u32 fixed = wrqu->bitrate.fixed;
1419 u32 ratevalue = 0;
1420 u8 mpdatarate[NumRates] = {11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0xff};
1421
Larry Fingera2c60d42013-08-21 22:33:58 -05001422
1423 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_set_rate\n"));
1424 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("target_rate = %d, fixed = %d\n", target_rate, fixed));
1425
1426 if (target_rate == -1) {
1427 ratevalue = 11;
1428 goto set_rate;
1429 }
1430 target_rate = target_rate/100000;
1431
1432 switch (target_rate) {
1433 case 10:
1434 ratevalue = 0;
1435 break;
1436 case 20:
1437 ratevalue = 1;
1438 break;
1439 case 55:
1440 ratevalue = 2;
1441 break;
1442 case 60:
1443 ratevalue = 3;
1444 break;
1445 case 90:
1446 ratevalue = 4;
1447 break;
1448 case 110:
1449 ratevalue = 5;
1450 break;
1451 case 120:
1452 ratevalue = 6;
1453 break;
1454 case 180:
1455 ratevalue = 7;
1456 break;
1457 case 240:
1458 ratevalue = 8;
1459 break;
1460 case 360:
1461 ratevalue = 9;
1462 break;
1463 case 480:
1464 ratevalue = 10;
1465 break;
1466 case 540:
1467 ratevalue = 11;
1468 break;
1469 default:
1470 ratevalue = 11;
1471 break;
1472 }
1473
1474set_rate:
1475
1476 for (i = 0; i < NumRates; i++) {
1477 if (ratevalue == mpdatarate[i]) {
1478 datarates[i] = mpdatarate[i];
1479 if (fixed == 0)
1480 break;
1481 } else {
1482 datarates[i] = 0xff;
1483 }
1484
1485 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("datarate_inx =%d\n", datarates[i]));
1486 }
1487
Sudip Mukherjeec78a9642014-11-07 16:50:45 +05301488 return 0;
Larry Fingera2c60d42013-08-21 22:33:58 -05001489}
1490
1491static int rtw_wx_get_rate(struct net_device *dev,
1492 struct iw_request_info *info,
1493 union iwreq_data *wrqu, char *extra)
1494{
1495 u16 max_rate = 0;
1496
1497 max_rate = rtw_get_cur_max_rate((struct adapter *)rtw_netdev_priv(dev));
1498
1499 if (max_rate == 0)
1500 return -EPERM;
1501
1502 wrqu->bitrate.fixed = 0; /* no auto select */
1503 wrqu->bitrate.value = max_rate * 100000;
1504
1505 return 0;
1506}
1507
1508static int rtw_wx_set_rts(struct net_device *dev,
1509 struct iw_request_info *info,
1510 union iwreq_data *wrqu, char *extra)
1511{
1512 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1513
Larry Fingera2c60d42013-08-21 22:33:58 -05001514
1515 if (wrqu->rts.disabled) {
1516 padapter->registrypriv.rts_thresh = 2347;
1517 } else {
1518 if (wrqu->rts.value < 0 ||
1519 wrqu->rts.value > 2347)
1520 return -EINVAL;
1521
1522 padapter->registrypriv.rts_thresh = wrqu->rts.value;
1523 }
1524
1525 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1526
Larry Fingera2c60d42013-08-21 22:33:58 -05001527
1528 return 0;
1529}
1530
1531static int rtw_wx_get_rts(struct net_device *dev,
1532 struct iw_request_info *info,
1533 union iwreq_data *wrqu, char *extra)
1534{
1535 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1536
Larry Fingera2c60d42013-08-21 22:33:58 -05001537
1538 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1539
1540 wrqu->rts.value = padapter->registrypriv.rts_thresh;
1541 wrqu->rts.fixed = 0; /* no auto select */
1542 /* wrqu->rts.disabled = (wrqu->rts.value == DEFAULT_RTS_THRESHOLD); */
1543
Larry Fingera2c60d42013-08-21 22:33:58 -05001544
1545 return 0;
1546}
1547
1548static int rtw_wx_set_frag(struct net_device *dev,
1549 struct iw_request_info *info,
1550 union iwreq_data *wrqu, char *extra)
1551{
1552 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1553
Larry Fingera2c60d42013-08-21 22:33:58 -05001554
1555 if (wrqu->frag.disabled) {
1556 padapter->xmitpriv.frag_len = MAX_FRAG_THRESHOLD;
1557 } else {
1558 if (wrqu->frag.value < MIN_FRAG_THRESHOLD ||
1559 wrqu->frag.value > MAX_FRAG_THRESHOLD)
1560 return -EINVAL;
1561
1562 padapter->xmitpriv.frag_len = wrqu->frag.value & ~0x1;
1563 }
1564
1565 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1566
Larry Fingera2c60d42013-08-21 22:33:58 -05001567
1568 return 0;
1569}
1570
1571static int rtw_wx_get_frag(struct net_device *dev,
1572 struct iw_request_info *info,
1573 union iwreq_data *wrqu, char *extra)
1574{
1575 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1576
Larry Fingera2c60d42013-08-21 22:33:58 -05001577
1578 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1579
1580 wrqu->frag.value = padapter->xmitpriv.frag_len;
1581 wrqu->frag.fixed = 0; /* no auto select */
1582
Larry Fingera2c60d42013-08-21 22:33:58 -05001583
1584 return 0;
1585}
1586
1587static int rtw_wx_get_retry(struct net_device *dev,
1588 struct iw_request_info *info,
1589 union iwreq_data *wrqu, char *extra)
1590{
1591 wrqu->retry.value = 7;
1592 wrqu->retry.fixed = 0; /* no auto select */
1593 wrqu->retry.disabled = 1;
1594
1595 return 0;
1596}
1597
1598static int rtw_wx_set_enc(struct net_device *dev,
1599 struct iw_request_info *info,
1600 union iwreq_data *wrqu, char *keybuf)
1601{
1602 u32 key, ret = 0;
1603 u32 keyindex_provided;
1604 struct ndis_802_11_wep wep;
1605 enum ndis_802_11_auth_mode authmode;
1606
1607 struct iw_point *erq = &(wrqu->encoding);
1608 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1609 struct pwrctrl_priv *pwrpriv = &padapter->pwrctrlpriv;
1610 DBG_88E("+rtw_wx_set_enc, flags = 0x%x\n", erq->flags);
1611
navin patidar1ce39842014-06-22 13:49:28 +05301612 memset(&wep, 0, sizeof(struct ndis_802_11_wep));
Larry Fingera2c60d42013-08-21 22:33:58 -05001613
1614 key = erq->flags & IW_ENCODE_INDEX;
1615
Larry Fingera2c60d42013-08-21 22:33:58 -05001616
1617 if (erq->flags & IW_ENCODE_DISABLED) {
1618 DBG_88E("EncryptionDisabled\n");
1619 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1620 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1621 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
Luca Ceresoli2ff4e792015-04-16 23:28:03 +02001622 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
Larry Fingera2c60d42013-08-21 22:33:58 -05001623 authmode = Ndis802_11AuthModeOpen;
1624 padapter->securitypriv.ndisauthtype = authmode;
1625
1626 goto exit;
1627 }
1628
1629 if (key) {
1630 if (key > WEP_KEYS)
1631 return -EINVAL;
1632 key--;
1633 keyindex_provided = 1;
1634 } else {
1635 keyindex_provided = 0;
1636 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1637 DBG_88E("rtw_wx_set_enc, key =%d\n", key);
1638 }
1639
1640 /* set authentication mode */
1641 if (erq->flags & IW_ENCODE_OPEN) {
1642 DBG_88E("rtw_wx_set_enc():IW_ENCODE_OPEN\n");
1643 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
1644 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1645 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1646 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1647 authmode = Ndis802_11AuthModeOpen;
1648 padapter->securitypriv.ndisauthtype = authmode;
1649 } else if (erq->flags & IW_ENCODE_RESTRICTED) {
1650 DBG_88E("rtw_wx_set_enc():IW_ENCODE_RESTRICTED\n");
1651 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
1652 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Shared;
1653 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1654 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
1655 authmode = Ndis802_11AuthModeShared;
1656 padapter->securitypriv.ndisauthtype = authmode;
1657 } else {
1658 DBG_88E("rtw_wx_set_enc():erq->flags = 0x%x\n", erq->flags);
1659
1660 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
Luca Ceresoli2ff4e792015-04-16 23:28:03 +02001661 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
Larry Fingera2c60d42013-08-21 22:33:58 -05001662 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1663 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1664 authmode = Ndis802_11AuthModeOpen;
1665 padapter->securitypriv.ndisauthtype = authmode;
1666 }
1667
1668 wep.KeyIndex = key;
1669 if (erq->length > 0) {
1670 wep.KeyLength = erq->length <= 5 ? 5 : 13;
1671
Ivan Safonov5e3027b2016-03-02 14:59:01 +07001672 wep.Length = wep.KeyLength + offsetof(struct ndis_802_11_wep, KeyMaterial);
Larry Fingera2c60d42013-08-21 22:33:58 -05001673 } else {
1674 wep.KeyLength = 0;
1675
1676 if (keyindex_provided == 1) {
1677 /* set key_id only, no given KeyMaterial(erq->length == 0). */
1678 padapter->securitypriv.dot11PrivacyKeyIndex = key;
1679
1680 DBG_88E("(keyindex_provided == 1), keyid =%d, key_len =%d\n", key, padapter->securitypriv.dot11DefKeylen[key]);
1681
1682 switch (padapter->securitypriv.dot11DefKeylen[key]) {
1683 case 5:
1684 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1685 break;
1686 case 13:
1687 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
1688 break;
1689 default:
1690 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1691 break;
1692 }
1693
1694 goto exit;
1695 }
1696 }
1697
1698 wep.KeyIndex |= 0x80000000;
1699
1700 memcpy(wep.KeyMaterial, keybuf, wep.KeyLength);
1701
1702 if (rtw_set_802_11_add_wep(padapter, &wep) == false) {
1703 if (rf_on == pwrpriv->rf_pwrstate)
1704 ret = -EOPNOTSUPP;
1705 goto exit;
1706 }
1707
1708exit:
1709
Larry Fingera2c60d42013-08-21 22:33:58 -05001710
1711 return ret;
1712}
1713
1714static int rtw_wx_get_enc(struct net_device *dev,
1715 struct iw_request_info *info,
1716 union iwreq_data *wrqu, char *keybuf)
1717{
1718 uint key, ret = 0;
1719 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1720 struct iw_point *erq = &(wrqu->encoding);
1721 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1722
Larry Fingera2c60d42013-08-21 22:33:58 -05001723
1724 if (check_fwstate(pmlmepriv, _FW_LINKED) != true) {
1725 if (!check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) {
1726 erq->length = 0;
1727 erq->flags |= IW_ENCODE_DISABLED;
1728 return 0;
1729 }
1730 }
1731
1732 key = erq->flags & IW_ENCODE_INDEX;
1733
1734 if (key) {
1735 if (key > WEP_KEYS)
1736 return -EINVAL;
1737 key--;
1738 } else {
1739 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1740 }
1741
1742 erq->flags = key + 1;
1743
1744 switch (padapter->securitypriv.ndisencryptstatus) {
1745 case Ndis802_11EncryptionNotSupported:
1746 case Ndis802_11EncryptionDisabled:
1747 erq->length = 0;
1748 erq->flags |= IW_ENCODE_DISABLED;
1749 break;
1750 case Ndis802_11Encryption1Enabled:
1751 erq->length = padapter->securitypriv.dot11DefKeylen[key];
1752 if (erq->length) {
1753 memcpy(keybuf, padapter->securitypriv.dot11DefKey[key].skey, padapter->securitypriv.dot11DefKeylen[key]);
1754
1755 erq->flags |= IW_ENCODE_ENABLED;
1756
1757 if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeOpen)
1758 erq->flags |= IW_ENCODE_OPEN;
1759 else if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeShared)
1760 erq->flags |= IW_ENCODE_RESTRICTED;
1761 } else {
1762 erq->length = 0;
1763 erq->flags |= IW_ENCODE_DISABLED;
1764 }
1765 break;
1766 case Ndis802_11Encryption2Enabled:
1767 case Ndis802_11Encryption3Enabled:
1768 erq->length = 16;
1769 erq->flags |= (IW_ENCODE_ENABLED | IW_ENCODE_OPEN | IW_ENCODE_NOKEY);
1770 break;
1771 default:
1772 erq->length = 0;
1773 erq->flags |= IW_ENCODE_DISABLED;
1774 break;
1775 }
Larry Fingera2c60d42013-08-21 22:33:58 -05001776
1777 return ret;
1778}
1779
1780static int rtw_wx_get_power(struct net_device *dev,
1781 struct iw_request_info *info,
1782 union iwreq_data *wrqu, char *extra)
1783{
1784 wrqu->power.value = 0;
1785 wrqu->power.fixed = 0; /* no auto select */
1786 wrqu->power.disabled = 1;
1787
1788 return 0;
1789}
1790
1791static int rtw_wx_set_gen_ie(struct net_device *dev,
1792 struct iw_request_info *info,
1793 union iwreq_data *wrqu, char *extra)
1794{
Larry Fingera2c60d42013-08-21 22:33:58 -05001795 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1796
Aya Mahfouz5f703062015-02-27 14:54:23 +02001797 return rtw_set_wpa_ie(padapter, extra, wrqu->data.length);
Larry Fingera2c60d42013-08-21 22:33:58 -05001798}
1799
1800static int rtw_wx_set_auth(struct net_device *dev,
1801 struct iw_request_info *info,
1802 union iwreq_data *wrqu, char *extra)
1803{
1804 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1805 struct iw_param *param = (struct iw_param *)&(wrqu->param);
1806 int ret = 0;
1807
1808 switch (param->flags & IW_AUTH_INDEX) {
1809 case IW_AUTH_WPA_VERSION:
1810 break;
1811 case IW_AUTH_CIPHER_PAIRWISE:
1812
1813 break;
1814 case IW_AUTH_CIPHER_GROUP:
1815
1816 break;
1817 case IW_AUTH_KEY_MGMT:
1818 /*
1819 * ??? does not use these parameters
1820 */
1821 break;
1822 case IW_AUTH_TKIP_COUNTERMEASURES:
1823 if (param->value) {
1824 /* wpa_supplicant is enabling the tkip countermeasure. */
1825 padapter->securitypriv.btkip_countermeasure = true;
1826 } else {
1827 /* wpa_supplicant is disabling the tkip countermeasure. */
1828 padapter->securitypriv.btkip_countermeasure = false;
1829 }
1830 break;
1831 case IW_AUTH_DROP_UNENCRYPTED:
1832 /* HACK:
1833 *
1834 * wpa_supplicant calls set_wpa_enabled when the driver
1835 * is loaded and unloaded, regardless of if WPA is being
1836 * used. No other calls are made which can be used to
1837 * determine if encryption will be used or not prior to
1838 * association being expected. If encryption is not being
1839 * used, drop_unencrypted is set to false, else true -- we
1840 * can use this to determine if the CAP_PRIVACY_ON bit should
1841 * be set.
1842 */
1843
1844 if (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption1Enabled)
1845 break;/* it means init value, or using wep, ndisencryptstatus = Ndis802_11Encryption1Enabled, */
1846 /* then it needn't reset it; */
1847
1848 if (param->value) {
1849 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1850 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1851 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
Luca Ceresoli2ff4e792015-04-16 23:28:03 +02001852 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
Larry Fingera2c60d42013-08-21 22:33:58 -05001853 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeOpen;
1854 }
1855
1856 break;
1857 case IW_AUTH_80211_AUTH_ALG:
1858 /*
1859 * It's the starting point of a link layer connection using wpa_supplicant
1860 */
1861 if (check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
1862 LeaveAllPowerSaveMode(padapter);
1863 rtw_disassoc_cmd(padapter, 500, false);
1864 DBG_88E("%s...call rtw_indicate_disconnect\n ", __func__);
1865 rtw_indicate_disconnect(padapter);
Buţiu Alexandru Octavianb4ba3b52015-07-24 00:13:27 +03001866 rtw_free_assoc_resources(padapter);
Larry Fingera2c60d42013-08-21 22:33:58 -05001867 }
1868 ret = wpa_set_auth_algs(dev, (u32)param->value);
1869 break;
1870 case IW_AUTH_WPA_ENABLED:
1871 break;
1872 case IW_AUTH_RX_UNENCRYPTED_EAPOL:
1873 break;
1874 case IW_AUTH_PRIVACY_INVOKED:
1875 break;
1876 default:
1877 return -EOPNOTSUPP;
1878 }
1879
1880 return ret;
1881}
1882
1883static int rtw_wx_set_enc_ext(struct net_device *dev,
1884 struct iw_request_info *info,
1885 union iwreq_data *wrqu, char *extra)
1886{
1887 char *alg_name;
1888 u32 param_len;
1889 struct ieee_param *param = NULL;
1890 struct iw_point *pencoding = &wrqu->encoding;
1891 struct iw_encode_ext *pext = (struct iw_encode_ext *)extra;
1892 int ret = 0;
1893
1894 param_len = sizeof(struct ieee_param) + pext->key_len;
1895 param = (struct ieee_param *)rtw_malloc(param_len);
Amit Ghadge4fe514d2016-09-04 22:53:35 +05301896 if (!param)
Larry Fingera2c60d42013-08-21 22:33:58 -05001897 return -1;
1898
navin patidar1ce39842014-06-22 13:49:28 +05301899 memset(param, 0, param_len);
Larry Fingera2c60d42013-08-21 22:33:58 -05001900
1901 param->cmd = IEEE_CMD_SET_ENCRYPTION;
Hari Prasath Gujulan Elangoa3124e72016-01-20 12:52:58 +00001902 eth_broadcast_addr(param->sta_addr);
Larry Fingera2c60d42013-08-21 22:33:58 -05001903
1904 switch (pext->alg) {
1905 case IW_ENCODE_ALG_NONE:
1906 /* todo: remove key */
1907 /* remove = 1; */
1908 alg_name = "none";
1909 break;
1910 case IW_ENCODE_ALG_WEP:
1911 alg_name = "WEP";
1912 break;
1913 case IW_ENCODE_ALG_TKIP:
1914 alg_name = "TKIP";
1915 break;
1916 case IW_ENCODE_ALG_CCMP:
1917 alg_name = "CCMP";
1918 break;
1919 default:
Christian Engelmayer5d57f1e2014-05-07 21:31:20 +02001920 ret = -1;
1921 goto exit;
Larry Fingera2c60d42013-08-21 22:33:58 -05001922 }
1923
1924 strncpy((char *)param->u.crypt.alg, alg_name, IEEE_CRYPT_ALG_NAME_LEN);
1925
1926 if (pext->ext_flags & IW_ENCODE_EXT_SET_TX_KEY)
1927 param->u.crypt.set_tx = 1;
1928
1929 /* cliW: WEP does not have group key
1930 * just not checking GROUP key setting
1931 */
1932 if ((pext->alg != IW_ENCODE_ALG_WEP) &&
1933 (pext->ext_flags & IW_ENCODE_EXT_GROUP_KEY))
1934 param->u.crypt.set_tx = 0;
1935
1936 param->u.crypt.idx = (pencoding->flags&0x00FF) - 1;
1937
1938 if (pext->ext_flags & IW_ENCODE_EXT_RX_SEQ_VALID)
1939 memcpy(param->u.crypt.seq, pext->rx_seq, 8);
1940
1941 if (pext->key_len) {
1942 param->u.crypt.key_len = pext->key_len;
1943 memcpy(param->u.crypt.key, pext + 1, pext->key_len);
1944 }
1945
1946 ret = wpa_set_encryption(dev, param, param_len);
1947
Christian Engelmayer5d57f1e2014-05-07 21:31:20 +02001948exit:
Larry Fingera2c60d42013-08-21 22:33:58 -05001949 kfree(param);
1950 return ret;
1951}
1952
1953static int rtw_wx_get_nick(struct net_device *dev,
1954 struct iw_request_info *info,
1955 union iwreq_data *wrqu, char *extra)
1956{
1957 if (extra) {
1958 wrqu->data.length = 14;
1959 wrqu->data.flags = 1;
1960 memcpy(extra, "<WIFI@REALTEK>", 14);
1961 }
1962
1963 /* dump debug info here */
1964 return 0;
1965}
1966
Larry Fingera2c60d42013-08-21 22:33:58 -05001967static int dummy(struct net_device *dev, struct iw_request_info *a,
1968 union iwreq_data *wrqu, char *b)
1969{
1970 return -1;
1971}
1972
Larry Fingera2c60d42013-08-21 22:33:58 -05001973static int wpa_set_param(struct net_device *dev, u8 name, u32 value)
1974{
1975 uint ret = 0;
1976 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1977
1978 switch (name) {
1979 case IEEE_PARAM_WPA_ENABLED:
1980 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X; /* 802.1x */
1981 switch ((value)&0xff) {
1982 case 1: /* WPA */
1983 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPAPSK; /* WPA_PSK */
1984 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
1985 break;
1986 case 2: /* WPA2 */
1987 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPA2PSK; /* WPA2_PSK */
1988 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
1989 break;
1990 }
1991 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1992 ("wpa_set_param:padapter->securitypriv.ndisauthtype =%d\n", padapter->securitypriv.ndisauthtype));
1993 break;
1994 case IEEE_PARAM_TKIP_COUNTERMEASURES:
1995 break;
1996 case IEEE_PARAM_DROP_UNENCRYPTED: {
1997 /* HACK:
1998 *
1999 * wpa_supplicant calls set_wpa_enabled when the driver
2000 * is loaded and unloaded, regardless of if WPA is being
2001 * used. No other calls are made which can be used to
2002 * determine if encryption will be used or not prior to
2003 * association being expected. If encryption is not being
2004 * used, drop_unencrypted is set to false, else true -- we
2005 * can use this to determine if the CAP_PRIVACY_ON bit should
2006 * be set.
2007 */
2008
2009 break;
2010 }
2011 case IEEE_PARAM_PRIVACY_INVOKED:
2012 break;
2013
2014 case IEEE_PARAM_AUTH_ALGS:
2015 ret = wpa_set_auth_algs(dev, value);
2016 break;
2017 case IEEE_PARAM_IEEE_802_1X:
2018 break;
2019 case IEEE_PARAM_WPAX_SELECT:
2020 break;
2021 default:
2022 ret = -EOPNOTSUPP;
2023 break;
2024 }
2025 return ret;
2026}
2027
2028static int wpa_mlme(struct net_device *dev, u32 command, u32 reason)
2029{
2030 int ret = 0;
2031 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2032
2033 switch (command) {
2034 case IEEE_MLME_STA_DEAUTH:
2035 if (!rtw_set_802_11_disassociate(padapter))
2036 ret = -1;
2037 break;
2038 case IEEE_MLME_STA_DISASSOC:
2039 if (!rtw_set_802_11_disassociate(padapter))
2040 ret = -1;
2041 break;
2042 default:
2043 ret = -EOPNOTSUPP;
2044 break;
2045 }
2046
2047 return ret;
2048}
2049
2050static int wpa_supplicant_ioctl(struct net_device *dev, struct iw_point *p)
2051{
2052 struct ieee_param *param;
2053 uint ret = 0;
2054
Larry Finger3404ad42020-02-10 12:02:32 -06002055 if (!p->pointer || p->length != sizeof(struct ieee_param)) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002056 ret = -EINVAL;
2057 goto out;
2058 }
2059
2060 param = (struct ieee_param *)rtw_malloc(p->length);
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302061 if (!param) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002062 ret = -ENOMEM;
2063 goto out;
2064 }
2065
2066 if (copy_from_user(param, p->pointer, p->length)) {
2067 kfree(param);
2068 ret = -EFAULT;
2069 goto out;
2070 }
2071
2072 switch (param->cmd) {
2073 case IEEE_CMD_SET_WPA_PARAM:
2074 ret = wpa_set_param(dev, param->u.wpa_param.name, param->u.wpa_param.value);
2075 break;
2076
2077 case IEEE_CMD_SET_WPA_IE:
2078 ret = rtw_set_wpa_ie((struct adapter *)rtw_netdev_priv(dev),
2079 (char *)param->u.wpa_ie.data, (u16)param->u.wpa_ie.len);
2080 break;
2081
2082 case IEEE_CMD_SET_ENCRYPTION:
2083 ret = wpa_set_encryption(dev, param, p->length);
2084 break;
2085
2086 case IEEE_CMD_MLME:
2087 ret = wpa_mlme(dev, param->u.mlme.command, param->u.mlme.reason_code);
2088 break;
2089
2090 default:
2091 DBG_88E("Unknown WPA supplicant request: %d\n", param->cmd);
2092 ret = -EOPNOTSUPP;
2093 break;
2094 }
2095
2096 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2097 ret = -EFAULT;
2098
2099 kfree(param);
2100
2101out:
2102
2103 return ret;
2104}
2105
2106#ifdef CONFIG_88EU_AP_MODE
2107static u8 set_pairwise_key(struct adapter *padapter, struct sta_info *psta)
2108{
2109 struct cmd_obj *ph2c;
2110 struct set_stakey_parm *psetstakey_para;
2111 struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
2112 u8 res = _SUCCESS;
2113
navin patidarfadbe0c2014-06-22 14:06:23 +05302114 ph2c = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
Bhaktipriya Shridhar4ada2952016-03-22 20:19:49 +05302115 if (!ph2c) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002116 res = _FAIL;
2117 goto exit;
2118 }
2119
navin patidarfadbe0c2014-06-22 14:06:23 +05302120 psetstakey_para = kzalloc(sizeof(struct set_stakey_parm), GFP_KERNEL);
Bhaktipriya Shridhar4ada2952016-03-22 20:19:49 +05302121 if (!psetstakey_para) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002122 kfree(ph2c);
2123 res = _FAIL;
2124 goto exit;
2125 }
2126
2127 init_h2fwcmd_w_parm_no_rsp(ph2c, psetstakey_para, _SetStaKey_CMD_);
2128
2129 psetstakey_para->algorithm = (u8)psta->dot118021XPrivacy;
2130
2131 memcpy(psetstakey_para->addr, psta->hwaddr, ETH_ALEN);
2132
2133 memcpy(psetstakey_para->key, &psta->dot118021x_UncstKey, 16);
2134
2135 res = rtw_enqueue_cmd(pcmdpriv, ph2c);
2136
2137exit:
2138
2139 return res;
2140}
2141
2142static int set_group_key(struct adapter *padapter, u8 *key, u8 alg, int keyid)
2143{
2144 u8 keylen;
2145 struct cmd_obj *pcmd;
2146 struct setkey_parm *psetkeyparm;
2147 struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
2148 int res = _SUCCESS;
2149
2150 DBG_88E("%s\n", __func__);
2151
navin patidarfadbe0c2014-06-22 14:06:23 +05302152 pcmd = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
Bhaktipriya Shridhar4ada2952016-03-22 20:19:49 +05302153 if (!pcmd) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002154 res = _FAIL;
2155 goto exit;
2156 }
navin patidarfadbe0c2014-06-22 14:06:23 +05302157 psetkeyparm = kzalloc(sizeof(struct setkey_parm), GFP_KERNEL);
Bhaktipriya Shridhar4ada2952016-03-22 20:19:49 +05302158 if (!psetkeyparm) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002159 kfree(pcmd);
2160 res = _FAIL;
2161 goto exit;
2162 }
2163
navin patidar1ce39842014-06-22 13:49:28 +05302164 memset(psetkeyparm, 0, sizeof(struct setkey_parm));
Larry Fingera2c60d42013-08-21 22:33:58 -05002165
2166 psetkeyparm->keyid = (u8)keyid;
2167
2168 psetkeyparm->algorithm = alg;
2169
2170 psetkeyparm->set_tx = 1;
2171
2172 switch (alg) {
2173 case _WEP40_:
2174 keylen = 5;
2175 break;
2176 case _WEP104_:
2177 keylen = 13;
2178 break;
2179 case _TKIP_:
2180 case _TKIP_WTMIC_:
2181 case _AES_:
Larry Fingera2c60d42013-08-21 22:33:58 -05002182 default:
2183 keylen = 16;
2184 }
2185
2186 memcpy(&(psetkeyparm->key[0]), key, keylen);
2187
2188 pcmd->cmdcode = _SetKey_CMD_;
2189 pcmd->parmbuf = (u8 *)psetkeyparm;
2190 pcmd->cmdsz = (sizeof(struct setkey_parm));
2191 pcmd->rsp = NULL;
2192 pcmd->rspsz = 0;
2193
navin patidaraa3f5cc2014-06-22 13:49:34 +05302194 INIT_LIST_HEAD(&pcmd->list);
Larry Fingera2c60d42013-08-21 22:33:58 -05002195
2196 res = rtw_enqueue_cmd(pcmdpriv, pcmd);
2197
2198exit:
2199
2200 return res;
2201}
2202
2203static int set_wep_key(struct adapter *padapter, u8 *key, u8 keylen, int keyid)
2204{
2205 u8 alg;
2206
2207 switch (keylen) {
2208 case 5:
2209 alg = _WEP40_;
2210 break;
2211 case 13:
2212 alg = _WEP104_;
2213 break;
2214 default:
2215 alg = _NO_PRIVACY_;
2216 }
2217
2218 return set_group_key(padapter, key, alg, keyid);
2219}
2220
2221static int rtw_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
2222{
2223 int ret = 0;
2224 u32 wep_key_idx, wep_key_len, wep_total_len;
2225 struct ndis_802_11_wep *pwep = NULL;
2226 struct sta_info *psta = NULL, *pbcmc_sta = NULL;
2227 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2228 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
2229 struct security_priv *psecuritypriv = &(padapter->securitypriv);
2230 struct sta_priv *pstapriv = &padapter->stapriv;
2231
2232 DBG_88E("%s\n", __func__);
2233 param->u.crypt.err = 0;
2234 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
2235 if (param_len != sizeof(struct ieee_param) + param->u.crypt.key_len) {
2236 ret = -EINVAL;
2237 goto exit;
2238 }
2239 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2240 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2241 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff) {
2242 if (param->u.crypt.idx >= WEP_KEYS) {
2243 ret = -EINVAL;
2244 goto exit;
2245 }
2246 } else {
2247 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2248 if (!psta) {
2249 DBG_88E("rtw_set_encryption(), sta has already been removed or never been added\n");
2250 goto exit;
2251 }
2252 }
2253
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302254 if (strcmp(param->u.crypt.alg, "none") == 0 && (!psta)) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002255 /* todo:clear default encryption keys */
2256
2257 DBG_88E("clear default encryption keys, keyid =%d\n", param->u.crypt.idx);
2258 goto exit;
2259 }
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302260 if (strcmp(param->u.crypt.alg, "WEP") == 0 && (!psta)) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002261 DBG_88E("r871x_set_encryption, crypt.alg = WEP\n");
2262 wep_key_idx = param->u.crypt.idx;
2263 wep_key_len = param->u.crypt.key_len;
2264 DBG_88E("r871x_set_encryption, wep_key_idx=%d, len=%d\n", wep_key_idx, wep_key_len);
2265 if ((wep_key_idx >= WEP_KEYS) || (wep_key_len <= 0)) {
2266 ret = -EINVAL;
2267 goto exit;
2268 }
2269
2270 if (wep_key_len > 0) {
2271 wep_key_len = wep_key_len <= 5 ? 5 : 13;
Ivan Safonov5e3027b2016-03-02 14:59:01 +07002272 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
Larry Fingera2c60d42013-08-21 22:33:58 -05002273 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302274 if (!pwep) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002275 DBG_88E(" r871x_set_encryption: pwep allocate fail !!!\n");
2276 goto exit;
2277 }
2278
navin patidar1ce39842014-06-22 13:49:28 +05302279 memset(pwep, 0, wep_total_len);
Larry Fingera2c60d42013-08-21 22:33:58 -05002280
2281 pwep->KeyLength = wep_key_len;
2282 pwep->Length = wep_total_len;
2283 }
2284
2285 pwep->KeyIndex = wep_key_idx;
2286
2287 memcpy(pwep->KeyMaterial, param->u.crypt.key, pwep->KeyLength);
2288
2289 if (param->u.crypt.set_tx) {
2290 DBG_88E("wep, set_tx = 1\n");
2291
2292 psecuritypriv->ndisencryptstatus = Ndis802_11Encryption1Enabled;
2293 psecuritypriv->dot11PrivacyAlgrthm = _WEP40_;
2294 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2295
2296 if (pwep->KeyLength == 13) {
2297 psecuritypriv->dot11PrivacyAlgrthm = _WEP104_;
2298 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2299 }
2300
2301 psecuritypriv->dot11PrivacyKeyIndex = wep_key_idx;
2302
2303 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2304
2305 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2306
2307 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2308 } else {
2309 DBG_88E("wep, set_tx = 0\n");
2310
2311 /* don't update "psecuritypriv->dot11PrivacyAlgrthm" and */
2312 /* psecuritypriv->dot11PrivacyKeyIndex = keyid", but can rtw_set_key to cam */
2313
2314 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2315
2316 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2317
2318 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2319 }
2320
2321 goto exit;
2322 }
2323
2324 if (!psta && check_fwstate(pmlmepriv, WIFI_AP_STATE)) { /* group key */
2325 if (param->u.crypt.set_tx == 1) {
2326 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2327 DBG_88E("%s, set group_key, WEP\n", __func__);
2328
2329 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002330 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002331
2332 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2333 if (param->u.crypt.key_len == 13)
2334 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2335 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2336 DBG_88E("%s, set group_key, TKIP\n", __func__);
2337 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2338 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002339 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002340 /* set mic key */
2341 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2342 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2343
2344 psecuritypriv->busetkipkey = true;
2345 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2346 DBG_88E("%s, set group_key, CCMP\n", __func__);
2347 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2348 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002349 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002350 } else {
2351 DBG_88E("%s, set group_key, none\n", __func__);
2352 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2353 }
2354 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2355 psecuritypriv->binstallGrpkey = true;
2356 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2357 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2358 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2359 if (pbcmc_sta) {
2360 pbcmc_sta->ieee8021x_blocked = false;
2361 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2362 }
2363 }
2364 goto exit;
2365 }
2366
2367 if (psecuritypriv->dot11AuthAlgrthm == dot11AuthAlgrthm_8021X && psta) { /* psk/802_1x */
2368 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
2369 if (param->u.crypt.set_tx == 1) {
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002370 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002371
2372 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2373 DBG_88E("%s, set pairwise key, WEP\n", __func__);
2374
2375 psta->dot118021XPrivacy = _WEP40_;
2376 if (param->u.crypt.key_len == 13)
2377 psta->dot118021XPrivacy = _WEP104_;
2378 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2379 DBG_88E("%s, set pairwise key, TKIP\n", __func__);
2380
2381 psta->dot118021XPrivacy = _TKIP_;
2382
2383 /* set mic key */
2384 memcpy(psta->dot11tkiptxmickey.skey, &(param->u.crypt.key[16]), 8);
2385 memcpy(psta->dot11tkiprxmickey.skey, &(param->u.crypt.key[24]), 8);
2386
2387 psecuritypriv->busetkipkey = true;
2388 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2389 DBG_88E("%s, set pairwise key, CCMP\n", __func__);
2390
2391 psta->dot118021XPrivacy = _AES_;
2392 } else {
2393 DBG_88E("%s, set pairwise key, none\n", __func__);
2394
2395 psta->dot118021XPrivacy = _NO_PRIVACY_;
2396 }
2397
2398 set_pairwise_key(padapter, psta);
2399
2400 psta->ieee8021x_blocked = false;
2401 } else { /* group key??? */
2402 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2403 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002404 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002405 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2406 if (param->u.crypt.key_len == 13)
2407 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2408 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2409 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2410
2411 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002412 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002413
2414 /* set mic key */
2415 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2416 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2417
2418 psecuritypriv->busetkipkey = true;
2419 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2420 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2421
2422 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
Ivan Safonovad8d8cd2015-10-27 22:18:10 +07002423 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
Larry Fingera2c60d42013-08-21 22:33:58 -05002424 } else {
2425 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2426 }
2427
2428 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2429
2430 psecuritypriv->binstallGrpkey = true;
2431
2432 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2433
2434 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2435
2436 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2437 if (pbcmc_sta) {
2438 pbcmc_sta->ieee8021x_blocked = false;
2439 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2440 }
2441 }
2442 }
2443 }
2444
2445exit:
2446
2447 kfree(pwep);
2448
2449 return ret;
2450}
2451
2452static int rtw_set_beacon(struct net_device *dev, struct ieee_param *param, int len)
2453{
2454 int ret = 0;
2455 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2456 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2457 struct sta_priv *pstapriv = &padapter->stapriv;
2458 unsigned char *pbuf = param->u.bcn_ie.buf;
2459
2460 DBG_88E("%s, len =%d\n", __func__, len);
2461
2462 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2463 return -EINVAL;
2464
2465 memcpy(&pstapriv->max_num_sta, param->u.bcn_ie.reserved, 2);
2466
2467 if ((pstapriv->max_num_sta > NUM_STA) || (pstapriv->max_num_sta <= 0))
2468 pstapriv->max_num_sta = NUM_STA;
2469
2470 if (rtw_check_beacon_data(padapter, pbuf, (len-12-2)) == _SUCCESS)/* 12 = param header, 2:no packed */
2471 ret = 0;
2472 else
2473 ret = -EINVAL;
2474
2475 return ret;
2476}
2477
2478static int rtw_hostapd_sta_flush(struct net_device *dev)
2479{
Larry Fingera2c60d42013-08-21 22:33:58 -05002480 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2481
2482 DBG_88E("%s\n", __func__);
2483
2484 flush_all_cam_entry(padapter); /* clear CAM */
2485
Shraddha Barkef523c7d2015-07-27 19:07:57 +05302486 return rtw_sta_flush(padapter);
Larry Fingera2c60d42013-08-21 22:33:58 -05002487}
2488
2489static int rtw_add_sta(struct net_device *dev, struct ieee_param *param)
2490{
2491 int ret = 0;
2492 struct sta_info *psta = NULL;
2493 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2494 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2495 struct sta_priv *pstapriv = &padapter->stapriv;
2496
2497 DBG_88E("rtw_add_sta(aid =%d) =%pM\n", param->u.add_sta.aid, (param->sta_addr));
2498
2499 if (!check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)))
2500 return -EINVAL;
2501
2502 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2503 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2504 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2505 return -EINVAL;
2506
2507 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2508 if (psta) {
2509 int flags = param->u.add_sta.flags;
2510
2511 psta->aid = param->u.add_sta.aid;/* aid = 1~2007 */
2512
2513 memcpy(psta->bssrateset, param->u.add_sta.tx_supp_rates, 16);
2514
2515 /* check wmm cap. */
2516 if (WLAN_STA_WME&flags)
2517 psta->qos_option = 1;
2518 else
2519 psta->qos_option = 0;
2520
2521 if (pmlmepriv->qospriv.qos_option == 0)
2522 psta->qos_option = 0;
2523
2524 /* chec 802.11n ht cap. */
2525 if (WLAN_STA_HT&flags) {
2526 psta->htpriv.ht_option = true;
2527 psta->qos_option = 1;
Ivan Safonov81a2b8e2016-08-24 15:21:18 +07002528 memcpy(&psta->htpriv.ht_cap, &param->u.add_sta.ht_cap,
2529 sizeof(struct ieee80211_ht_cap));
Larry Fingera2c60d42013-08-21 22:33:58 -05002530 } else {
2531 psta->htpriv.ht_option = false;
2532 }
2533
2534 if (pmlmepriv->htpriv.ht_option == false)
2535 psta->htpriv.ht_option = false;
2536
2537 update_sta_info_apmode(padapter, psta);
2538 } else {
2539 ret = -ENOMEM;
2540 }
2541
2542 return ret;
2543}
2544
2545static int rtw_del_sta(struct net_device *dev, struct ieee_param *param)
2546{
Larry Fingera2c60d42013-08-21 22:33:58 -05002547 int ret = 0;
2548 struct sta_info *psta = NULL;
2549 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2550 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2551 struct sta_priv *pstapriv = &padapter->stapriv;
2552 int updated = 0;
2553
2554 DBG_88E("rtw_del_sta =%pM\n", (param->sta_addr));
2555
2556 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2557 return -EINVAL;
2558
2559 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2560 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2561 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2562 return -EINVAL;
2563
2564 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2565 if (psta) {
Larry Finger7057dcb2013-12-19 22:38:34 -06002566 spin_lock_bh(&pstapriv->asoc_list_lock);
navin patidar9c4b0e72014-06-22 13:49:33 +05302567 if (!list_empty(&psta->asoc_list)) {
navin patidar8d5bdec2014-06-22 14:06:27 +05302568 list_del_init(&psta->asoc_list);
Larry Fingera2c60d42013-08-21 22:33:58 -05002569 pstapriv->asoc_list_cnt--;
2570 updated = ap_free_sta(padapter, psta, true, WLAN_REASON_DEAUTH_LEAVING);
2571 }
Larry Fingere02bcf62013-12-19 22:38:35 -06002572 spin_unlock_bh(&pstapriv->asoc_list_lock);
Larry Fingera2c60d42013-08-21 22:33:58 -05002573 associated_clients_update(padapter, updated);
2574 psta = NULL;
2575 } else {
2576 DBG_88E("rtw_del_sta(), sta has already been removed or never been added\n");
2577 }
2578
2579 return ret;
2580}
2581
2582static int rtw_ioctl_get_sta_data(struct net_device *dev, struct ieee_param *param, int len)
2583{
2584 int ret = 0;
2585 struct sta_info *psta = NULL;
2586 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2587 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2588 struct sta_priv *pstapriv = &padapter->stapriv;
2589 struct ieee_param_ex *param_ex = (struct ieee_param_ex *)param;
2590 struct sta_data *psta_data = (struct sta_data *)param_ex->data;
2591
2592 DBG_88E("rtw_ioctl_get_sta_info, sta_addr: %pM\n", (param_ex->sta_addr));
2593
2594 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2595 return -EINVAL;
2596
2597 if (param_ex->sta_addr[0] == 0xff && param_ex->sta_addr[1] == 0xff &&
2598 param_ex->sta_addr[2] == 0xff && param_ex->sta_addr[3] == 0xff &&
2599 param_ex->sta_addr[4] == 0xff && param_ex->sta_addr[5] == 0xff)
2600 return -EINVAL;
2601
2602 psta = rtw_get_stainfo(pstapriv, param_ex->sta_addr);
2603 if (psta) {
2604 psta_data->aid = (u16)psta->aid;
2605 psta_data->capability = psta->capability;
2606 psta_data->flags = psta->flags;
2607
2608/*
2609 nonerp_set : BIT(0)
2610 no_short_slot_time_set : BIT(1)
2611 no_short_preamble_set : BIT(2)
2612 no_ht_gf_set : BIT(3)
2613 no_ht_set : BIT(4)
2614 ht_20mhz_set : BIT(5)
2615*/
2616
2617 psta_data->sta_set = ((psta->nonerp_set) |
2618 (psta->no_short_slot_time_set << 1) |
2619 (psta->no_short_preamble_set << 2) |
2620 (psta->no_ht_gf_set << 3) |
2621 (psta->no_ht_set << 4) |
2622 (psta->ht_20mhz_set << 5));
2623 psta_data->tx_supp_rates_len = psta->bssratelen;
2624 memcpy(psta_data->tx_supp_rates, psta->bssrateset, psta->bssratelen);
Ivan Safonov81a2b8e2016-08-24 15:21:18 +07002625 memcpy(&psta_data->ht_cap,
2626 &psta->htpriv.ht_cap, sizeof(struct ieee80211_ht_cap));
Larry Fingera2c60d42013-08-21 22:33:58 -05002627 psta_data->rx_pkts = psta->sta_stats.rx_data_pkts;
2628 psta_data->rx_bytes = psta->sta_stats.rx_bytes;
2629 psta_data->rx_drops = psta->sta_stats.rx_drops;
2630 psta_data->tx_pkts = psta->sta_stats.tx_pkts;
2631 psta_data->tx_bytes = psta->sta_stats.tx_bytes;
2632 psta_data->tx_drops = psta->sta_stats.tx_drops;
2633 } else {
2634 ret = -1;
2635 }
2636
2637 return ret;
2638}
2639
2640static int rtw_get_sta_wpaie(struct net_device *dev, struct ieee_param *param)
2641{
2642 int ret = 0;
2643 struct sta_info *psta = NULL;
2644 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2645 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2646 struct sta_priv *pstapriv = &padapter->stapriv;
2647
2648 DBG_88E("rtw_get_sta_wpaie, sta_addr: %pM\n", (param->sta_addr));
2649
2650 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2651 return -EINVAL;
2652
2653 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2654 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2655 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2656 return -EINVAL;
2657
2658 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2659 if (psta) {
Jakub Sitnicki63d6c962015-07-29 10:15:16 +02002660 if (psta->wpa_ie[0] == WLAN_EID_RSN ||
2661 psta->wpa_ie[0] == WLAN_EID_VENDOR_SPECIFIC) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002662 int wpa_ie_len;
2663 int copy_len;
2664
2665 wpa_ie_len = psta->wpa_ie[1];
Ivan Safonov530c9b12015-10-27 22:19:50 +07002666 copy_len = min_t(int, wpa_ie_len + 2, sizeof(psta->wpa_ie));
Larry Fingera2c60d42013-08-21 22:33:58 -05002667 param->u.wpa_ie.len = copy_len;
2668 memcpy(param->u.wpa_ie.reserved, psta->wpa_ie, copy_len);
2669 } else {
2670 DBG_88E("sta's wpa_ie is NONE\n");
2671 }
2672 } else {
2673 ret = -1;
2674 }
2675
2676 return ret;
2677}
2678
2679static int rtw_set_wps_beacon(struct net_device *dev, struct ieee_param *param, int len)
2680{
2681 int ret = 0;
2682 unsigned char wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2683 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2684 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2685 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2686 int ie_len;
2687
2688 DBG_88E("%s, len =%d\n", __func__, len);
2689
2690 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2691 return -EINVAL;
2692
2693 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2694
Tapasweni Pathak157b8e52014-10-24 21:45:11 +05302695 kfree(pmlmepriv->wps_beacon_ie);
2696 pmlmepriv->wps_beacon_ie = NULL;
Larry Fingera2c60d42013-08-21 22:33:58 -05002697
2698 if (ie_len > 0) {
2699 pmlmepriv->wps_beacon_ie = rtw_malloc(ie_len);
2700 pmlmepriv->wps_beacon_ie_len = ie_len;
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302701 if (!pmlmepriv->wps_beacon_ie) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002702 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2703 return -EINVAL;
2704 }
2705
2706 memcpy(pmlmepriv->wps_beacon_ie, param->u.bcn_ie.buf, ie_len);
2707
2708 update_beacon(padapter, _VENDOR_SPECIFIC_IE_, wps_oui, true);
2709
2710 pmlmeext->bstart_bss = true;
2711 }
2712
2713 return ret;
2714}
2715
2716static int rtw_set_wps_probe_resp(struct net_device *dev, struct ieee_param *param, int len)
2717{
2718 int ret = 0;
2719 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2720 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2721 int ie_len;
2722
2723 DBG_88E("%s, len =%d\n", __func__, len);
2724
2725 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2726 return -EINVAL;
2727
2728 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2729
Tapasweni Pathak157b8e52014-10-24 21:45:11 +05302730 kfree(pmlmepriv->wps_probe_resp_ie);
2731 pmlmepriv->wps_probe_resp_ie = NULL;
Larry Fingera2c60d42013-08-21 22:33:58 -05002732
2733 if (ie_len > 0) {
2734 pmlmepriv->wps_probe_resp_ie = rtw_malloc(ie_len);
2735 pmlmepriv->wps_probe_resp_ie_len = ie_len;
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302736 if (!pmlmepriv->wps_probe_resp_ie) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002737 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2738 return -EINVAL;
2739 }
2740 memcpy(pmlmepriv->wps_probe_resp_ie, param->u.bcn_ie.buf, ie_len);
2741 }
2742
2743 return ret;
2744}
2745
2746static int rtw_set_wps_assoc_resp(struct net_device *dev, struct ieee_param *param, int len)
2747{
2748 int ret = 0;
2749 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2750 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2751 int ie_len;
2752
2753 DBG_88E("%s, len =%d\n", __func__, len);
2754
2755 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2756 return -EINVAL;
2757
2758 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2759
Tapasweni Pathak157b8e52014-10-24 21:45:11 +05302760 kfree(pmlmepriv->wps_assoc_resp_ie);
2761 pmlmepriv->wps_assoc_resp_ie = NULL;
Larry Fingera2c60d42013-08-21 22:33:58 -05002762
2763 if (ie_len > 0) {
2764 pmlmepriv->wps_assoc_resp_ie = rtw_malloc(ie_len);
2765 pmlmepriv->wps_assoc_resp_ie_len = ie_len;
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302766 if (!pmlmepriv->wps_assoc_resp_ie) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002767 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2768 return -EINVAL;
2769 }
2770
2771 memcpy(pmlmepriv->wps_assoc_resp_ie, param->u.bcn_ie.buf, ie_len);
2772 }
2773
2774 return ret;
2775}
2776
2777static int rtw_set_hidden_ssid(struct net_device *dev, struct ieee_param *param, int len)
2778{
2779 int ret = 0;
2780 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2781 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2782 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2783 struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
2784
2785 u8 value;
2786
2787 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2788 return -EINVAL;
2789
2790 if (param->u.wpa_param.name != 0) /* dummy test... */
2791 DBG_88E("%s name(%u) != 0\n", __func__, param->u.wpa_param.name);
2792 value = param->u.wpa_param.value;
2793
2794 /* use the same definition of hostapd's ignore_broadcast_ssid */
2795 if (value != 1 && value != 2)
2796 value = 0;
2797 DBG_88E("%s value(%u)\n", __func__, value);
2798 pmlmeinfo->hidden_ssid_mode = value;
2799 return ret;
2800}
2801
2802static int rtw_ioctl_acl_remove_sta(struct net_device *dev, struct ieee_param *param, int len)
2803{
Larry Fingera2c60d42013-08-21 22:33:58 -05002804 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2805 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2806
2807 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2808 return -EINVAL;
2809
2810 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2811 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2812 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2813 return -EINVAL;
Shraddha Barkef523c7d2015-07-27 19:07:57 +05302814 return rtw_acl_remove_sta(padapter, param->sta_addr);
Larry Fingera2c60d42013-08-21 22:33:58 -05002815}
2816
2817static int rtw_ioctl_acl_add_sta(struct net_device *dev, struct ieee_param *param, int len)
2818{
Larry Fingera2c60d42013-08-21 22:33:58 -05002819 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2820 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2821
2822 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2823 return -EINVAL;
2824
2825 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2826 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2827 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2828 return -EINVAL;
Shraddha Barkef523c7d2015-07-27 19:07:57 +05302829 return rtw_acl_add_sta(padapter, param->sta_addr);
Larry Fingera2c60d42013-08-21 22:33:58 -05002830}
2831
2832static int rtw_ioctl_set_macaddr_acl(struct net_device *dev, struct ieee_param *param, int len)
2833{
2834 int ret = 0;
2835 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2836 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2837
2838 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2839 return -EINVAL;
2840
2841 rtw_set_macaddr_acl(padapter, param->u.mlme.command);
2842
2843 return ret;
2844}
2845
2846static int rtw_hostapd_ioctl(struct net_device *dev, struct iw_point *p)
2847{
2848 struct ieee_param *param;
2849 int ret = 0;
2850 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2851
2852 /*
2853 * this function is expect to call in master mode, which allows no power saving
2854 * so, we just check hw_init_completed
2855 */
2856
2857 if (!padapter->hw_init_completed) {
2858 ret = -EPERM;
2859 goto out;
2860 }
2861
Larry Fingerd6cb5332020-02-10 12:02:30 -06002862 if (!p->pointer || p->length != sizeof(struct ieee_param)) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002863 ret = -EINVAL;
2864 goto out;
2865 }
2866
2867 param = (struct ieee_param *)rtw_malloc(p->length);
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302868 if (!param) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002869 ret = -ENOMEM;
2870 goto out;
2871 }
2872
2873 if (copy_from_user(param, p->pointer, p->length)) {
2874 kfree(param);
2875 ret = -EFAULT;
2876 goto out;
2877 }
2878
2879 switch (param->cmd) {
2880 case RTL871X_HOSTAPD_FLUSH:
2881 ret = rtw_hostapd_sta_flush(dev);
2882 break;
2883 case RTL871X_HOSTAPD_ADD_STA:
2884 ret = rtw_add_sta(dev, param);
2885 break;
2886 case RTL871X_HOSTAPD_REMOVE_STA:
2887 ret = rtw_del_sta(dev, param);
2888 break;
2889 case RTL871X_HOSTAPD_SET_BEACON:
2890 ret = rtw_set_beacon(dev, param, p->length);
2891 break;
2892 case RTL871X_SET_ENCRYPTION:
2893 ret = rtw_set_encryption(dev, param, p->length);
2894 break;
2895 case RTL871X_HOSTAPD_GET_WPAIE_STA:
2896 ret = rtw_get_sta_wpaie(dev, param);
2897 break;
2898 case RTL871X_HOSTAPD_SET_WPS_BEACON:
2899 ret = rtw_set_wps_beacon(dev, param, p->length);
2900 break;
2901 case RTL871X_HOSTAPD_SET_WPS_PROBE_RESP:
2902 ret = rtw_set_wps_probe_resp(dev, param, p->length);
2903 break;
2904 case RTL871X_HOSTAPD_SET_WPS_ASSOC_RESP:
2905 ret = rtw_set_wps_assoc_resp(dev, param, p->length);
2906 break;
2907 case RTL871X_HOSTAPD_SET_HIDDEN_SSID:
2908 ret = rtw_set_hidden_ssid(dev, param, p->length);
2909 break;
2910 case RTL871X_HOSTAPD_GET_INFO_STA:
2911 ret = rtw_ioctl_get_sta_data(dev, param, p->length);
2912 break;
2913 case RTL871X_HOSTAPD_SET_MACADDR_ACL:
2914 ret = rtw_ioctl_set_macaddr_acl(dev, param, p->length);
2915 break;
2916 case RTL871X_HOSTAPD_ACL_ADD_STA:
2917 ret = rtw_ioctl_acl_add_sta(dev, param, p->length);
2918 break;
2919 case RTL871X_HOSTAPD_ACL_REMOVE_STA:
2920 ret = rtw_ioctl_acl_remove_sta(dev, param, p->length);
2921 break;
2922 default:
2923 DBG_88E("Unknown hostapd request: %d\n", param->cmd);
2924 ret = -EOPNOTSUPP;
2925 break;
2926 }
2927
2928 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2929 ret = -EFAULT;
2930 kfree(param);
2931out:
2932 return ret;
2933}
2934#endif
2935
2936#include <rtw_android.h>
2937static int rtw_wx_set_priv(struct net_device *dev,
2938 struct iw_request_info *info,
2939 union iwreq_data *awrq,
2940 char *extra)
2941{
2942 int ret = 0;
2943 int len = 0;
2944 char *ext;
2945 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2946 struct iw_point *dwrq = (struct iw_point *)awrq;
2947
2948 if (dwrq->length == 0)
2949 return -EFAULT;
2950
2951 len = dwrq->length;
Larry Finger2397c6e2014-02-06 20:45:42 -06002952 ext = vmalloc(len);
Larry Fingera2c60d42013-08-21 22:33:58 -05002953 if (!ext)
2954 return -ENOMEM;
2955
2956 if (copy_from_user(ext, dwrq->pointer, len)) {
Larry Finger03bd6ae2014-02-06 20:45:43 -06002957 vfree(ext);
Larry Fingera2c60d42013-08-21 22:33:58 -05002958 return -EFAULT;
2959 }
2960
2961 /* added for wps2.0 @20110524 */
2962 if (dwrq->flags == 0x8766 && len > 8) {
2963 u32 cp_sz;
2964 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2965 u8 *probereq_wpsie = ext;
2966 int probereq_wpsie_len = len;
2967 u8 wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2968
2969 if ((_VENDOR_SPECIFIC_IE_ == probereq_wpsie[0]) &&
2970 (!memcmp(&probereq_wpsie[2], wps_oui, 4))) {
Ivan Safonov20e76532015-10-27 22:16:25 +07002971 cp_sz = min(probereq_wpsie_len, MAX_WPS_IE_LEN);
Larry Fingera2c60d42013-08-21 22:33:58 -05002972
2973 pmlmepriv->wps_probe_req_ie_len = 0;
2974 kfree(pmlmepriv->wps_probe_req_ie);
2975 pmlmepriv->wps_probe_req_ie = NULL;
2976
2977 pmlmepriv->wps_probe_req_ie = rtw_malloc(cp_sz);
Amit Ghadge4fe514d2016-09-04 22:53:35 +05302978 if (!pmlmepriv->wps_probe_req_ie) {
Larry Fingera2c60d42013-08-21 22:33:58 -05002979 pr_info("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2980 ret = -EINVAL;
2981 goto FREE_EXT;
2982 }
2983 memcpy(pmlmepriv->wps_probe_req_ie, probereq_wpsie, cp_sz);
2984 pmlmepriv->wps_probe_req_ie_len = cp_sz;
2985 }
2986 goto FREE_EXT;
2987 }
2988
2989 if (len >= WEXT_CSCAN_HEADER_SIZE &&
2990 !memcmp(ext, WEXT_CSCAN_HEADER, WEXT_CSCAN_HEADER_SIZE)) {
2991 ret = rtw_wx_set_scan(dev, info, awrq, ext);
2992 goto FREE_EXT;
2993 }
2994
2995FREE_EXT:
2996
Larry Finger03bd6ae2014-02-06 20:45:43 -06002997 vfree(ext);
Larry Fingera2c60d42013-08-21 22:33:58 -05002998
2999 return ret;
3000}
3001
Larry Fingera2c60d42013-08-21 22:33:58 -05003002static iw_handler rtw_handlers[] = {
3003 NULL, /* SIOCSIWCOMMIT */
3004 rtw_wx_get_name, /* SIOCGIWNAME */
3005 dummy, /* SIOCSIWNWID */
3006 dummy, /* SIOCGIWNWID */
3007 rtw_wx_set_freq, /* SIOCSIWFREQ */
3008 rtw_wx_get_freq, /* SIOCGIWFREQ */
3009 rtw_wx_set_mode, /* SIOCSIWMODE */
3010 rtw_wx_get_mode, /* SIOCGIWMODE */
3011 dummy, /* SIOCSIWSENS */
3012 rtw_wx_get_sens, /* SIOCGIWSENS */
3013 NULL, /* SIOCSIWRANGE */
3014 rtw_wx_get_range, /* SIOCGIWRANGE */
3015 rtw_wx_set_priv, /* SIOCSIWPRIV */
3016 NULL, /* SIOCGIWPRIV */
3017 NULL, /* SIOCSIWSTATS */
3018 NULL, /* SIOCGIWSTATS */
3019 dummy, /* SIOCSIWSPY */
3020 dummy, /* SIOCGIWSPY */
3021 NULL, /* SIOCGIWTHRSPY */
3022 NULL, /* SIOCWIWTHRSPY */
3023 rtw_wx_set_wap, /* SIOCSIWAP */
3024 rtw_wx_get_wap, /* SIOCGIWAP */
3025 rtw_wx_set_mlme, /* request MLME operation; uses struct iw_mlme */
3026 dummy, /* SIOCGIWAPLIST -- depricated */
3027 rtw_wx_set_scan, /* SIOCSIWSCAN */
3028 rtw_wx_get_scan, /* SIOCGIWSCAN */
3029 rtw_wx_set_essid, /* SIOCSIWESSID */
3030 rtw_wx_get_essid, /* SIOCGIWESSID */
3031 dummy, /* SIOCSIWNICKN */
3032 rtw_wx_get_nick, /* SIOCGIWNICKN */
3033 NULL, /* -- hole -- */
3034 NULL, /* -- hole -- */
3035 rtw_wx_set_rate, /* SIOCSIWRATE */
3036 rtw_wx_get_rate, /* SIOCGIWRATE */
3037 rtw_wx_set_rts, /* SIOCSIWRTS */
3038 rtw_wx_get_rts, /* SIOCGIWRTS */
3039 rtw_wx_set_frag, /* SIOCSIWFRAG */
3040 rtw_wx_get_frag, /* SIOCGIWFRAG */
3041 dummy, /* SIOCSIWTXPOW */
3042 dummy, /* SIOCGIWTXPOW */
3043 dummy, /* SIOCSIWRETRY */
3044 rtw_wx_get_retry, /* SIOCGIWRETRY */
3045 rtw_wx_set_enc, /* SIOCSIWENCODE */
3046 rtw_wx_get_enc, /* SIOCGIWENCODE */
3047 dummy, /* SIOCSIWPOWER */
3048 rtw_wx_get_power, /* SIOCGIWPOWER */
3049 NULL, /*---hole---*/
3050 NULL, /*---hole---*/
3051 rtw_wx_set_gen_ie, /* SIOCSIWGENIE */
3052 NULL, /* SIOCGWGENIE */
3053 rtw_wx_set_auth, /* SIOCSIWAUTH */
3054 NULL, /* SIOCGIWAUTH */
3055 rtw_wx_set_enc_ext, /* SIOCSIWENCODEEXT */
3056 NULL, /* SIOCGIWENCODEEXT */
3057 rtw_wx_set_pmkid, /* SIOCSIWPMKSA */
3058 NULL, /*---hole---*/
3059};
3060
Larry Fingera2c60d42013-08-21 22:33:58 -05003061static struct iw_statistics *rtw_get_wireless_stats(struct net_device *dev)
3062{
3063 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
3064 struct iw_statistics *piwstats = &padapter->iwstats;
3065 int tmp_level = 0;
3066 int tmp_qual = 0;
3067 int tmp_noise = 0;
3068
3069 if (!check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
3070 piwstats->qual.qual = 0;
3071 piwstats->qual.level = 0;
3072 piwstats->qual.noise = 0;
3073 } else {
3074 tmp_level = padapter->recvpriv.signal_strength;
3075 tmp_qual = padapter->recvpriv.signal_qual;
3076 tmp_noise = padapter->recvpriv.noise;
3077
3078 piwstats->qual.level = tmp_level;
3079 piwstats->qual.qual = tmp_qual;
3080 piwstats->qual.noise = tmp_noise;
3081 }
3082 piwstats->qual.updated = IW_QUAL_ALL_UPDATED;/* IW_QUAL_DBM; */
3083 return &padapter->iwstats;
3084}
3085
3086struct iw_handler_def rtw_handlers_def = {
3087 .standard = rtw_handlers,
Ivan Safonovbbef4722015-10-27 22:13:57 +07003088 .num_standard = ARRAY_SIZE(rtw_handlers),
Larry Fingera2c60d42013-08-21 22:33:58 -05003089 .get_wireless_stats = rtw_get_wireless_stats,
3090};
3091
Larry Fingera2c60d42013-08-21 22:33:58 -05003092int rtw_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
3093{
3094 struct iwreq *wrq = (struct iwreq *)rq;
3095 int ret = 0;
3096
3097 switch (cmd) {
3098 case RTL_IOCTL_WPA_SUPPLICANT:
3099 ret = wpa_supplicant_ioctl(dev, &wrq->u.data);
3100 break;
3101#ifdef CONFIG_88EU_AP_MODE
3102 case RTL_IOCTL_HOSTAPD:
3103 ret = rtw_hostapd_ioctl(dev, &wrq->u.data);
3104 break;
3105#endif /* CONFIG_88EU_AP_MODE */
Larry Fingera2c60d42013-08-21 22:33:58 -05003106 case (SIOCDEVPRIVATE+1):
3107 ret = rtw_android_priv_cmd(dev, rq, cmd);
3108 break;
3109 default:
3110 ret = -EOPNOTSUPP;
3111 break;
3112 }
3113 return ret;
3114}