1 /******************************************************************************
3 * Copyright(c) 2007 - 2012 Realtek Corporation. All rights reserved.
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.
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
14 ******************************************************************************/
15 #define _IOCTL_LINUX_C_
17 #include <linux/ieee80211.h>
19 #include <osdep_service.h>
20 #include <drv_types.h>
21 #include <wlan_bssdef.h>
22 #include <rtw_debug.h>
25 #include <rtw_mlme_ext.h>
26 #include <rtw_ioctl.h>
27 #include <rtw_ioctl_set.h>
28 #include <rtl8188e_hal.h>
31 #include <linux/vmalloc.h>
32 #include <linux/etherdevice.h>
34 #include "osdep_intf.h"
36 #define RTL_IOCTL_WPA_SUPPLICANT (SIOCIWFIRSTPRIV + 30)
38 #define SCAN_ITEM_SIZE 768
39 #define MAX_CUSTOM_LEN 64
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'
55 static u32 rtw_rates[] = {1000000, 2000000, 5500000, 11000000,
56 6000000, 9000000, 12000000, 18000000, 24000000, 36000000,
59 static const char * const iw_operation_mode[] = {
60 "Auto", "Ad-Hoc", "Managed", "Master", "Repeater",
61 "Secondary", "Monitor"
64 void indicate_wx_scan_complete_event(struct adapter *padapter)
66 union iwreq_data wrqu;
68 memset(&wrqu, 0, sizeof(union iwreq_data));
69 wireless_send_event(padapter->pnetdev, SIOCGIWSCAN, &wrqu, NULL);
72 void rtw_indicate_wx_assoc_event(struct adapter *padapter)
74 union iwreq_data wrqu;
75 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
77 memset(&wrqu, 0, sizeof(union iwreq_data));
79 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
81 memcpy(wrqu.ap_addr.sa_data, pmlmepriv->cur_network.network.MacAddress, ETH_ALEN);
83 DBG_88E_LEVEL(_drv_always_, "assoc success\n");
84 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
87 void rtw_indicate_wx_disassoc_event(struct adapter *padapter)
89 union iwreq_data wrqu;
91 memset(&wrqu, 0, sizeof(union iwreq_data));
93 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
94 eth_zero_addr(wrqu.ap_addr.sa_data);
96 DBG_88E_LEVEL(_drv_always_, "indicate disassoc\n");
97 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
100 static char *translate_scan(struct adapter *padapter,
101 struct iw_request_info *info,
102 struct wlan_network *pnetwork,
103 char *start, char *stop)
105 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
110 char custom[MAX_CUSTOM_LEN];
112 u16 max_rate = 0, rate, ht_cap = false;
114 u8 bw_40MHz = 0, short_GI = 0;
120 iwe.u.ap_addr.sa_family = ARPHRD_ETHER;
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);
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);
131 /* parsing HT_CAP_IE */
132 p = rtw_get_ie(&pnetwork->network.IEs[12], _HT_CAPABILITY_IE_, &ht_ielen, pnetwork->network.IELength-12);
134 if (p && ht_ielen > 0) {
135 struct ieee80211_ht_cap *pht_capie;
137 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));
146 /* Add the protocol name */
147 iwe.cmd = SIOCGIWNAME;
148 if ((rtw_is_cckratesonly_included((u8 *)&pnetwork->network.SupportedRates))) {
150 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bn");
152 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11b");
153 } else if ((rtw_is_cckrates_included((u8 *)&pnetwork->network.SupportedRates))) {
155 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bgn");
157 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bg");
159 if (pnetwork->network.Configuration.DSConfig > 14) {
161 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11an");
163 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11a");
166 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11gn");
168 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11g");
172 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_CHAR_LEN);
175 iwe.cmd = SIOCGIWMODE;
176 memcpy(&le_tmp, rtw_get_capability_from_ie(pnetwork->network.IEs), 2);
178 cap = le16_to_cpu(le_tmp);
180 if (!WLAN_CAPABILITY_IS_STA_BSS(cap)) {
181 if (cap & WLAN_CAPABILITY_ESS)
182 iwe.u.mode = IW_MODE_MASTER;
184 iwe.u.mode = IW_MODE_ADHOC;
186 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_UINT_LEN);
189 if (pnetwork->network.Configuration.DSConfig < 1)
190 pnetwork->network.Configuration.DSConfig = 1;
192 /* Add frequency/channel */
193 iwe.cmd = SIOCGIWFREQ;
194 iwe.u.freq.m = rtw_ch2freq(pnetwork->network.Configuration.DSConfig) * 100000;
196 iwe.u.freq.i = pnetwork->network.Configuration.DSConfig;
197 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_FREQ_LEN);
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;
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);
208 /*Add basic and extended rates */
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;
216 p += snprintf(p, MAX_CUSTOM_LEN - (p - custom),
217 "%d%s ", rate >> 1, (rate & 1) ? ".5" : "");
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 */
226 else/* default MCS7 */
227 max_rate = (bw_40MHz) ? ((short_GI) ? 150 : 135) : ((short_GI) ? 72 : 65);
229 max_rate = max_rate*2;/* Mbps/2; */
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);
238 /* parsing WPA/WPA2 IE */
240 u8 buf[MAX_WPA_IE_LEN];
241 u8 wpa_ie[255], rsn_ie[255];
242 u16 wpa_len = 0, rsn_len = 0;
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));
251 memset(buf, 0, MAX_WPA_IE_LEN);
252 p += sprintf(p, "wpa_ie=");
253 for (i = 0; i < wpa_len; i++)
254 p += sprintf(p, "%02x", wpa_ie[i]);
256 memset(&iwe, 0, sizeof(iwe));
257 iwe.cmd = IWEVCUSTOM;
258 iwe.u.data.length = strlen(buf);
259 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
261 memset(&iwe, 0, sizeof(iwe));
263 iwe.u.data.length = wpa_len;
264 start = iwe_stream_add_point(info, start, stop, &iwe, wpa_ie);
268 memset(buf, 0, MAX_WPA_IE_LEN);
269 p += sprintf(p, "rsn_ie=");
270 for (i = 0; i < rsn_len; i++)
271 p += sprintf(p, "%02x", rsn_ie[i]);
272 memset(&iwe, 0, sizeof(iwe));
273 iwe.cmd = IWEVCUSTOM;
274 iwe.u.data.length = strlen(buf);
275 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
277 memset(&iwe, 0, sizeof(iwe));
279 iwe.u.data.length = rsn_len;
280 start = iwe_stream_add_point(info, start, stop, &iwe, rsn_ie);
284 {/* parsing WPS IE */
285 uint cnt = 0, total_ielen;
286 u8 *wpsie_ptr = NULL;
289 u8 *ie_ptr = pnetwork->network.IEs + _FIXED_IE_LENGTH_;
290 total_ielen = pnetwork->network.IELength - _FIXED_IE_LENGTH_;
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];
296 iwe.u.data.length = (u16)wps_ielen;
297 start = iwe_stream_add_point(info, start, stop, &iwe, wpsie_ptr);
299 cnt += ie_ptr[cnt+1]+2; /* goto next */
303 /* Add quality statistics */
305 iwe.u.qual.updated = IW_QUAL_QUAL_UPDATED | IW_QUAL_LEVEL_UPDATED | IW_QUAL_NOISE_INVALID;
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;
312 ss = pnetwork->network.PhyInfo.SignalStrength;
313 sq = pnetwork->network.PhyInfo.SignalQuality;
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);
323 static int wpa_set_auth_algs(struct net_device *dev, u32 value)
325 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
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;
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;
345 } else if (value & AUTH_ALG_LEAP) {
346 DBG_88E("wpa_set_auth_algs, AUTH_ALG_LEAP\n");
348 DBG_88E("wpa_set_auth_algs, error!\n");
354 static int wpa_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
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;
363 param->u.crypt.err = 0;
364 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
366 if (param_len < (u32)((u8 *)param->u.crypt.key - (u8 *)param) + param->u.crypt.key_len) {
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) {
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");
387 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
388 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
389 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
391 wep_key_idx = param->u.crypt.idx;
392 wep_key_len = param->u.crypt.key_len;
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);
397 if (wep_key_idx > WEP_KEYS)
400 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("(2)wep_key_idx =%d\n", wep_key_idx));
402 if (wep_key_len > 0) {
403 wep_key_len = wep_key_len <= 5 ? 5 : 13;
404 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
405 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
407 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, (" wpa_set_encryption: pwep allocate fail !!!\n"));
410 memset(pwep, 0, wep_total_len);
411 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_;
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)
429 DBG_88E("wep, set_tx = 0\n");
430 if (wep_key_idx >= WEP_KEYS) {
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);
441 if (padapter->securitypriv.dot11AuthAlgrthm == dot11AuthAlgrthm_8021X) { /* 802_1x */
442 struct sta_info *psta, *pbcmc_sta;
443 struct sta_priv *pstapriv = &padapter->stapriv;
445 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE | WIFI_MP_STATE)) { /* sta mode */
446 psta = rtw_get_stainfo(pstapriv, get_bssid(pmlmepriv));
450 if (strcmp(param->u.crypt.alg, "none") != 0)
451 psta->ieee8021x_blocked = false;
453 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
454 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
455 psta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
457 if (param->u.crypt.set_tx == 1) { /* pairwise key */
458 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
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;
466 DBG_88E(" ~~~~set sta key:unicastkey\n");
468 rtw_setstakey_cmd(padapter, (unsigned char *)psta, true);
469 } else { /* group key */
470 memcpy(padapter->securitypriv.dot118021XGrpKey[param->u.crypt.idx].skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16 ));
471 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");
476 padapter->securitypriv.dot118021XGrpKeyid = param->u.crypt.idx;
478 rtw_set_key(padapter, &padapter->securitypriv, param->u.crypt.idx, 1);
481 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
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;
489 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
490 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
491 pbcmc_sta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
502 static int rtw_set_wpa_ie(struct adapter *padapter, char *pie, unsigned short ielen)
505 int group_cipher = 0, pairwise_cipher = 0;
508 if ((ielen > MAX_WPA_IE_LEN) || (!pie)) {
509 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
517 buf = kmemdup(pie, ielen, GFP_KERNEL);
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]);
531 if (ielen < RSN_HEADER_LEN) {
532 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("Ie len too short %d\n", ielen));
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);
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);
549 switch (group_cipher) {
550 case WPA_CIPHER_NONE:
551 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
552 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
554 case WPA_CIPHER_WEP40:
555 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
556 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
558 case WPA_CIPHER_TKIP:
559 padapter->securitypriv.dot118021XGrpPrivacy = _TKIP_;
560 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
562 case WPA_CIPHER_CCMP:
563 padapter->securitypriv.dot118021XGrpPrivacy = _AES_;
564 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
566 case WPA_CIPHER_WEP104:
567 padapter->securitypriv.dot118021XGrpPrivacy = _WEP104_;
568 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
572 switch (pairwise_cipher) {
573 case WPA_CIPHER_NONE:
574 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
575 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
577 case WPA_CIPHER_WEP40:
578 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
579 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
581 case WPA_CIPHER_TKIP:
582 padapter->securitypriv.dot11PrivacyAlgrthm = _TKIP_;
583 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
585 case WPA_CIPHER_CCMP:
586 padapter->securitypriv.dot11PrivacyAlgrthm = _AES_;
587 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
589 case WPA_CIPHER_WEP104:
590 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
591 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
595 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
598 u8 eid, wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
600 while (cnt < ielen) {
602 if ((eid == _VENDOR_SPECIFIC_IE_) && (!memcmp(&buf[cnt+2], wps_oui, 4))) {
603 DBG_88E("SET WPS_IE\n");
605 padapter->securitypriv.wps_ie_len = min(buf[cnt + 1] + 2, MAX_WPA_IE_LEN << 2);
607 memcpy(padapter->securitypriv.wps_ie, &buf[cnt], padapter->securitypriv.wps_ie_len);
609 set_fwstate(&padapter->mlmepriv, WIFI_UNDER_WPS);
613 cnt += buf[cnt+1]+2; /* goto next */
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));
627 typedef unsigned char NDIS_802_11_RATES_EX[NDIS_802_11_LENGTH_RATES_EX];
629 static int rtw_wx_get_name(struct net_device *dev,
630 struct iw_request_info *info,
631 union iwreq_data *wrqu, char *extra)
633 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
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;
641 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("cmd_code =%x\n", info->cmd));
643 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)
649 prates = &pcur_bss->SupportedRates;
651 if (rtw_is_cckratesonly_included((u8 *)prates) == true) {
653 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bn");
655 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11b");
656 } else if ((rtw_is_cckrates_included((u8 *)prates)) == true) {
658 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bgn");
660 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bg");
662 if (pcur_bss->Configuration.DSConfig > 14) {
664 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11an");
666 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11a");
669 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11gn");
671 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11g");
675 snprintf(wrqu->name, IFNAMSIZ, "unassociated");
680 static int rtw_wx_set_freq(struct net_device *dev,
681 struct iw_request_info *info,
682 union iwreq_data *wrqu, char *extra)
684 RT_TRACE(_module_rtl871x_mlme_c_, _drv_notice_, ("+rtw_wx_set_freq\n"));
688 static int rtw_wx_get_freq(struct net_device *dev,
689 struct iw_request_info *info,
690 union iwreq_data *wrqu, char *extra)
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;
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;
700 wrqu->freq.i = pcur_bss->Configuration.DSConfig;
702 wrqu->freq.m = rtw_ch2freq(padapter->mlmeextpriv.cur_channel) * 100000;
704 wrqu->freq.i = padapter->mlmeextpriv.cur_channel;
710 static int rtw_wx_set_mode(struct net_device *dev, struct iw_request_info *a,
711 union iwreq_data *wrqu, char *b)
713 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
714 enum ndis_802_11_network_infra networkType;
717 if (_FAIL == rtw_pwr_wakeup(padapter)) {
722 if (!padapter->hw_init_completed) {
727 switch (wrqu->mode) {
729 networkType = Ndis802_11AutoUnknown;
730 DBG_88E("set_mode = IW_MODE_AUTO\n");
733 networkType = Ndis802_11IBSS;
734 DBG_88E("set_mode = IW_MODE_ADHOC\n");
737 networkType = Ndis802_11APMode;
738 DBG_88E("set_mode = IW_MODE_MASTER\n");
741 networkType = Ndis802_11Infrastructure;
742 DBG_88E("set_mode = IW_MODE_INFRA\n");
746 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("\n Mode: %s is not supported\n", iw_operation_mode[wrqu->mode]));
749 if (rtw_set_802_11_infrastructure_mode(padapter, networkType) == false) {
753 rtw_setopmode_cmd(padapter, networkType);
758 static int rtw_wx_get_mode(struct net_device *dev, struct iw_request_info *a,
759 union iwreq_data *wrqu, char *b)
761 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
762 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
764 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_get_mode\n"));
766 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;
774 wrqu->mode = IW_MODE_AUTO;
779 static int rtw_wx_set_pmkid(struct net_device *dev,
780 struct iw_request_info *a,
781 union iwreq_data *wrqu, char *extra)
783 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
784 u8 j, blInserted = 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};
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");
794 if (!memcmp(strIssueBssid, strZeroMacAddress, ETH_ALEN))
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;
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);
818 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].Bssid, strIssueBssid, ETH_ALEN);
819 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].PMKID, pPMK->pmkid, IW_PMKID_LEN);
821 psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].bUsed = true;
822 psecuritypriv->PMKIDIndex++;
823 if (psecuritypriv->PMKIDIndex == 16)
824 psecuritypriv->PMKIDIndex = 0;
826 } else if (pPMK->cmd == IW_PMKSA_REMOVE) {
827 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_REMOVE!\n");
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. */
832 eth_zero_addr(psecuritypriv->PMKIDList[j].Bssid);
833 psecuritypriv->PMKIDList[j].bUsed = false;
837 } else if (pPMK->cmd == IW_PMKSA_FLUSH) {
838 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_FLUSH!\n");
839 memset(&psecuritypriv->PMKIDList[0], 0x00, sizeof(struct rt_pmkid_list) * NUM_PMKID_CACHE);
840 psecuritypriv->PMKIDIndex = 0;
846 static int rtw_wx_get_sens(struct net_device *dev,
847 struct iw_request_info *info,
848 union iwreq_data *wrqu, char *extra)
850 wrqu->sens.value = 0;
851 wrqu->sens.fixed = 0; /* no auto select */
852 wrqu->sens.disabled = 1;
856 static int rtw_wx_get_range(struct net_device *dev,
857 struct iw_request_info *info,
858 union iwreq_data *wrqu, char *extra)
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;
867 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_range. cmd_code =%x\n", info->cmd));
869 wrqu->data.length = sizeof(*range);
870 memset(range, 0, sizeof(*range));
872 /* Let's try to keep this struct in the same order as in
873 * linux/include/wireless.h
876 /* TODO: See what values we can set, and remove the ones we can't
877 * set, or fill them with some default data.
880 /* ~5 Mb/s real (802.11b) */
881 range->throughput = 5 * 1000 * 1000;
883 /* signal level threshold range */
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 */
891 range->avg_qual.qual = 92; /* > 8% missed beacons is 'bad' */
892 /* TODO: Find real 'good' to 'bad' threshol value for RSSI */
893 range->avg_qual.level = 178; /* -78 dBm */
894 range->avg_qual.noise = 0;
895 range->avg_qual.updated = 7; /* Updated all three */
897 range->num_bitrates = RATE_COUNT;
899 for (i = 0; i < RATE_COUNT && i < IW_MAX_BITRATES; i++)
900 range->bitrate[i] = rtw_rates[i];
902 range->min_frag = MIN_FRAG_THRESHOLD;
903 range->max_frag = MAX_FRAG_THRESHOLD;
907 range->we_version_compiled = WIRELESS_EXT;
908 range->we_version_source = 16;
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;
919 if (val == IW_MAX_FREQUENCIES)
923 range->num_channels = val;
924 range->num_frequency = val;
926 /* The following code will proivde the security capability to network manager. */
927 /* If the driver doesn't provide this capability to network manager, */
928 /* the WPA/WPA2 routers can't be chosen in the network manager. */
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
941 range->enc_capa = IW_ENC_CAPA_WPA | IW_ENC_CAPA_WPA2 |
942 IW_ENC_CAPA_CIPHER_TKIP | IW_ENC_CAPA_CIPHER_CCMP;
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;
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() */
955 static int rtw_wx_set_wap(struct net_device *dev,
956 struct iw_request_info *info,
957 union iwreq_data *awrq,
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;
970 if (_FAIL == rtw_pwr_wakeup(padapter)) {
975 if (!padapter->bup) {
980 if (temp->sa_family != ARPHRD_ETHER) {
985 authmode = padapter->securitypriv.ndisauthtype;
986 spin_lock_bh(&queue->lock);
987 phead = get_list_head(queue);
988 pmlmepriv->pscanned = phead->next;
990 while (phead != pmlmepriv->pscanned) {
991 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
993 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
995 dst_bssid = pnetwork->network.MacAddress;
997 src_bssid = temp->sa_data;
999 if ((!memcmp(dst_bssid, src_bssid, ETH_ALEN))) {
1000 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
1002 spin_unlock_bh(&queue->lock);
1009 spin_unlock_bh(&queue->lock);
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) {
1023 static int rtw_wx_get_wap(struct net_device *dev,
1024 struct iw_request_info *info,
1025 union iwreq_data *wrqu, char *extra)
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;
1031 wrqu->ap_addr.sa_family = ARPHRD_ETHER;
1033 eth_zero_addr(wrqu->ap_addr.sa_data);
1035 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_wap\n"));
1037 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);
1042 eth_zero_addr(wrqu->ap_addr.sa_data);
1046 static int rtw_wx_set_mlme(struct net_device *dev,
1047 struct iw_request_info *info,
1048 union iwreq_data *wrqu, char *extra)
1052 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1053 struct iw_mlme *mlme = (struct iw_mlme *)extra;
1058 DBG_88E("%s\n", __func__);
1060 reason = mlme->reason_code;
1062 DBG_88E("%s, cmd =%d, reason =%d\n", __func__, mlme->cmd, reason);
1064 switch (mlme->cmd) {
1065 case IW_MLME_DEAUTH:
1066 if (!rtw_set_802_11_disassociate(padapter))
1069 case IW_MLME_DISASSOC:
1070 if (!rtw_set_802_11_disassociate(padapter))
1079 static int rtw_wx_set_scan(struct net_device *dev, struct iw_request_info *a,
1080 union iwreq_data *wrqu, char *extra)
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];
1087 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_set_scan\n"));
1089 if (padapter->registrypriv.mp_mode == 1) {
1090 if (check_fwstate(pmlmepriv, WIFI_MP_STATE)) {
1095 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1100 if (padapter->bDriverStopped) {
1101 DBG_88E("bDriverStopped =%d\n", padapter->bDriverStopped);
1106 if (!padapter->bup) {
1111 if (!padapter->hw_init_completed) {
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);
1124 if (check_fwstate(pmlmepriv, _FW_UNDER_SURVEY|_FW_UNDER_LINKING)) {
1125 indicate_wx_scan_complete_event(padapter);
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. */
1133 memset(ssid, 0, sizeof(struct ndis_802_11_ssid)*RTW_SSID_SCAN_AMOUNT);
1135 if (wrqu->data.length == sizeof(struct iw_scan_req)) {
1136 struct iw_scan_req *req = (struct iw_scan_req *)extra;
1138 if (wrqu->data.flags & IW_SCAN_THIS_ESSID) {
1139 int len = min_t(int, req->essid_len,
1142 memcpy(ssid[0].Ssid, req->essid, len);
1143 ssid[0].SsidLength = len;
1145 DBG_88E("IW_SCAN_THIS_ESSID, ssid =%s, len =%d\n", req->essid, req->essid_len);
1147 spin_lock_bh(&pmlmepriv->lock);
1149 _status = rtw_sitesurvey_cmd(padapter, ssid, 1, NULL, 0);
1151 spin_unlock_bh(&pmlmepriv->lock);
1152 } else if (req->scan_type == IW_SCAN_TYPE_PASSIVE) {
1153 DBG_88E("rtw_wx_set_scan, req->scan_type == IW_SCAN_TYPE_PASSIVE\n");
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;
1169 case WEXT_CSCAN_SSID_SECTION:
1174 sec_len = *(pos++); len -= 1;
1178 ssid[ssid_index].SsidLength = sec_len;
1179 memcpy(ssid[ssid_index].Ssid, pos, sec_len);
1185 case WEXT_CSCAN_TYPE_SECTION:
1186 case WEXT_CSCAN_CHANNEL_SECTION:
1190 case WEXT_CSCAN_PASV_DWELL_SECTION:
1191 case WEXT_CSCAN_HOME_DWELL_SECTION:
1192 case WEXT_CSCAN_ACTV_DWELL_SECTION:
1197 len = 0; /* stop parsing */
1201 /* it has still some scan parameter to parse, we only do this now... */
1202 _status = rtw_set_802_11_bssid_list_scan(padapter, ssid, RTW_SSID_SCAN_AMOUNT);
1204 _status = rtw_set_802_11_bssid_list_scan(padapter, NULL, 0);
1216 static int rtw_wx_get_scan(struct net_device *dev, struct iw_request_info *a,
1217 union iwreq_data *wrqu, char *extra)
1219 struct list_head *plist, *phead;
1220 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1221 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1222 struct __queue *queue = &(pmlmepriv->scanned_queue);
1223 struct wlan_network *pnetwork = NULL;
1225 char *stop = ev + wrqu->data.length;
1228 u32 wait_for_surveydone;
1230 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan\n"));
1231 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, (" Start of Query SIOCGIWSCAN .\n"));
1233 if (padapter->pwrctrlpriv.brfoffbyhw && padapter->bDriverStopped) {
1238 wait_for_surveydone = 100;
1240 wait_status = _FW_UNDER_SURVEY | _FW_UNDER_LINKING;
1242 while (check_fwstate(pmlmepriv, wait_status)) {
1245 if (cnt > wait_for_surveydone)
1249 spin_lock_bh(&(pmlmepriv->scanned_queue.lock));
1251 phead = get_list_head(queue);
1252 plist = phead->next;
1254 while (phead != plist) {
1255 if ((stop - ev) < SCAN_ITEM_SIZE) {
1260 pnetwork = container_of(plist, struct wlan_network, list);
1262 /* report network only if the current channel set contains the channel to which this network belongs */
1263 if (rtw_ch_set_search_ch(padapter->mlmeextpriv.channel_set, pnetwork->network.Configuration.DSConfig) >= 0)
1264 ev = translate_scan(padapter, a, pnetwork, ev, stop);
1266 plist = plist->next;
1269 spin_unlock_bh(&pmlmepriv->scanned_queue.lock);
1271 wrqu->data.length = ev-extra;
1272 wrqu->data.flags = 0;
1279 /* s1. rtw_set_802_11_infrastructure_mode() */
1280 /* s2. set_802_11_authenticaion_mode() */
1281 /* s3. set_802_11_encryption_mode() */
1282 /* s4. rtw_set_802_11_ssid() */
1283 static int rtw_wx_set_essid(struct net_device *dev,
1284 struct iw_request_info *a,
1285 union iwreq_data *wrqu, char *extra)
1287 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1288 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
1289 struct __queue *queue = &pmlmepriv->scanned_queue;
1290 struct list_head *phead;
1291 struct wlan_network *pnetwork = NULL;
1292 enum ndis_802_11_auth_mode authmode;
1293 struct ndis_802_11_ssid ndis_ssid;
1294 u8 *dst_ssid, *src_ssid;
1299 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1300 ("+rtw_wx_set_essid: fw_state = 0x%08x\n", get_fwstate(pmlmepriv)));
1301 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1306 if (!padapter->bup) {
1311 if (wrqu->essid.length > IW_ESSID_MAX_SIZE) {
1316 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
1321 authmode = padapter->securitypriv.ndisauthtype;
1322 DBG_88E("=>%s\n", __func__);
1323 if (wrqu->essid.flags && wrqu->essid.length) {
1324 len = min_t(uint, wrqu->essid.length, IW_ESSID_MAX_SIZE);
1326 if (wrqu->essid.length != 33)
1327 DBG_88E("ssid =%s, len =%d\n", extra, wrqu->essid.length);
1329 memset(&ndis_ssid, 0, sizeof(struct ndis_802_11_ssid));
1330 ndis_ssid.SsidLength = len;
1331 memcpy(ndis_ssid.Ssid, extra, len);
1332 src_ssid = ndis_ssid.Ssid;
1334 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("rtw_wx_set_essid: ssid =[%s]\n", src_ssid));
1335 spin_lock_bh(&queue->lock);
1336 phead = get_list_head(queue);
1337 pmlmepriv->pscanned = phead->next;
1339 while (phead != pmlmepriv->pscanned) {
1340 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
1342 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
1344 dst_ssid = pnetwork->network.Ssid.Ssid;
1346 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1347 ("rtw_wx_set_essid: dst_ssid =%s\n",
1348 pnetwork->network.Ssid.Ssid));
1350 if ((!memcmp(dst_ssid, src_ssid, ndis_ssid.SsidLength)) &&
1351 (pnetwork->network.Ssid.SsidLength == ndis_ssid.SsidLength)) {
1352 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1353 ("rtw_wx_set_essid: find match, set infra mode\n"));
1355 if (check_fwstate(pmlmepriv, WIFI_ADHOC_STATE) == true) {
1356 if (pnetwork->network.InfrastructureMode != pmlmepriv->cur_network.network.InfrastructureMode)
1360 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
1362 spin_unlock_bh(&queue->lock);
1369 spin_unlock_bh(&queue->lock);
1370 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1371 ("set ssid: set_802_11_auth. mode =%d\n", authmode));
1372 rtw_set_802_11_authentication_mode(padapter, authmode);
1373 if (rtw_set_802_11_ssid(padapter, &ndis_ssid) == false) {
1381 DBG_88E("<=%s, ret %d\n", __func__, ret);
1387 static int rtw_wx_get_essid(struct net_device *dev,
1388 struct iw_request_info *a,
1389 union iwreq_data *wrqu, char *extra)
1392 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1393 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1394 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
1396 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_essid\n"));
1399 if ((check_fwstate(pmlmepriv, _FW_LINKED)) ||
1400 (check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE))) {
1401 len = pcur_bss->Ssid.SsidLength;
1402 memcpy(extra, pcur_bss->Ssid.Ssid, len);
1407 wrqu->essid.length = len;
1408 wrqu->essid.flags = 1;
1413 static int rtw_wx_set_rate(struct net_device *dev,
1414 struct iw_request_info *a,
1415 union iwreq_data *wrqu, char *extra)
1418 u8 datarates[NumRates];
1419 u32 target_rate = wrqu->bitrate.value;
1420 u32 fixed = wrqu->bitrate.fixed;
1422 u8 mpdatarate[NumRates] = {11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0xff};
1425 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_set_rate\n"));
1426 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("target_rate = %d, fixed = %d\n", target_rate, fixed));
1428 if (target_rate == -1) {
1432 target_rate = target_rate/100000;
1434 switch (target_rate) {
1478 for (i = 0; i < NumRates; i++) {
1479 if (ratevalue == mpdatarate[i]) {
1480 datarates[i] = mpdatarate[i];
1484 datarates[i] = 0xff;
1487 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("datarate_inx =%d\n", datarates[i]));
1493 static int rtw_wx_get_rate(struct net_device *dev,
1494 struct iw_request_info *info,
1495 union iwreq_data *wrqu, char *extra)
1499 max_rate = rtw_get_cur_max_rate((struct adapter *)rtw_netdev_priv(dev));
1504 wrqu->bitrate.fixed = 0; /* no auto select */
1505 wrqu->bitrate.value = max_rate * 100000;
1510 static int rtw_wx_set_rts(struct net_device *dev,
1511 struct iw_request_info *info,
1512 union iwreq_data *wrqu, char *extra)
1514 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1517 if (wrqu->rts.disabled) {
1518 padapter->registrypriv.rts_thresh = 2347;
1520 if (wrqu->rts.value < 0 ||
1521 wrqu->rts.value > 2347)
1524 padapter->registrypriv.rts_thresh = wrqu->rts.value;
1527 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1533 static int rtw_wx_get_rts(struct net_device *dev,
1534 struct iw_request_info *info,
1535 union iwreq_data *wrqu, char *extra)
1537 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1540 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1542 wrqu->rts.value = padapter->registrypriv.rts_thresh;
1543 wrqu->rts.fixed = 0; /* no auto select */
1544 /* wrqu->rts.disabled = (wrqu->rts.value == DEFAULT_RTS_THRESHOLD); */
1550 static int rtw_wx_set_frag(struct net_device *dev,
1551 struct iw_request_info *info,
1552 union iwreq_data *wrqu, char *extra)
1554 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1557 if (wrqu->frag.disabled) {
1558 padapter->xmitpriv.frag_len = MAX_FRAG_THRESHOLD;
1560 if (wrqu->frag.value < MIN_FRAG_THRESHOLD ||
1561 wrqu->frag.value > MAX_FRAG_THRESHOLD)
1564 padapter->xmitpriv.frag_len = wrqu->frag.value & ~0x1;
1567 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1573 static int rtw_wx_get_frag(struct net_device *dev,
1574 struct iw_request_info *info,
1575 union iwreq_data *wrqu, char *extra)
1577 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1580 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1582 wrqu->frag.value = padapter->xmitpriv.frag_len;
1583 wrqu->frag.fixed = 0; /* no auto select */
1589 static int rtw_wx_get_retry(struct net_device *dev,
1590 struct iw_request_info *info,
1591 union iwreq_data *wrqu, char *extra)
1593 wrqu->retry.value = 7;
1594 wrqu->retry.fixed = 0; /* no auto select */
1595 wrqu->retry.disabled = 1;
1600 static int rtw_wx_set_enc(struct net_device *dev,
1601 struct iw_request_info *info,
1602 union iwreq_data *wrqu, char *keybuf)
1605 u32 keyindex_provided;
1606 struct ndis_802_11_wep wep;
1607 enum ndis_802_11_auth_mode authmode;
1609 struct iw_point *erq = &(wrqu->encoding);
1610 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1611 struct pwrctrl_priv *pwrpriv = &padapter->pwrctrlpriv;
1612 DBG_88E("+rtw_wx_set_enc, flags = 0x%x\n", erq->flags);
1614 memset(&wep, 0, sizeof(struct ndis_802_11_wep));
1616 key = erq->flags & IW_ENCODE_INDEX;
1619 if (erq->flags & IW_ENCODE_DISABLED) {
1620 DBG_88E("EncryptionDisabled\n");
1621 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1622 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1623 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1624 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1625 authmode = Ndis802_11AuthModeOpen;
1626 padapter->securitypriv.ndisauthtype = authmode;
1635 keyindex_provided = 1;
1637 keyindex_provided = 0;
1638 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1639 DBG_88E("rtw_wx_set_enc, key =%d\n", key);
1642 /* set authentication mode */
1643 if (erq->flags & IW_ENCODE_OPEN) {
1644 DBG_88E("rtw_wx_set_enc():IW_ENCODE_OPEN\n");
1645 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
1646 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1647 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1648 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1649 authmode = Ndis802_11AuthModeOpen;
1650 padapter->securitypriv.ndisauthtype = authmode;
1651 } else if (erq->flags & IW_ENCODE_RESTRICTED) {
1652 DBG_88E("rtw_wx_set_enc():IW_ENCODE_RESTRICTED\n");
1653 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
1654 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Shared;
1655 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1656 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
1657 authmode = Ndis802_11AuthModeShared;
1658 padapter->securitypriv.ndisauthtype = authmode;
1660 DBG_88E("rtw_wx_set_enc():erq->flags = 0x%x\n", erq->flags);
1662 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
1663 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1664 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1665 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1666 authmode = Ndis802_11AuthModeOpen;
1667 padapter->securitypriv.ndisauthtype = authmode;
1671 if (erq->length > 0) {
1672 wep.KeyLength = erq->length <= 5 ? 5 : 13;
1674 wep.Length = wep.KeyLength + offsetof(struct ndis_802_11_wep, KeyMaterial);
1678 if (keyindex_provided == 1) {
1679 /* set key_id only, no given KeyMaterial(erq->length == 0). */
1680 padapter->securitypriv.dot11PrivacyKeyIndex = key;
1682 DBG_88E("(keyindex_provided == 1), keyid =%d, key_len =%d\n", key, padapter->securitypriv.dot11DefKeylen[key]);
1684 switch (padapter->securitypriv.dot11DefKeylen[key]) {
1686 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1689 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
1692 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1700 wep.KeyIndex |= 0x80000000;
1702 memcpy(wep.KeyMaterial, keybuf, wep.KeyLength);
1704 if (rtw_set_802_11_add_wep(padapter, &wep) == false) {
1705 if (rf_on == pwrpriv->rf_pwrstate)
1716 static int rtw_wx_get_enc(struct net_device *dev,
1717 struct iw_request_info *info,
1718 union iwreq_data *wrqu, char *keybuf)
1721 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1722 struct iw_point *erq = &(wrqu->encoding);
1723 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1726 if (check_fwstate(pmlmepriv, _FW_LINKED) != true) {
1727 if (!check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) {
1729 erq->flags |= IW_ENCODE_DISABLED;
1734 key = erq->flags & IW_ENCODE_INDEX;
1741 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1744 erq->flags = key + 1;
1746 switch (padapter->securitypriv.ndisencryptstatus) {
1747 case Ndis802_11EncryptionNotSupported:
1748 case Ndis802_11EncryptionDisabled:
1750 erq->flags |= IW_ENCODE_DISABLED;
1752 case Ndis802_11Encryption1Enabled:
1753 erq->length = padapter->securitypriv.dot11DefKeylen[key];
1755 memcpy(keybuf, padapter->securitypriv.dot11DefKey[key].skey, padapter->securitypriv.dot11DefKeylen[key]);
1757 erq->flags |= IW_ENCODE_ENABLED;
1759 if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeOpen)
1760 erq->flags |= IW_ENCODE_OPEN;
1761 else if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeShared)
1762 erq->flags |= IW_ENCODE_RESTRICTED;
1765 erq->flags |= IW_ENCODE_DISABLED;
1768 case Ndis802_11Encryption2Enabled:
1769 case Ndis802_11Encryption3Enabled:
1771 erq->flags |= (IW_ENCODE_ENABLED | IW_ENCODE_OPEN | IW_ENCODE_NOKEY);
1775 erq->flags |= IW_ENCODE_DISABLED;
1782 static int rtw_wx_get_power(struct net_device *dev,
1783 struct iw_request_info *info,
1784 union iwreq_data *wrqu, char *extra)
1786 wrqu->power.value = 0;
1787 wrqu->power.fixed = 0; /* no auto select */
1788 wrqu->power.disabled = 1;
1793 static int rtw_wx_set_gen_ie(struct net_device *dev,
1794 struct iw_request_info *info,
1795 union iwreq_data *wrqu, char *extra)
1797 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1799 return rtw_set_wpa_ie(padapter, extra, wrqu->data.length);
1802 static int rtw_wx_set_auth(struct net_device *dev,
1803 struct iw_request_info *info,
1804 union iwreq_data *wrqu, char *extra)
1806 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1807 struct iw_param *param = (struct iw_param *)&(wrqu->param);
1810 switch (param->flags & IW_AUTH_INDEX) {
1811 case IW_AUTH_WPA_VERSION:
1813 case IW_AUTH_CIPHER_PAIRWISE:
1816 case IW_AUTH_CIPHER_GROUP:
1819 case IW_AUTH_KEY_MGMT:
1821 * ??? does not use these parameters
1824 case IW_AUTH_TKIP_COUNTERMEASURES:
1826 /* wpa_supplicant is enabling the tkip countermeasure. */
1827 padapter->securitypriv.btkip_countermeasure = true;
1829 /* wpa_supplicant is disabling the tkip countermeasure. */
1830 padapter->securitypriv.btkip_countermeasure = false;
1833 case IW_AUTH_DROP_UNENCRYPTED:
1836 * wpa_supplicant calls set_wpa_enabled when the driver
1837 * is loaded and unloaded, regardless of if WPA is being
1838 * used. No other calls are made which can be used to
1839 * determine if encryption will be used or not prior to
1840 * association being expected. If encryption is not being
1841 * used, drop_unencrypted is set to false, else true -- we
1842 * can use this to determine if the CAP_PRIVACY_ON bit should
1846 if (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption1Enabled)
1847 break;/* it means init value, or using wep, ndisencryptstatus = Ndis802_11Encryption1Enabled, */
1848 /* then it needn't reset it; */
1851 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1852 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1853 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1854 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1855 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeOpen;
1859 case IW_AUTH_80211_AUTH_ALG:
1861 * It's the starting point of a link layer connection using wpa_supplicant
1863 if (check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
1864 LeaveAllPowerSaveMode(padapter);
1865 rtw_disassoc_cmd(padapter, 500, false);
1866 DBG_88E("%s...call rtw_indicate_disconnect\n ", __func__);
1867 rtw_indicate_disconnect(padapter);
1868 rtw_free_assoc_resources(padapter);
1870 ret = wpa_set_auth_algs(dev, (u32)param->value);
1872 case IW_AUTH_WPA_ENABLED:
1874 case IW_AUTH_RX_UNENCRYPTED_EAPOL:
1876 case IW_AUTH_PRIVACY_INVOKED:
1885 static int rtw_wx_set_enc_ext(struct net_device *dev,
1886 struct iw_request_info *info,
1887 union iwreq_data *wrqu, char *extra)
1891 struct ieee_param *param = NULL;
1892 struct iw_point *pencoding = &wrqu->encoding;
1893 struct iw_encode_ext *pext = (struct iw_encode_ext *)extra;
1896 param_len = sizeof(struct ieee_param) + pext->key_len;
1897 param = (struct ieee_param *)rtw_malloc(param_len);
1901 memset(param, 0, param_len);
1903 param->cmd = IEEE_CMD_SET_ENCRYPTION;
1904 eth_broadcast_addr(param->sta_addr);
1906 switch (pext->alg) {
1907 case IW_ENCODE_ALG_NONE:
1908 /* todo: remove key */
1912 case IW_ENCODE_ALG_WEP:
1915 case IW_ENCODE_ALG_TKIP:
1918 case IW_ENCODE_ALG_CCMP:
1926 strncpy((char *)param->u.crypt.alg, alg_name, IEEE_CRYPT_ALG_NAME_LEN);
1928 if (pext->ext_flags & IW_ENCODE_EXT_SET_TX_KEY)
1929 param->u.crypt.set_tx = 1;
1931 /* cliW: WEP does not have group key
1932 * just not checking GROUP key setting
1934 if ((pext->alg != IW_ENCODE_ALG_WEP) &&
1935 (pext->ext_flags & IW_ENCODE_EXT_GROUP_KEY))
1936 param->u.crypt.set_tx = 0;
1938 param->u.crypt.idx = (pencoding->flags&0x00FF) - 1;
1940 if (pext->ext_flags & IW_ENCODE_EXT_RX_SEQ_VALID)
1941 memcpy(param->u.crypt.seq, pext->rx_seq, 8);
1943 if (pext->key_len) {
1944 param->u.crypt.key_len = pext->key_len;
1945 memcpy(param->u.crypt.key, pext + 1, pext->key_len);
1948 ret = wpa_set_encryption(dev, param, param_len);
1955 static int rtw_wx_get_nick(struct net_device *dev,
1956 struct iw_request_info *info,
1957 union iwreq_data *wrqu, char *extra)
1960 wrqu->data.length = 14;
1961 wrqu->data.flags = 1;
1962 memcpy(extra, "<WIFI@REALTEK>", 14);
1965 /* dump debug info here */
1969 static int dummy(struct net_device *dev, struct iw_request_info *a,
1970 union iwreq_data *wrqu, char *b)
1975 static int wpa_set_param(struct net_device *dev, u8 name, u32 value)
1978 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1981 case IEEE_PARAM_WPA_ENABLED:
1982 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X; /* 802.1x */
1983 switch ((value)&0xff) {
1985 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPAPSK; /* WPA_PSK */
1986 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
1989 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPA2PSK; /* WPA2_PSK */
1990 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
1993 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1994 ("wpa_set_param:padapter->securitypriv.ndisauthtype =%d\n", padapter->securitypriv.ndisauthtype));
1996 case IEEE_PARAM_TKIP_COUNTERMEASURES:
1998 case IEEE_PARAM_DROP_UNENCRYPTED: {
2001 * wpa_supplicant calls set_wpa_enabled when the driver
2002 * is loaded and unloaded, regardless of if WPA is being
2003 * used. No other calls are made which can be used to
2004 * determine if encryption will be used or not prior to
2005 * association being expected. If encryption is not being
2006 * used, drop_unencrypted is set to false, else true -- we
2007 * can use this to determine if the CAP_PRIVACY_ON bit should
2013 case IEEE_PARAM_PRIVACY_INVOKED:
2016 case IEEE_PARAM_AUTH_ALGS:
2017 ret = wpa_set_auth_algs(dev, value);
2019 case IEEE_PARAM_IEEE_802_1X:
2021 case IEEE_PARAM_WPAX_SELECT:
2030 static int wpa_mlme(struct net_device *dev, u32 command, u32 reason)
2033 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2036 case IEEE_MLME_STA_DEAUTH:
2037 if (!rtw_set_802_11_disassociate(padapter))
2040 case IEEE_MLME_STA_DISASSOC:
2041 if (!rtw_set_802_11_disassociate(padapter))
2052 static int wpa_supplicant_ioctl(struct net_device *dev, struct iw_point *p)
2054 struct ieee_param *param;
2057 if (!p->pointer || p->length != sizeof(struct ieee_param)) {
2062 param = (struct ieee_param *)rtw_malloc(p->length);
2068 if (copy_from_user(param, p->pointer, p->length)) {
2074 switch (param->cmd) {
2075 case IEEE_CMD_SET_WPA_PARAM:
2076 ret = wpa_set_param(dev, param->u.wpa_param.name, param->u.wpa_param.value);
2079 case IEEE_CMD_SET_WPA_IE:
2080 ret = rtw_set_wpa_ie((struct adapter *)rtw_netdev_priv(dev),
2081 (char *)param->u.wpa_ie.data, (u16)param->u.wpa_ie.len);
2084 case IEEE_CMD_SET_ENCRYPTION:
2085 ret = wpa_set_encryption(dev, param, p->length);
2089 ret = wpa_mlme(dev, param->u.mlme.command, param->u.mlme.reason_code);
2093 DBG_88E("Unknown WPA supplicant request: %d\n", param->cmd);
2098 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2108 #ifdef CONFIG_88EU_AP_MODE
2109 static u8 set_pairwise_key(struct adapter *padapter, struct sta_info *psta)
2111 struct cmd_obj *ph2c;
2112 struct set_stakey_parm *psetstakey_para;
2113 struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
2116 ph2c = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
2122 psetstakey_para = kzalloc(sizeof(struct set_stakey_parm), GFP_KERNEL);
2123 if (!psetstakey_para) {
2129 init_h2fwcmd_w_parm_no_rsp(ph2c, psetstakey_para, _SetStaKey_CMD_);
2131 psetstakey_para->algorithm = (u8)psta->dot118021XPrivacy;
2133 memcpy(psetstakey_para->addr, psta->hwaddr, ETH_ALEN);
2135 memcpy(psetstakey_para->key, &psta->dot118021x_UncstKey, 16);
2137 res = rtw_enqueue_cmd(pcmdpriv, ph2c);
2144 static int set_group_key(struct adapter *padapter, u8 *key, u8 alg, int keyid)
2147 struct cmd_obj *pcmd;
2148 struct setkey_parm *psetkeyparm;
2149 struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
2152 DBG_88E("%s\n", __func__);
2154 pcmd = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
2159 psetkeyparm = kzalloc(sizeof(struct setkey_parm), GFP_KERNEL);
2166 memset(psetkeyparm, 0, sizeof(struct setkey_parm));
2168 psetkeyparm->keyid = (u8)keyid;
2170 psetkeyparm->algorithm = alg;
2172 psetkeyparm->set_tx = 1;
2188 memcpy(&(psetkeyparm->key[0]), key, keylen);
2190 pcmd->cmdcode = _SetKey_CMD_;
2191 pcmd->parmbuf = (u8 *)psetkeyparm;
2192 pcmd->cmdsz = (sizeof(struct setkey_parm));
2196 INIT_LIST_HEAD(&pcmd->list);
2198 res = rtw_enqueue_cmd(pcmdpriv, pcmd);
2205 static int set_wep_key(struct adapter *padapter, u8 *key, u8 keylen, int keyid)
2220 return set_group_key(padapter, key, alg, keyid);
2223 static int rtw_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
2226 u32 wep_key_idx, wep_key_len, wep_total_len;
2227 struct ndis_802_11_wep *pwep = NULL;
2228 struct sta_info *psta = NULL, *pbcmc_sta = NULL;
2229 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2230 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
2231 struct security_priv *psecuritypriv = &(padapter->securitypriv);
2232 struct sta_priv *pstapriv = &padapter->stapriv;
2234 DBG_88E("%s\n", __func__);
2235 param->u.crypt.err = 0;
2236 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
2237 if (param_len != sizeof(struct ieee_param) + param->u.crypt.key_len) {
2241 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2242 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2243 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff) {
2244 if (param->u.crypt.idx >= WEP_KEYS) {
2249 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2251 DBG_88E("rtw_set_encryption(), sta has already been removed or never been added\n");
2256 if (strcmp(param->u.crypt.alg, "none") == 0 && (!psta)) {
2257 /* todo:clear default encryption keys */
2259 DBG_88E("clear default encryption keys, keyid =%d\n", param->u.crypt.idx);
2262 if (strcmp(param->u.crypt.alg, "WEP") == 0 && (!psta)) {
2263 DBG_88E("r871x_set_encryption, crypt.alg = WEP\n");
2264 wep_key_idx = param->u.crypt.idx;
2265 wep_key_len = param->u.crypt.key_len;
2266 DBG_88E("r871x_set_encryption, wep_key_idx=%d, len=%d\n", wep_key_idx, wep_key_len);
2267 if ((wep_key_idx >= WEP_KEYS) || (wep_key_len <= 0)) {
2272 if (wep_key_len > 0) {
2273 wep_key_len = wep_key_len <= 5 ? 5 : 13;
2274 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
2275 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
2277 DBG_88E(" r871x_set_encryption: pwep allocate fail !!!\n");
2281 memset(pwep, 0, wep_total_len);
2283 pwep->KeyLength = wep_key_len;
2284 pwep->Length = wep_total_len;
2287 pwep->KeyIndex = wep_key_idx;
2289 memcpy(pwep->KeyMaterial, param->u.crypt.key, pwep->KeyLength);
2291 if (param->u.crypt.set_tx) {
2292 DBG_88E("wep, set_tx = 1\n");
2294 psecuritypriv->ndisencryptstatus = Ndis802_11Encryption1Enabled;
2295 psecuritypriv->dot11PrivacyAlgrthm = _WEP40_;
2296 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2298 if (pwep->KeyLength == 13) {
2299 psecuritypriv->dot11PrivacyAlgrthm = _WEP104_;
2300 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2303 psecuritypriv->dot11PrivacyKeyIndex = wep_key_idx;
2305 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2307 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2309 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2311 DBG_88E("wep, set_tx = 0\n");
2313 /* don't update "psecuritypriv->dot11PrivacyAlgrthm" and */
2314 /* psecuritypriv->dot11PrivacyKeyIndex = keyid", but can rtw_set_key to cam */
2316 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2318 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2320 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2326 if (!psta && check_fwstate(pmlmepriv, WIFI_AP_STATE)) { /* group key */
2327 if (param->u.crypt.set_tx == 1) {
2328 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2329 DBG_88E("%s, set group_key, WEP\n", __func__);
2331 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2332 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2334 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2335 if (param->u.crypt.key_len == 13)
2336 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2337 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2338 DBG_88E("%s, set group_key, TKIP\n", __func__);
2339 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2340 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2341 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2343 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2344 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2346 psecuritypriv->busetkipkey = true;
2347 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2348 DBG_88E("%s, set group_key, CCMP\n", __func__);
2349 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2350 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2351 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2353 DBG_88E("%s, set group_key, none\n", __func__);
2354 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2356 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2357 psecuritypriv->binstallGrpkey = true;
2358 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2359 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2360 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2362 pbcmc_sta->ieee8021x_blocked = false;
2363 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2369 if (psecuritypriv->dot11AuthAlgrthm == dot11AuthAlgrthm_8021X && psta) { /* psk/802_1x */
2370 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
2371 if (param->u.crypt.set_tx == 1) {
2372 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2374 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2375 DBG_88E("%s, set pairwise key, WEP\n", __func__);
2377 psta->dot118021XPrivacy = _WEP40_;
2378 if (param->u.crypt.key_len == 13)
2379 psta->dot118021XPrivacy = _WEP104_;
2380 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2381 DBG_88E("%s, set pairwise key, TKIP\n", __func__);
2383 psta->dot118021XPrivacy = _TKIP_;
2386 memcpy(psta->dot11tkiptxmickey.skey, &(param->u.crypt.key[16]), 8);
2387 memcpy(psta->dot11tkiprxmickey.skey, &(param->u.crypt.key[24]), 8);
2389 psecuritypriv->busetkipkey = true;
2390 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2391 DBG_88E("%s, set pairwise key, CCMP\n", __func__);
2393 psta->dot118021XPrivacy = _AES_;
2395 DBG_88E("%s, set pairwise key, none\n", __func__);
2397 psta->dot118021XPrivacy = _NO_PRIVACY_;
2400 set_pairwise_key(padapter, psta);
2402 psta->ieee8021x_blocked = false;
2403 } else { /* group key??? */
2404 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2405 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2406 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2407 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2408 if (param->u.crypt.key_len == 13)
2409 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2410 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2411 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2413 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2414 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2417 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2418 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2420 psecuritypriv->busetkipkey = true;
2421 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2422 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2424 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2425 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2427 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2430 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2432 psecuritypriv->binstallGrpkey = true;
2434 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2436 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2438 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2440 pbcmc_sta->ieee8021x_blocked = false;
2441 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2454 static int rtw_set_beacon(struct net_device *dev, struct ieee_param *param, int len)
2457 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2458 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2459 struct sta_priv *pstapriv = &padapter->stapriv;
2460 unsigned char *pbuf = param->u.bcn_ie.buf;
2462 DBG_88E("%s, len =%d\n", __func__, len);
2464 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2467 memcpy(&pstapriv->max_num_sta, param->u.bcn_ie.reserved, 2);
2469 if ((pstapriv->max_num_sta > NUM_STA) || (pstapriv->max_num_sta <= 0))
2470 pstapriv->max_num_sta = NUM_STA;
2472 if (rtw_check_beacon_data(padapter, pbuf, (len-12-2)) == _SUCCESS)/* 12 = param header, 2:no packed */
2480 static int rtw_hostapd_sta_flush(struct net_device *dev)
2482 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2484 DBG_88E("%s\n", __func__);
2486 flush_all_cam_entry(padapter); /* clear CAM */
2488 return rtw_sta_flush(padapter);
2491 static int rtw_add_sta(struct net_device *dev, struct ieee_param *param)
2494 struct sta_info *psta = NULL;
2495 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2496 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2497 struct sta_priv *pstapriv = &padapter->stapriv;
2499 DBG_88E("rtw_add_sta(aid =%d) =%pM\n", param->u.add_sta.aid, (param->sta_addr));
2501 if (!check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)))
2504 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2505 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2506 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2509 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2511 int flags = param->u.add_sta.flags;
2513 psta->aid = param->u.add_sta.aid;/* aid = 1~2007 */
2515 memcpy(psta->bssrateset, param->u.add_sta.tx_supp_rates, 16);
2517 /* check wmm cap. */
2518 if (WLAN_STA_WME&flags)
2519 psta->qos_option = 1;
2521 psta->qos_option = 0;
2523 if (pmlmepriv->qospriv.qos_option == 0)
2524 psta->qos_option = 0;
2526 /* chec 802.11n ht cap. */
2527 if (WLAN_STA_HT&flags) {
2528 psta->htpriv.ht_option = true;
2529 psta->qos_option = 1;
2530 memcpy(&psta->htpriv.ht_cap, ¶m->u.add_sta.ht_cap,
2531 sizeof(struct ieee80211_ht_cap));
2533 psta->htpriv.ht_option = false;
2536 if (pmlmepriv->htpriv.ht_option == false)
2537 psta->htpriv.ht_option = false;
2539 update_sta_info_apmode(padapter, psta);
2547 static int rtw_del_sta(struct net_device *dev, struct ieee_param *param)
2550 struct sta_info *psta = NULL;
2551 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2552 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2553 struct sta_priv *pstapriv = &padapter->stapriv;
2556 DBG_88E("rtw_del_sta =%pM\n", (param->sta_addr));
2558 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2561 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2562 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2563 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2566 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2568 spin_lock_bh(&pstapriv->asoc_list_lock);
2569 if (!list_empty(&psta->asoc_list)) {
2570 list_del_init(&psta->asoc_list);
2571 pstapriv->asoc_list_cnt--;
2572 updated = ap_free_sta(padapter, psta, true, WLAN_REASON_DEAUTH_LEAVING);
2574 spin_unlock_bh(&pstapriv->asoc_list_lock);
2575 associated_clients_update(padapter, updated);
2578 DBG_88E("rtw_del_sta(), sta has already been removed or never been added\n");
2584 static int rtw_ioctl_get_sta_data(struct net_device *dev, struct ieee_param *param, int len)
2587 struct sta_info *psta = NULL;
2588 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2589 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2590 struct sta_priv *pstapriv = &padapter->stapriv;
2591 struct ieee_param_ex *param_ex = (struct ieee_param_ex *)param;
2592 struct sta_data *psta_data = (struct sta_data *)param_ex->data;
2594 DBG_88E("rtw_ioctl_get_sta_info, sta_addr: %pM\n", (param_ex->sta_addr));
2596 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2599 if (param_ex->sta_addr[0] == 0xff && param_ex->sta_addr[1] == 0xff &&
2600 param_ex->sta_addr[2] == 0xff && param_ex->sta_addr[3] == 0xff &&
2601 param_ex->sta_addr[4] == 0xff && param_ex->sta_addr[5] == 0xff)
2604 psta = rtw_get_stainfo(pstapriv, param_ex->sta_addr);
2606 psta_data->aid = (u16)psta->aid;
2607 psta_data->capability = psta->capability;
2608 psta_data->flags = psta->flags;
2612 no_short_slot_time_set : BIT(1)
2613 no_short_preamble_set : BIT(2)
2614 no_ht_gf_set : BIT(3)
2616 ht_20mhz_set : BIT(5)
2619 psta_data->sta_set = ((psta->nonerp_set) |
2620 (psta->no_short_slot_time_set << 1) |
2621 (psta->no_short_preamble_set << 2) |
2622 (psta->no_ht_gf_set << 3) |
2623 (psta->no_ht_set << 4) |
2624 (psta->ht_20mhz_set << 5));
2625 psta_data->tx_supp_rates_len = psta->bssratelen;
2626 memcpy(psta_data->tx_supp_rates, psta->bssrateset, psta->bssratelen);
2627 memcpy(&psta_data->ht_cap,
2628 &psta->htpriv.ht_cap, sizeof(struct ieee80211_ht_cap));
2629 psta_data->rx_pkts = psta->sta_stats.rx_data_pkts;
2630 psta_data->rx_bytes = psta->sta_stats.rx_bytes;
2631 psta_data->rx_drops = psta->sta_stats.rx_drops;
2632 psta_data->tx_pkts = psta->sta_stats.tx_pkts;
2633 psta_data->tx_bytes = psta->sta_stats.tx_bytes;
2634 psta_data->tx_drops = psta->sta_stats.tx_drops;
2642 static int rtw_get_sta_wpaie(struct net_device *dev, struct ieee_param *param)
2645 struct sta_info *psta = NULL;
2646 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2647 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2648 struct sta_priv *pstapriv = &padapter->stapriv;
2650 DBG_88E("rtw_get_sta_wpaie, sta_addr: %pM\n", (param->sta_addr));
2652 if (check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)) != true)
2655 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2656 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2657 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2660 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2662 if (psta->wpa_ie[0] == WLAN_EID_RSN ||
2663 psta->wpa_ie[0] == WLAN_EID_VENDOR_SPECIFIC) {
2667 wpa_ie_len = psta->wpa_ie[1];
2668 copy_len = min_t(int, wpa_ie_len + 2, sizeof(psta->wpa_ie));
2669 param->u.wpa_ie.len = copy_len;
2670 memcpy(param->u.wpa_ie.reserved, psta->wpa_ie, copy_len);
2672 DBG_88E("sta's wpa_ie is NONE\n");
2681 static int rtw_set_wps_beacon(struct net_device *dev, struct ieee_param *param, int len)
2684 unsigned char wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2685 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2686 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2687 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2690 DBG_88E("%s, len =%d\n", __func__, len);
2692 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2695 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2697 kfree(pmlmepriv->wps_beacon_ie);
2698 pmlmepriv->wps_beacon_ie = NULL;
2701 pmlmepriv->wps_beacon_ie = rtw_malloc(ie_len);
2702 pmlmepriv->wps_beacon_ie_len = ie_len;
2703 if (!pmlmepriv->wps_beacon_ie) {
2704 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2708 memcpy(pmlmepriv->wps_beacon_ie, param->u.bcn_ie.buf, ie_len);
2710 update_beacon(padapter, _VENDOR_SPECIFIC_IE_, wps_oui, true);
2712 pmlmeext->bstart_bss = true;
2718 static int rtw_set_wps_probe_resp(struct net_device *dev, struct ieee_param *param, int len)
2721 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2722 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2725 DBG_88E("%s, len =%d\n", __func__, len);
2727 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2730 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2732 kfree(pmlmepriv->wps_probe_resp_ie);
2733 pmlmepriv->wps_probe_resp_ie = NULL;
2736 pmlmepriv->wps_probe_resp_ie = rtw_malloc(ie_len);
2737 pmlmepriv->wps_probe_resp_ie_len = ie_len;
2738 if (!pmlmepriv->wps_probe_resp_ie) {
2739 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2742 memcpy(pmlmepriv->wps_probe_resp_ie, param->u.bcn_ie.buf, ie_len);
2748 static int rtw_set_wps_assoc_resp(struct net_device *dev, struct ieee_param *param, int len)
2751 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2752 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2755 DBG_88E("%s, len =%d\n", __func__, len);
2757 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2760 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2762 kfree(pmlmepriv->wps_assoc_resp_ie);
2763 pmlmepriv->wps_assoc_resp_ie = NULL;
2766 pmlmepriv->wps_assoc_resp_ie = rtw_malloc(ie_len);
2767 pmlmepriv->wps_assoc_resp_ie_len = ie_len;
2768 if (!pmlmepriv->wps_assoc_resp_ie) {
2769 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2773 memcpy(pmlmepriv->wps_assoc_resp_ie, param->u.bcn_ie.buf, ie_len);
2779 static int rtw_set_hidden_ssid(struct net_device *dev, struct ieee_param *param, int len)
2782 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2783 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2784 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2785 struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
2789 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2792 if (param->u.wpa_param.name != 0) /* dummy test... */
2793 DBG_88E("%s name(%u) != 0\n", __func__, param->u.wpa_param.name);
2794 value = param->u.wpa_param.value;
2796 /* use the same definition of hostapd's ignore_broadcast_ssid */
2797 if (value != 1 && value != 2)
2799 DBG_88E("%s value(%u)\n", __func__, value);
2800 pmlmeinfo->hidden_ssid_mode = value;
2804 static int rtw_ioctl_acl_remove_sta(struct net_device *dev, struct ieee_param *param, int len)
2806 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2807 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2809 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2812 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2813 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2814 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2816 return rtw_acl_remove_sta(padapter, param->sta_addr);
2819 static int rtw_ioctl_acl_add_sta(struct net_device *dev, struct ieee_param *param, int len)
2821 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2822 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2824 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2827 if (param->sta_addr[0] == 0xff && param->sta_addr[1] == 0xff &&
2828 param->sta_addr[2] == 0xff && param->sta_addr[3] == 0xff &&
2829 param->sta_addr[4] == 0xff && param->sta_addr[5] == 0xff)
2831 return rtw_acl_add_sta(padapter, param->sta_addr);
2834 static int rtw_ioctl_set_macaddr_acl(struct net_device *dev, struct ieee_param *param, int len)
2837 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2838 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2840 if (check_fwstate(pmlmepriv, WIFI_AP_STATE) != true)
2843 rtw_set_macaddr_acl(padapter, param->u.mlme.command);
2848 static int rtw_hostapd_ioctl(struct net_device *dev, struct iw_point *p)
2850 struct ieee_param *param;
2852 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2855 * this function is expect to call in master mode, which allows no power saving
2856 * so, we just check hw_init_completed
2859 if (!padapter->hw_init_completed) {
2864 if (!p->pointer || p->length != sizeof(struct ieee_param)) {
2869 param = (struct ieee_param *)rtw_malloc(p->length);
2875 if (copy_from_user(param, p->pointer, p->length)) {
2881 switch (param->cmd) {
2882 case RTL871X_HOSTAPD_FLUSH:
2883 ret = rtw_hostapd_sta_flush(dev);
2885 case RTL871X_HOSTAPD_ADD_STA:
2886 ret = rtw_add_sta(dev, param);
2888 case RTL871X_HOSTAPD_REMOVE_STA:
2889 ret = rtw_del_sta(dev, param);
2891 case RTL871X_HOSTAPD_SET_BEACON:
2892 ret = rtw_set_beacon(dev, param, p->length);
2894 case RTL871X_SET_ENCRYPTION:
2895 ret = rtw_set_encryption(dev, param, p->length);
2897 case RTL871X_HOSTAPD_GET_WPAIE_STA:
2898 ret = rtw_get_sta_wpaie(dev, param);
2900 case RTL871X_HOSTAPD_SET_WPS_BEACON:
2901 ret = rtw_set_wps_beacon(dev, param, p->length);
2903 case RTL871X_HOSTAPD_SET_WPS_PROBE_RESP:
2904 ret = rtw_set_wps_probe_resp(dev, param, p->length);
2906 case RTL871X_HOSTAPD_SET_WPS_ASSOC_RESP:
2907 ret = rtw_set_wps_assoc_resp(dev, param, p->length);
2909 case RTL871X_HOSTAPD_SET_HIDDEN_SSID:
2910 ret = rtw_set_hidden_ssid(dev, param, p->length);
2912 case RTL871X_HOSTAPD_GET_INFO_STA:
2913 ret = rtw_ioctl_get_sta_data(dev, param, p->length);
2915 case RTL871X_HOSTAPD_SET_MACADDR_ACL:
2916 ret = rtw_ioctl_set_macaddr_acl(dev, param, p->length);
2918 case RTL871X_HOSTAPD_ACL_ADD_STA:
2919 ret = rtw_ioctl_acl_add_sta(dev, param, p->length);
2921 case RTL871X_HOSTAPD_ACL_REMOVE_STA:
2922 ret = rtw_ioctl_acl_remove_sta(dev, param, p->length);
2925 DBG_88E("Unknown hostapd request: %d\n", param->cmd);
2930 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2938 #include <rtw_android.h>
2939 static int rtw_wx_set_priv(struct net_device *dev,
2940 struct iw_request_info *info,
2941 union iwreq_data *awrq,
2947 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2948 struct iw_point *dwrq = (struct iw_point *)awrq;
2950 if (dwrq->length == 0)
2958 if (copy_from_user(ext, dwrq->pointer, len)) {
2963 /* added for wps2.0 @20110524 */
2964 if (dwrq->flags == 0x8766 && len > 8) {
2966 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2967 u8 *probereq_wpsie = ext;
2968 int probereq_wpsie_len = len;
2969 u8 wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2971 if ((_VENDOR_SPECIFIC_IE_ == probereq_wpsie[0]) &&
2972 (!memcmp(&probereq_wpsie[2], wps_oui, 4))) {
2973 cp_sz = min(probereq_wpsie_len, MAX_WPS_IE_LEN);
2975 pmlmepriv->wps_probe_req_ie_len = 0;
2976 kfree(pmlmepriv->wps_probe_req_ie);
2977 pmlmepriv->wps_probe_req_ie = NULL;
2979 pmlmepriv->wps_probe_req_ie = rtw_malloc(cp_sz);
2980 if (!pmlmepriv->wps_probe_req_ie) {
2981 pr_info("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2985 memcpy(pmlmepriv->wps_probe_req_ie, probereq_wpsie, cp_sz);
2986 pmlmepriv->wps_probe_req_ie_len = cp_sz;
2991 if (len >= WEXT_CSCAN_HEADER_SIZE &&
2992 !memcmp(ext, WEXT_CSCAN_HEADER, WEXT_CSCAN_HEADER_SIZE)) {
2993 ret = rtw_wx_set_scan(dev, info, awrq, ext);
3004 static iw_handler rtw_handlers[] = {
3005 NULL, /* SIOCSIWCOMMIT */
3006 rtw_wx_get_name, /* SIOCGIWNAME */
3007 dummy, /* SIOCSIWNWID */
3008 dummy, /* SIOCGIWNWID */
3009 rtw_wx_set_freq, /* SIOCSIWFREQ */
3010 rtw_wx_get_freq, /* SIOCGIWFREQ */
3011 rtw_wx_set_mode, /* SIOCSIWMODE */
3012 rtw_wx_get_mode, /* SIOCGIWMODE */
3013 dummy, /* SIOCSIWSENS */
3014 rtw_wx_get_sens, /* SIOCGIWSENS */
3015 NULL, /* SIOCSIWRANGE */
3016 rtw_wx_get_range, /* SIOCGIWRANGE */
3017 rtw_wx_set_priv, /* SIOCSIWPRIV */
3018 NULL, /* SIOCGIWPRIV */
3019 NULL, /* SIOCSIWSTATS */
3020 NULL, /* SIOCGIWSTATS */
3021 dummy, /* SIOCSIWSPY */
3022 dummy, /* SIOCGIWSPY */
3023 NULL, /* SIOCGIWTHRSPY */
3024 NULL, /* SIOCWIWTHRSPY */
3025 rtw_wx_set_wap, /* SIOCSIWAP */
3026 rtw_wx_get_wap, /* SIOCGIWAP */
3027 rtw_wx_set_mlme, /* request MLME operation; uses struct iw_mlme */
3028 dummy, /* SIOCGIWAPLIST -- depricated */
3029 rtw_wx_set_scan, /* SIOCSIWSCAN */
3030 rtw_wx_get_scan, /* SIOCGIWSCAN */
3031 rtw_wx_set_essid, /* SIOCSIWESSID */
3032 rtw_wx_get_essid, /* SIOCGIWESSID */
3033 dummy, /* SIOCSIWNICKN */
3034 rtw_wx_get_nick, /* SIOCGIWNICKN */
3035 NULL, /* -- hole -- */
3036 NULL, /* -- hole -- */
3037 rtw_wx_set_rate, /* SIOCSIWRATE */
3038 rtw_wx_get_rate, /* SIOCGIWRATE */
3039 rtw_wx_set_rts, /* SIOCSIWRTS */
3040 rtw_wx_get_rts, /* SIOCGIWRTS */
3041 rtw_wx_set_frag, /* SIOCSIWFRAG */
3042 rtw_wx_get_frag, /* SIOCGIWFRAG */
3043 dummy, /* SIOCSIWTXPOW */
3044 dummy, /* SIOCGIWTXPOW */
3045 dummy, /* SIOCSIWRETRY */
3046 rtw_wx_get_retry, /* SIOCGIWRETRY */
3047 rtw_wx_set_enc, /* SIOCSIWENCODE */
3048 rtw_wx_get_enc, /* SIOCGIWENCODE */
3049 dummy, /* SIOCSIWPOWER */
3050 rtw_wx_get_power, /* SIOCGIWPOWER */
3051 NULL, /*---hole---*/
3052 NULL, /*---hole---*/
3053 rtw_wx_set_gen_ie, /* SIOCSIWGENIE */
3054 NULL, /* SIOCGWGENIE */
3055 rtw_wx_set_auth, /* SIOCSIWAUTH */
3056 NULL, /* SIOCGIWAUTH */
3057 rtw_wx_set_enc_ext, /* SIOCSIWENCODEEXT */
3058 NULL, /* SIOCGIWENCODEEXT */
3059 rtw_wx_set_pmkid, /* SIOCSIWPMKSA */
3060 NULL, /*---hole---*/
3063 static struct iw_statistics *rtw_get_wireless_stats(struct net_device *dev)
3065 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
3066 struct iw_statistics *piwstats = &padapter->iwstats;
3071 if (!check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
3072 piwstats->qual.qual = 0;
3073 piwstats->qual.level = 0;
3074 piwstats->qual.noise = 0;
3076 tmp_level = padapter->recvpriv.signal_strength;
3077 tmp_qual = padapter->recvpriv.signal_qual;
3078 tmp_noise = padapter->recvpriv.noise;
3080 piwstats->qual.level = tmp_level;
3081 piwstats->qual.qual = tmp_qual;
3082 piwstats->qual.noise = tmp_noise;
3084 piwstats->qual.updated = IW_QUAL_ALL_UPDATED;/* IW_QUAL_DBM; */
3085 return &padapter->iwstats;
3088 struct iw_handler_def rtw_handlers_def = {
3089 .standard = rtw_handlers,
3090 .num_standard = ARRAY_SIZE(rtw_handlers),
3091 .get_wireless_stats = rtw_get_wireless_stats,
3094 int rtw_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
3096 struct iwreq *wrq = (struct iwreq *)rq;
3100 case RTL_IOCTL_WPA_SUPPLICANT:
3101 ret = wpa_supplicant_ioctl(dev, &wrq->u.data);
3103 #ifdef CONFIG_88EU_AP_MODE
3104 case RTL_IOCTL_HOSTAPD:
3105 ret = rtw_hostapd_ioctl(dev, &wrq->u.data);
3107 #endif /* CONFIG_88EU_AP_MODE */
3108 case (SIOCDEVPRIVATE+1):
3109 ret = rtw_android_priv_cmd(dev, rq, cmd);