1 // SPDX-License-Identifier: GPL-2.0
2 /******************************************************************************
4 * Copyright(c) 2007 - 2012 Realtek Corporation. All rights reserved.
6 ******************************************************************************/
7 #define _IOCTL_LINUX_C_
9 #include <linux/ieee80211.h>
11 #include <osdep_service.h>
12 #include <drv_types.h>
13 #include <wlan_bssdef.h>
14 #include <rtw_debug.h>
17 #include <rtw_mlme_ext.h>
18 #include <rtw_ioctl.h>
19 #include <rtw_ioctl_set.h>
20 #include <rtl8188e_hal.h>
22 #include <linux/vmalloc.h>
23 #include <linux/etherdevice.h>
25 #include "osdep_intf.h"
27 #define RTL_IOCTL_WPA_SUPPLICANT (SIOCIWFIRSTPRIV + 30)
29 #define SCAN_ITEM_SIZE 768
30 #define MAX_CUSTOM_LEN 64
34 #define WEXT_CSCAN_AMOUNT 9
35 #define WEXT_CSCAN_BUF_LEN 360
36 #define WEXT_CSCAN_HEADER "CSCAN S\x01\x00\x00S\x00"
37 #define WEXT_CSCAN_HEADER_SIZE 12
38 #define WEXT_CSCAN_SSID_SECTION 'S'
39 #define WEXT_CSCAN_CHANNEL_SECTION 'C'
40 #define WEXT_CSCAN_NPROBE_SECTION 'N'
41 #define WEXT_CSCAN_ACTV_DWELL_SECTION 'A'
42 #define WEXT_CSCAN_PASV_DWELL_SECTION 'P'
43 #define WEXT_CSCAN_HOME_DWELL_SECTION 'H'
44 #define WEXT_CSCAN_TYPE_SECTION 'T'
46 static u32 rtw_rates[] = {1000000, 2000000, 5500000, 11000000,
47 6000000, 9000000, 12000000, 18000000, 24000000, 36000000,
50 static const char * const iw_operation_mode[] = {
51 "Auto", "Ad-Hoc", "Managed", "Master", "Repeater",
52 "Secondary", "Monitor"
55 void indicate_wx_scan_complete_event(struct adapter *padapter)
57 union iwreq_data wrqu;
59 memset(&wrqu, 0, sizeof(union iwreq_data));
60 wireless_send_event(padapter->pnetdev, SIOCGIWSCAN, &wrqu, NULL);
63 void rtw_indicate_wx_assoc_event(struct adapter *padapter)
65 union iwreq_data wrqu;
66 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
68 memset(&wrqu, 0, sizeof(union iwreq_data));
70 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
72 memcpy(wrqu.ap_addr.sa_data, pmlmepriv->cur_network.network.MacAddress, ETH_ALEN);
74 DBG_88E_LEVEL(_drv_always_, "assoc success\n");
75 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
78 void rtw_indicate_wx_disassoc_event(struct adapter *padapter)
80 union iwreq_data wrqu;
82 memset(&wrqu, 0, sizeof(union iwreq_data));
84 wrqu.ap_addr.sa_family = ARPHRD_ETHER;
85 eth_zero_addr(wrqu.ap_addr.sa_data);
87 DBG_88E_LEVEL(_drv_always_, "indicate disassoc\n");
88 wireless_send_event(padapter->pnetdev, SIOCGIWAP, &wrqu, NULL);
91 static char *translate_scan(struct adapter *padapter,
92 struct iw_request_info *info,
93 struct wlan_network *pnetwork,
94 char *start, char *stop)
96 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
101 char custom[MAX_CUSTOM_LEN];
103 u16 max_rate = 0, rate, ht_cap = false;
105 u8 bw_40MHz = 0, short_GI = 0;
111 iwe.u.ap_addr.sa_family = ARPHRD_ETHER;
113 memcpy(iwe.u.ap_addr.sa_data, pnetwork->network.MacAddress, ETH_ALEN);
114 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_ADDR_LEN);
117 iwe.cmd = SIOCGIWESSID;
118 iwe.u.data.flags = 1;
119 iwe.u.data.length = min_t(u16, pnetwork->network.ssid.ssid_length, 32);
120 start = iwe_stream_add_point(info, start, stop, &iwe, pnetwork->network.ssid.ssid);
122 /* parsing HT_CAP_IE */
123 p = rtw_get_ie(&pnetwork->network.ies[12], _HT_CAPABILITY_IE_, &ht_ielen, pnetwork->network.ie_length-12);
125 if (p && ht_ielen > 0) {
126 struct ieee80211_ht_cap *pht_capie;
129 pht_capie = (struct ieee80211_ht_cap *)(p + 2);
130 memcpy(&mcs_rate, pht_capie->mcs.rx_mask, 2);
131 bw_40MHz = !!(le16_to_cpu(pht_capie->cap_info) &
132 IEEE80211_HT_CAP_SUP_WIDTH);
133 short_GI = !!(le16_to_cpu(pht_capie->cap_info) &
134 (IEEE80211_HT_CAP_SGI_20 |
135 IEEE80211_HT_CAP_SGI_40));
138 /* Add the protocol name */
139 iwe.cmd = SIOCGIWNAME;
140 if ((rtw_is_cckratesonly_included((u8 *)&pnetwork->network.SupportedRates))) {
142 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bn");
144 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11b");
145 } else if ((rtw_is_cckrates_included((u8 *)&pnetwork->network.SupportedRates))) {
147 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bgn");
149 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11bg");
151 if (pnetwork->network.Configuration.DSConfig > 14) {
153 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11an");
155 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11a");
158 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11gn");
160 snprintf(iwe.u.name, IFNAMSIZ, "IEEE 802.11g");
164 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_CHAR_LEN);
167 iwe.cmd = SIOCGIWMODE;
168 memcpy(&le_tmp, rtw_get_capability_from_ie(pnetwork->network.ies), 2);
170 cap = le16_to_cpu(le_tmp);
172 if (!WLAN_CAPABILITY_IS_STA_BSS(cap)) {
173 if (cap & WLAN_CAPABILITY_ESS)
174 iwe.u.mode = IW_MODE_MASTER;
176 iwe.u.mode = IW_MODE_ADHOC;
178 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_UINT_LEN);
181 if (pnetwork->network.Configuration.DSConfig < 1)
182 pnetwork->network.Configuration.DSConfig = 1;
184 /* Add frequency/channel */
185 iwe.cmd = SIOCGIWFREQ;
186 iwe.u.freq.m = rtw_ch2freq(pnetwork->network.Configuration.DSConfig) * 100000;
188 iwe.u.freq.i = pnetwork->network.Configuration.DSConfig;
189 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_FREQ_LEN);
191 /* Add encryption capability */
192 iwe.cmd = SIOCGIWENCODE;
193 if (cap & WLAN_CAPABILITY_PRIVACY)
194 iwe.u.data.flags = IW_ENCODE_ENABLED | IW_ENCODE_NOKEY;
196 iwe.u.data.flags = IW_ENCODE_DISABLED;
197 iwe.u.data.length = 0;
198 start = iwe_stream_add_point(info, start, stop, &iwe, pnetwork->network.ssid.ssid);
200 /*Add basic and extended rates */
203 p += snprintf(p, MAX_CUSTOM_LEN - (p - custom), " Rates (Mb/s): ");
204 while (pnetwork->network.SupportedRates[i] != 0) {
205 rate = pnetwork->network.SupportedRates[i]&0x7F;
208 p += snprintf(p, MAX_CUSTOM_LEN - (p - custom),
209 "%d%s ", rate >> 1, (rate & 1) ? ".5" : "");
214 if (mcs_rate&0x8000)/* MCS15 */
215 max_rate = (bw_40MHz) ? ((short_GI) ? 300 : 270) : ((short_GI) ? 144 : 130);
216 else if (mcs_rate&0x0080)/* MCS7 */
218 else/* default MCS7 */
219 max_rate = (bw_40MHz) ? ((short_GI) ? 150 : 135) : ((short_GI) ? 72 : 65);
221 max_rate = max_rate*2;/* Mbps/2; */
224 iwe.cmd = SIOCGIWRATE;
225 iwe.u.bitrate.fixed = 0;
226 iwe.u.bitrate.disabled = 0;
227 iwe.u.bitrate.value = max_rate * 500000;
228 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_PARAM_LEN);
230 /* parsing WPA/WPA2 IE */
232 u8 buf[MAX_WPA_IE_LEN];
233 u8 wpa_ie[255], rsn_ie[255];
234 u16 wpa_len = 0, rsn_len = 0;
237 rtw_get_sec_ie(pnetwork->network.ies, pnetwork->network.ie_length, rsn_ie, &rsn_len, wpa_ie, &wpa_len);
238 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan: ssid =%s\n", pnetwork->network.ssid.ssid));
239 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan: wpa_len =%d rsn_len =%d\n", wpa_len, rsn_len));
243 memset(buf, 0, MAX_WPA_IE_LEN);
244 p += sprintf(p, "wpa_ie=");
245 for (i = 0; i < wpa_len; i++)
246 p += sprintf(p, "%02x", wpa_ie[i]);
248 memset(&iwe, 0, sizeof(iwe));
249 iwe.cmd = IWEVCUSTOM;
250 iwe.u.data.length = strlen(buf);
251 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
253 memset(&iwe, 0, sizeof(iwe));
255 iwe.u.data.length = wpa_len;
256 start = iwe_stream_add_point(info, start, stop, &iwe, wpa_ie);
260 memset(buf, 0, MAX_WPA_IE_LEN);
261 p += sprintf(p, "rsn_ie=");
262 for (i = 0; i < rsn_len; i++)
263 p += sprintf(p, "%02x", rsn_ie[i]);
264 memset(&iwe, 0, sizeof(iwe));
265 iwe.cmd = IWEVCUSTOM;
266 iwe.u.data.length = strlen(buf);
267 start = iwe_stream_add_point(info, start, stop, &iwe, buf);
269 memset(&iwe, 0, sizeof(iwe));
271 iwe.u.data.length = rsn_len;
272 start = iwe_stream_add_point(info, start, stop, &iwe, rsn_ie);
276 {/* parsing WPS IE */
277 uint cnt = 0, total_ielen;
278 u8 *wpsie_ptr = NULL;
280 u8 *ie_ptr = pnetwork->network.ies + _FIXED_IE_LENGTH_;
282 total_ielen = pnetwork->network.ie_length - _FIXED_IE_LENGTH_;
284 while (cnt < total_ielen) {
285 if (rtw_is_wps_ie(&ie_ptr[cnt], &wps_ielen) && (wps_ielen > 2)) {
286 wpsie_ptr = &ie_ptr[cnt];
288 iwe.u.data.length = (u16)wps_ielen;
289 start = iwe_stream_add_point(info, start, stop, &iwe, wpsie_ptr);
291 cnt += ie_ptr[cnt+1]+2; /* goto next */
295 /* Add quality statistics */
297 iwe.u.qual.updated = IW_QUAL_QUAL_UPDATED | IW_QUAL_LEVEL_UPDATED | IW_QUAL_NOISE_INVALID;
299 if (check_fwstate(pmlmepriv, _FW_LINKED) &&
300 is_same_network(&pmlmepriv->cur_network.network, &pnetwork->network)) {
301 ss = padapter->recvpriv.signal_strength;
302 sq = padapter->recvpriv.signal_qual;
304 ss = pnetwork->network.PhyInfo.SignalStrength;
305 sq = pnetwork->network.PhyInfo.SignalQuality;
308 iwe.u.qual.level = (u8)ss;
309 iwe.u.qual.qual = (u8)sq; /* signal quality */
310 iwe.u.qual.noise = 0; /* noise level */
311 start = iwe_stream_add_event(info, start, stop, &iwe, IW_EV_QUAL_LEN);
315 static int wpa_set_auth_algs(struct net_device *dev, u32 value)
317 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
320 if ((value & AUTH_ALG_SHARED_KEY) && (value & AUTH_ALG_OPEN_SYSTEM)) {
321 DBG_88E("wpa_set_auth_algs, AUTH_ALG_SHARED_KEY and AUTH_ALG_OPEN_SYSTEM [value:0x%x]\n", value);
322 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
323 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeAutoSwitch;
324 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Auto;
325 } else if (value & AUTH_ALG_SHARED_KEY) {
326 DBG_88E("wpa_set_auth_algs, AUTH_ALG_SHARED_KEY [value:0x%x]\n", value);
327 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
329 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeShared;
330 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Shared;
331 } else if (value & AUTH_ALG_OPEN_SYSTEM) {
332 DBG_88E("wpa_set_auth_algs, AUTH_ALG_OPEN_SYSTEM\n");
333 if (padapter->securitypriv.ndisauthtype < Ndis802_11AuthModeWPAPSK) {
334 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeOpen;
335 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
337 } else if (value & AUTH_ALG_LEAP) {
338 DBG_88E("wpa_set_auth_algs, AUTH_ALG_LEAP\n");
340 DBG_88E("wpa_set_auth_algs, error!\n");
346 static int wpa_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
349 u32 wep_key_idx, wep_key_len, wep_total_len;
350 struct ndis_802_11_wep *pwep = NULL;
351 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
352 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
353 struct security_priv *psecuritypriv = &padapter->securitypriv;
355 param->u.crypt.err = 0;
356 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
358 if (param_len < (u32)((u8 *)param->u.crypt.key - (u8 *)param) + param->u.crypt.key_len) {
363 if (is_broadcast_ether_addr(param->sta_addr)) {
364 if (param->u.crypt.idx >= WEP_KEYS) {
373 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
374 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("wpa_set_encryption, crypt.alg = WEP\n"));
375 DBG_88E("wpa_set_encryption, crypt.alg = WEP\n");
377 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
378 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
379 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
381 wep_key_idx = param->u.crypt.idx;
382 wep_key_len = param->u.crypt.key_len;
384 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("(1)wep_key_idx =%d\n", wep_key_idx));
385 DBG_88E("(1)wep_key_idx =%d\n", wep_key_idx);
387 if (wep_key_idx > WEP_KEYS)
390 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("(2)wep_key_idx =%d\n", wep_key_idx));
392 if (wep_key_len > 0) {
393 wep_key_len = wep_key_len <= 5 ? 5 : 13;
394 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
395 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
397 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, (" wpa_set_encryption: pwep allocate fail !!!\n"));
400 memset(pwep, 0, wep_total_len);
401 pwep->KeyLength = wep_key_len;
402 pwep->Length = wep_total_len;
403 if (wep_key_len == 13) {
404 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
405 padapter->securitypriv.dot118021XGrpPrivacy = _WEP104_;
411 pwep->KeyIndex = wep_key_idx;
412 pwep->KeyIndex |= 0x80000000;
413 memcpy(pwep->KeyMaterial, param->u.crypt.key, pwep->KeyLength);
414 if (param->u.crypt.set_tx) {
415 DBG_88E("wep, set_tx = 1\n");
416 if (rtw_set_802_11_add_wep(padapter, pwep) == (u8)_FAIL)
419 DBG_88E("wep, set_tx = 0\n");
420 if (wep_key_idx >= WEP_KEYS) {
424 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
425 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
426 rtw_set_key(padapter, psecuritypriv, wep_key_idx, 0);
431 if (padapter->securitypriv.dot11AuthAlgrthm == dot11AuthAlgrthm_8021X) { /* 802_1x */
432 struct sta_info *psta, *pbcmc_sta;
433 struct sta_priv *pstapriv = &padapter->stapriv;
435 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE)) { /* sta mode */
436 psta = rtw_get_stainfo(pstapriv, get_bssid(pmlmepriv));
440 if (strcmp(param->u.crypt.alg, "none") != 0)
441 psta->ieee8021x_blocked = false;
443 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
444 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
445 psta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
447 if (param->u.crypt.set_tx == 1) { /* pairwise key */
448 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
450 if (strcmp(param->u.crypt.alg, "TKIP") == 0) { /* set mic key */
451 memcpy(psta->dot11tkiptxmickey.skey, &(param->u.crypt.key[16]), 8);
452 memcpy(psta->dot11tkiprxmickey.skey, &(param->u.crypt.key[24]), 8);
453 padapter->securitypriv.busetkipkey = false;
456 DBG_88E(" ~~~~set sta key:unicastkey\n");
458 rtw_setstakey_cmd(padapter, (unsigned char *)psta, true);
459 } else { /* group key */
460 memcpy(padapter->securitypriv.dot118021XGrpKey[param->u.crypt.idx].skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16 ));
461 memcpy(padapter->securitypriv.dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
462 memcpy(padapter->securitypriv.dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
463 padapter->securitypriv.binstallGrpkey = true;
464 DBG_88E(" ~~~~set sta key:groupkey\n");
466 padapter->securitypriv.dot118021XGrpKeyid = param->u.crypt.idx;
468 rtw_set_key(padapter, &padapter->securitypriv, param->u.crypt.idx, 1);
471 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
475 /* Jeff: don't disable ieee8021x_blocked while clearing key */
476 if (strcmp(param->u.crypt.alg, "none") != 0)
477 pbcmc_sta->ieee8021x_blocked = false;
479 if ((padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption2Enabled) ||
480 (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption3Enabled))
481 pbcmc_sta->dot118021XPrivacy = padapter->securitypriv.dot11PrivacyAlgrthm;
492 static int rtw_set_wpa_ie(struct adapter *padapter, char *pie, unsigned short ielen)
495 int group_cipher = 0, pairwise_cipher = 0;
498 if ((ielen > MAX_WPA_IE_LEN) || (!pie)) {
499 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
507 buf = kmemdup(pie, ielen, GFP_KERNEL);
517 DBG_88E("\n wpa_ie(length:%d):\n", ielen);
518 for (i = 0; i < ielen; i += 8)
519 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]);
522 if (ielen < RSN_HEADER_LEN) {
523 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("Ie len too short %d\n", ielen));
528 if (rtw_parse_wpa_ie(buf, ielen, &group_cipher, &pairwise_cipher, NULL) == _SUCCESS) {
529 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X;
530 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPAPSK;
531 memcpy(padapter->securitypriv.supplicant_ie, &buf[0], ielen);
534 if (rtw_parse_wpa2_ie(buf, ielen, &group_cipher, &pairwise_cipher, NULL) == _SUCCESS) {
535 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X;
536 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPA2PSK;
537 memcpy(padapter->securitypriv.supplicant_ie, &buf[0], ielen);
540 switch (group_cipher) {
541 case WPA_CIPHER_NONE:
542 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
543 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
545 case WPA_CIPHER_WEP40:
546 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
547 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
549 case WPA_CIPHER_TKIP:
550 padapter->securitypriv.dot118021XGrpPrivacy = _TKIP_;
551 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
553 case WPA_CIPHER_CCMP:
554 padapter->securitypriv.dot118021XGrpPrivacy = _AES_;
555 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
557 case WPA_CIPHER_WEP104:
558 padapter->securitypriv.dot118021XGrpPrivacy = _WEP104_;
559 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
563 switch (pairwise_cipher) {
564 case WPA_CIPHER_NONE:
565 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
566 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
568 case WPA_CIPHER_WEP40:
569 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
570 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
572 case WPA_CIPHER_TKIP:
573 padapter->securitypriv.dot11PrivacyAlgrthm = _TKIP_;
574 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
576 case WPA_CIPHER_CCMP:
577 padapter->securitypriv.dot11PrivacyAlgrthm = _AES_;
578 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
580 case WPA_CIPHER_WEP104:
581 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
582 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
586 _clr_fwstate_(&padapter->mlmepriv, WIFI_UNDER_WPS);
589 u8 eid, wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
591 while (cnt < ielen) {
593 if ((eid == _VENDOR_SPECIFIC_IE_) && (!memcmp(&buf[cnt+2], wps_oui, 4))) {
594 DBG_88E("SET WPS_IE\n");
596 padapter->securitypriv.wps_ie_len = min(buf[cnt + 1] + 2, MAX_WPA_IE_LEN << 2);
598 memcpy(padapter->securitypriv.wps_ie, &buf[cnt], padapter->securitypriv.wps_ie_len);
600 set_fwstate(&padapter->mlmepriv, WIFI_UNDER_WPS);
604 cnt += buf[cnt+1]+2; /* goto next */
610 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
611 ("rtw_set_wpa_ie: pairwise_cipher = 0x%08x padapter->securitypriv.ndisencryptstatus =%d padapter->securitypriv.ndisauthtype =%d\n",
612 pairwise_cipher, padapter->securitypriv.ndisencryptstatus, padapter->securitypriv.ndisauthtype));
618 typedef unsigned char NDIS_802_11_RATES_EX[NDIS_802_11_LENGTH_RATES_EX];
620 static int rtw_wx_get_name(struct net_device *dev,
621 struct iw_request_info *info,
622 union iwreq_data *wrqu, char *extra)
624 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
628 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
629 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
630 NDIS_802_11_RATES_EX *prates = NULL;
632 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("cmd_code =%x\n", info->cmd));
634 if (check_fwstate(pmlmepriv, _FW_LINKED | WIFI_ADHOC_MASTER_STATE)) {
635 /* parsing HT_CAP_IE */
636 p = rtw_get_ie(&pcur_bss->ies[12], _HT_CAPABILITY_IE_, &ht_ielen, pcur_bss->ie_length-12);
637 if (p && ht_ielen > 0)
640 prates = &pcur_bss->SupportedRates;
642 if (rtw_is_cckratesonly_included((u8 *)prates)) {
644 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bn");
646 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11b");
647 } else if (rtw_is_cckrates_included((u8 *)prates)) {
649 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bgn");
651 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11bg");
653 if (pcur_bss->Configuration.DSConfig > 14) {
655 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11an");
657 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11a");
660 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11gn");
662 snprintf(wrqu->name, IFNAMSIZ, "IEEE 802.11g");
666 snprintf(wrqu->name, IFNAMSIZ, "unassociated");
671 static int rtw_wx_set_freq(struct net_device *dev,
672 struct iw_request_info *info,
673 union iwreq_data *wrqu, char *extra)
675 RT_TRACE(_module_rtl871x_mlme_c_, _drv_notice_, ("+rtw_wx_set_freq\n"));
679 static int rtw_wx_get_freq(struct net_device *dev,
680 struct iw_request_info *info,
681 union iwreq_data *wrqu, char *extra)
683 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
684 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
685 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
687 if (check_fwstate(pmlmepriv, _FW_LINKED)) {
688 /* wrqu->freq.m = ieee80211_wlan_frequencies[pcur_bss->Configuration.DSConfig-1] * 100000; */
689 wrqu->freq.m = rtw_ch2freq(pcur_bss->Configuration.DSConfig) * 100000;
691 wrqu->freq.i = pcur_bss->Configuration.DSConfig;
693 wrqu->freq.m = rtw_ch2freq(padapter->mlmeextpriv.cur_channel) * 100000;
695 wrqu->freq.i = padapter->mlmeextpriv.cur_channel;
701 static int rtw_wx_set_mode(struct net_device *dev, struct iw_request_info *a,
702 union iwreq_data *wrqu, char *b)
704 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
705 enum ndis_802_11_network_infra networkType;
708 if (_FAIL == rtw_pwr_wakeup(padapter)) {
713 if (!padapter->hw_init_completed) {
718 switch (wrqu->mode) {
720 networkType = Ndis802_11AutoUnknown;
721 DBG_88E("set_mode = IW_MODE_AUTO\n");
724 networkType = Ndis802_11IBSS;
725 DBG_88E("set_mode = IW_MODE_ADHOC\n");
728 networkType = Ndis802_11APMode;
729 DBG_88E("set_mode = IW_MODE_MASTER\n");
732 networkType = Ndis802_11Infrastructure;
733 DBG_88E("set_mode = IW_MODE_INFRA\n");
737 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_err_, ("\n Mode: %s is not supported\n", iw_operation_mode[wrqu->mode]));
740 if (!rtw_set_802_11_infrastructure_mode(padapter, networkType)) {
744 rtw_setopmode_cmd(padapter, networkType);
749 static int rtw_wx_get_mode(struct net_device *dev, struct iw_request_info *a,
750 union iwreq_data *wrqu, char *b)
752 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
753 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
755 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_get_mode\n"));
757 if (check_fwstate(pmlmepriv, WIFI_STATION_STATE))
758 wrqu->mode = IW_MODE_INFRA;
759 else if ((check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) ||
760 (check_fwstate(pmlmepriv, WIFI_ADHOC_STATE)))
761 wrqu->mode = IW_MODE_ADHOC;
762 else if (check_fwstate(pmlmepriv, WIFI_AP_STATE))
763 wrqu->mode = IW_MODE_MASTER;
765 wrqu->mode = IW_MODE_AUTO;
770 static int rtw_wx_set_pmkid(struct net_device *dev,
771 struct iw_request_info *a,
772 union iwreq_data *wrqu, char *extra)
774 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
775 u8 j, blInserted = false;
777 struct security_priv *psecuritypriv = &padapter->securitypriv;
778 struct iw_pmksa *pPMK = (struct iw_pmksa *)extra;
779 u8 strZeroMacAddress[ETH_ALEN] = {0x00};
780 u8 strIssueBssid[ETH_ALEN] = {0x00};
782 memcpy(strIssueBssid, pPMK->bssid.sa_data, ETH_ALEN);
783 if (pPMK->cmd == IW_PMKSA_ADD) {
784 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_ADD!\n");
785 if (!memcmp(strIssueBssid, strZeroMacAddress, ETH_ALEN))
791 /* overwrite PMKID */
792 for (j = 0; j < NUM_PMKID_CACHE; j++) {
793 if (!memcmp(psecuritypriv->PMKIDList[j].Bssid, strIssueBssid, ETH_ALEN)) {
794 /* BSSID is matched, the same AP => rewrite with new PMKID. */
795 DBG_88E("[rtw_wx_set_pmkid] BSSID exists in the PMKList.\n");
796 memcpy(psecuritypriv->PMKIDList[j].PMKID, pPMK->pmkid, IW_PMKID_LEN);
797 psecuritypriv->PMKIDList[j].bUsed = true;
798 psecuritypriv->PMKIDIndex = j+1;
805 /* Find a new entry */
806 DBG_88E("[rtw_wx_set_pmkid] Use the new entry index = %d for this PMKID.\n",
807 psecuritypriv->PMKIDIndex);
809 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].Bssid, strIssueBssid, ETH_ALEN);
810 memcpy(psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].PMKID, pPMK->pmkid, IW_PMKID_LEN);
812 psecuritypriv->PMKIDList[psecuritypriv->PMKIDIndex].bUsed = true;
813 psecuritypriv->PMKIDIndex++;
814 if (psecuritypriv->PMKIDIndex == 16)
815 psecuritypriv->PMKIDIndex = 0;
817 } else if (pPMK->cmd == IW_PMKSA_REMOVE) {
818 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_REMOVE!\n");
820 for (j = 0; j < NUM_PMKID_CACHE; j++) {
821 if (!memcmp(psecuritypriv->PMKIDList[j].Bssid, strIssueBssid, ETH_ALEN)) {
822 /* BSSID is matched, the same AP => Remove this PMKID information and reset it. */
823 eth_zero_addr(psecuritypriv->PMKIDList[j].Bssid);
824 psecuritypriv->PMKIDList[j].bUsed = false;
828 } else if (pPMK->cmd == IW_PMKSA_FLUSH) {
829 DBG_88E("[rtw_wx_set_pmkid] IW_PMKSA_FLUSH!\n");
830 memset(&psecuritypriv->PMKIDList[0], 0x00, sizeof(struct rt_pmkid_list) * NUM_PMKID_CACHE);
831 psecuritypriv->PMKIDIndex = 0;
837 static int rtw_wx_get_sens(struct net_device *dev,
838 struct iw_request_info *info,
839 union iwreq_data *wrqu, char *extra)
841 wrqu->sens.value = 0;
842 wrqu->sens.fixed = 0; /* no auto select */
843 wrqu->sens.disabled = 1;
847 static int rtw_wx_get_range(struct net_device *dev,
848 struct iw_request_info *info,
849 union iwreq_data *wrqu, char *extra)
851 struct iw_range *range = (struct iw_range *)extra;
852 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
853 struct mlme_ext_priv *pmlmeext = &padapter->mlmeextpriv;
858 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_range. cmd_code =%x\n", info->cmd));
860 wrqu->data.length = sizeof(*range);
861 memset(range, 0, sizeof(*range));
863 /* Let's try to keep this struct in the same order as in
864 * linux/include/wireless.h
867 /* TODO: See what values we can set, and remove the ones we can't
868 * set, or fill them with some default data.
871 /* ~5 Mb/s real (802.11b) */
872 range->throughput = 5 * 1000 * 1000;
874 /* signal level threshold range */
876 /* percent values between 0 and 100. */
877 range->max_qual.qual = 100;
878 range->max_qual.level = 100;
879 range->max_qual.noise = 100;
880 range->max_qual.updated = 7; /* Updated all three */
882 range->avg_qual.qual = 92; /* > 8% missed beacons is 'bad' */
883 /* TODO: Find real 'good' to 'bad' threshol value for RSSI */
884 range->avg_qual.level = 178; /* -78 dBm */
885 range->avg_qual.noise = 0;
886 range->avg_qual.updated = 7; /* Updated all three */
888 range->num_bitrates = RATE_COUNT;
890 for (i = 0; i < RATE_COUNT && i < IW_MAX_BITRATES; i++)
891 range->bitrate[i] = rtw_rates[i];
893 range->min_frag = MIN_FRAG_THRESHOLD;
894 range->max_frag = MAX_FRAG_THRESHOLD;
898 range->we_version_compiled = WIRELESS_EXT;
899 range->we_version_source = 16;
901 for (i = 0, val = 0; i < MAX_CHANNEL_NUM; i++) {
902 /* Include only legal frequencies for some countries */
903 if (pmlmeext->channel_set[i].ChannelNum != 0) {
904 range->freq[val].i = pmlmeext->channel_set[i].ChannelNum;
905 range->freq[val].m = rtw_ch2freq(pmlmeext->channel_set[i].ChannelNum) * 100000;
906 range->freq[val].e = 1;
910 if (val == IW_MAX_FREQUENCIES)
914 range->num_channels = val;
915 range->num_frequency = val;
917 /* The following code will proivde the security capability to network manager. */
918 /* If the driver doesn't provide this capability to network manager, */
919 /* the WPA/WPA2 routers can't be chosen in the network manager. */
922 #define IW_SCAN_CAPA_NONE 0x00
923 #define IW_SCAN_CAPA_ESSID 0x01
924 #define IW_SCAN_CAPA_BSSID 0x02
925 #define IW_SCAN_CAPA_CHANNEL 0x04
926 #define IW_SCAN_CAPA_MODE 0x08
927 #define IW_SCAN_CAPA_RATE 0x10
928 #define IW_SCAN_CAPA_TYPE 0x20
929 #define IW_SCAN_CAPA_TIME 0x40
932 range->enc_capa = IW_ENC_CAPA_WPA | IW_ENC_CAPA_WPA2 |
933 IW_ENC_CAPA_CIPHER_TKIP | IW_ENC_CAPA_CIPHER_CCMP;
935 range->scan_capa = IW_SCAN_CAPA_ESSID | IW_SCAN_CAPA_TYPE |
936 IW_SCAN_CAPA_BSSID | IW_SCAN_CAPA_CHANNEL |
937 IW_SCAN_CAPA_MODE | IW_SCAN_CAPA_RATE;
942 /* s1. rtw_set_802_11_infrastructure_mode() */
943 /* s2. rtw_set_802_11_authentication_mode() */
944 /* s3. set_802_11_encryption_mode() */
945 /* s4. rtw_set_802_11_bssid() */
946 static int rtw_wx_set_wap(struct net_device *dev,
947 struct iw_request_info *info,
948 union iwreq_data *awrq,
952 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
953 struct sockaddr *temp = (struct sockaddr *)awrq;
954 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
955 struct list_head *phead;
956 u8 *dst_bssid, *src_bssid;
957 struct __queue *queue = &(pmlmepriv->scanned_queue);
958 struct wlan_network *pnetwork = NULL;
959 enum ndis_802_11_auth_mode authmode;
961 if (_FAIL == rtw_pwr_wakeup(padapter)) {
966 if (!padapter->bup) {
971 if (temp->sa_family != ARPHRD_ETHER) {
976 authmode = padapter->securitypriv.ndisauthtype;
977 spin_lock_bh(&queue->lock);
978 phead = get_list_head(queue);
979 pmlmepriv->pscanned = phead->next;
981 while (phead != pmlmepriv->pscanned) {
982 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
984 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
986 dst_bssid = pnetwork->network.MacAddress;
988 src_bssid = temp->sa_data;
990 if ((!memcmp(dst_bssid, src_bssid, ETH_ALEN))) {
991 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
993 spin_unlock_bh(&queue->lock);
1000 spin_unlock_bh(&queue->lock);
1002 rtw_set_802_11_authentication_mode(padapter, authmode);
1003 if (!rtw_set_802_11_bssid(padapter, temp->sa_data)) {
1013 static int rtw_wx_get_wap(struct net_device *dev,
1014 struct iw_request_info *info,
1015 union iwreq_data *wrqu, char *extra)
1017 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1018 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1019 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
1021 wrqu->ap_addr.sa_family = ARPHRD_ETHER;
1023 eth_zero_addr(wrqu->ap_addr.sa_data);
1025 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_wap\n"));
1027 if (check_fwstate(pmlmepriv, _FW_LINKED) ||
1028 check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE) ||
1029 check_fwstate(pmlmepriv, WIFI_AP_STATE))
1030 memcpy(wrqu->ap_addr.sa_data, pcur_bss->MacAddress, ETH_ALEN);
1032 eth_zero_addr(wrqu->ap_addr.sa_data);
1036 static int rtw_wx_set_mlme(struct net_device *dev,
1037 struct iw_request_info *info,
1038 union iwreq_data *wrqu, char *extra)
1042 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1043 struct iw_mlme *mlme = (struct iw_mlme *)extra;
1048 DBG_88E("%s\n", __func__);
1050 reason = mlme->reason_code;
1052 DBG_88E("%s, cmd =%d, reason =%d\n", __func__, mlme->cmd, reason);
1054 switch (mlme->cmd) {
1055 case IW_MLME_DEAUTH:
1056 if (!rtw_set_802_11_disassociate(padapter))
1059 case IW_MLME_DISASSOC:
1060 if (!rtw_set_802_11_disassociate(padapter))
1069 static int rtw_wx_set_scan(struct net_device *dev, struct iw_request_info *a,
1070 union iwreq_data *wrqu, char *extra)
1074 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1075 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
1076 struct ndis_802_11_ssid ssid[RTW_SSID_SCAN_AMOUNT];
1078 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_set_scan\n"));
1080 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1085 if (padapter->bDriverStopped) {
1086 DBG_88E("bDriverStopped =%d\n", padapter->bDriverStopped);
1091 if (!padapter->bup) {
1096 if (!padapter->hw_init_completed) {
1101 /* When Busy Traffic, driver do not site survey. So driver return success. */
1102 /* wpa_supplicant will not issue SIOCSIWSCAN cmd again after scan timeout. */
1103 /* modify by thomas 2011-02-22. */
1104 if (pmlmepriv->LinkDetectInfo.bBusyTraffic) {
1105 indicate_wx_scan_complete_event(padapter);
1109 if (check_fwstate(pmlmepriv, _FW_UNDER_SURVEY|_FW_UNDER_LINKING)) {
1110 indicate_wx_scan_complete_event(padapter);
1114 /* For the DMP WiFi Display project, the driver won't to scan because */
1115 /* the pmlmepriv->scan_interval is always equal to 3. */
1116 /* So, the wpa_supplicant won't find out the WPS SoftAP. */
1118 memset(ssid, 0, sizeof(struct ndis_802_11_ssid)*RTW_SSID_SCAN_AMOUNT);
1120 if (wrqu->data.length == sizeof(struct iw_scan_req)) {
1121 struct iw_scan_req *req = (struct iw_scan_req *)extra;
1123 if (wrqu->data.flags & IW_SCAN_THIS_ESSID) {
1124 int len = min_t(int, req->essid_len,
1127 memcpy(ssid[0].ssid, req->essid, len);
1128 ssid[0].ssid_length = len;
1130 DBG_88E("IW_SCAN_THIS_ESSID, ssid =%s, len =%d\n", req->essid, req->essid_len);
1132 spin_lock_bh(&pmlmepriv->lock);
1134 _status = rtw_sitesurvey_cmd(padapter, ssid, 1, NULL, 0);
1136 spin_unlock_bh(&pmlmepriv->lock);
1137 } else if (req->scan_type == IW_SCAN_TYPE_PASSIVE) {
1138 DBG_88E("rtw_wx_set_scan, req->scan_type == IW_SCAN_TYPE_PASSIVE\n");
1141 if (wrqu->data.length >= WEXT_CSCAN_HEADER_SIZE &&
1142 !memcmp(extra, WEXT_CSCAN_HEADER, WEXT_CSCAN_HEADER_SIZE)) {
1143 int len = wrqu->data.length - WEXT_CSCAN_HEADER_SIZE;
1144 char *pos = extra+WEXT_CSCAN_HEADER_SIZE;
1154 case WEXT_CSCAN_SSID_SECTION:
1159 sec_len = *(pos++); len -= 1;
1160 if (sec_len > 0 && sec_len <= len) {
1161 ssid[ssid_index].ssid_length = sec_len;
1162 memcpy(ssid[ssid_index].ssid, pos, ssid[ssid_index].ssid_length);
1168 case WEXT_CSCAN_TYPE_SECTION:
1169 case WEXT_CSCAN_CHANNEL_SECTION:
1173 case WEXT_CSCAN_PASV_DWELL_SECTION:
1174 case WEXT_CSCAN_HOME_DWELL_SECTION:
1175 case WEXT_CSCAN_ACTV_DWELL_SECTION:
1180 len = 0; /* stop parsing */
1184 /* it has still some scan parameter to parse, we only do this now... */
1185 _status = rtw_set_802_11_bssid_list_scan(padapter, ssid, RTW_SSID_SCAN_AMOUNT);
1187 _status = rtw_set_802_11_bssid_list_scan(padapter, NULL, 0);
1199 static int rtw_wx_get_scan(struct net_device *dev, struct iw_request_info *a,
1200 union iwreq_data *wrqu, char *extra)
1202 struct list_head *plist, *phead;
1203 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1204 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1205 struct __queue *queue = &(pmlmepriv->scanned_queue);
1206 struct wlan_network *pnetwork = NULL;
1208 char *stop = ev + wrqu->data.length;
1211 u32 wait_for_surveydone;
1214 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_scan\n"));
1215 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, (" Start of Query SIOCGIWSCAN .\n"));
1217 if (padapter->pwrctrlpriv.brfoffbyhw && padapter->bDriverStopped) {
1222 wait_for_surveydone = 100;
1224 wait_status = _FW_UNDER_SURVEY | _FW_UNDER_LINKING;
1226 while (check_fwstate(pmlmepriv, wait_status)) {
1229 if (cnt > wait_for_surveydone)
1233 spin_lock_bh(&(pmlmepriv->scanned_queue.lock));
1235 phead = get_list_head(queue);
1236 plist = phead->next;
1238 while (phead != plist) {
1239 if ((stop - ev) < SCAN_ITEM_SIZE) {
1244 pnetwork = container_of(plist, struct wlan_network, list);
1246 /* report network only if the current channel set contains the channel to which this network belongs */
1247 if (rtw_ch_set_search_ch(padapter->mlmeextpriv.channel_set, pnetwork->network.Configuration.DSConfig) >= 0)
1248 ev = translate_scan(padapter, a, pnetwork, ev, stop);
1250 plist = plist->next;
1253 spin_unlock_bh(&pmlmepriv->scanned_queue.lock);
1255 wrqu->data.length = ev-extra;
1256 wrqu->data.flags = 0;
1263 /* s1. rtw_set_802_11_infrastructure_mode() */
1264 /* s2. set_802_11_authenticaion_mode() */
1265 /* s3. set_802_11_encryption_mode() */
1266 /* s4. rtw_set_802_11_ssid() */
1267 static int rtw_wx_set_essid(struct net_device *dev,
1268 struct iw_request_info *a,
1269 union iwreq_data *wrqu, char *extra)
1271 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1272 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
1273 struct __queue *queue = &pmlmepriv->scanned_queue;
1274 struct list_head *phead;
1275 struct wlan_network *pnetwork = NULL;
1276 enum ndis_802_11_auth_mode authmode;
1277 struct ndis_802_11_ssid ndis_ssid;
1278 u8 *dst_ssid, *src_ssid;
1282 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1283 ("+rtw_wx_set_essid: fw_state = 0x%08x\n", get_fwstate(pmlmepriv)));
1284 if (_FAIL == rtw_pwr_wakeup(padapter)) {
1289 if (!padapter->bup) {
1294 if (wrqu->essid.length > IW_ESSID_MAX_SIZE) {
1299 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
1304 authmode = padapter->securitypriv.ndisauthtype;
1305 DBG_88E("=>%s\n", __func__);
1306 if (wrqu->essid.flags && wrqu->essid.length) {
1307 len = min_t(uint, wrqu->essid.length, IW_ESSID_MAX_SIZE);
1309 if (wrqu->essid.length != 33)
1310 DBG_88E("ssid =%s, len =%d\n", extra, wrqu->essid.length);
1312 memset(&ndis_ssid, 0, sizeof(struct ndis_802_11_ssid));
1313 ndis_ssid.ssid_length = len;
1314 memcpy(ndis_ssid.ssid, extra, len);
1315 src_ssid = ndis_ssid.ssid;
1317 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("rtw_wx_set_essid: ssid =[%s]\n", src_ssid));
1318 spin_lock_bh(&queue->lock);
1319 phead = get_list_head(queue);
1320 pmlmepriv->pscanned = phead->next;
1322 while (phead != pmlmepriv->pscanned) {
1323 pnetwork = container_of(pmlmepriv->pscanned, struct wlan_network, list);
1325 pmlmepriv->pscanned = pmlmepriv->pscanned->next;
1327 dst_ssid = pnetwork->network.ssid.ssid;
1329 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1330 ("rtw_wx_set_essid: dst_ssid =%s\n",
1331 pnetwork->network.ssid.ssid));
1333 if ((!memcmp(dst_ssid, src_ssid, ndis_ssid.ssid_length)) &&
1334 (pnetwork->network.ssid.ssid_length == ndis_ssid.ssid_length)) {
1335 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1336 ("rtw_wx_set_essid: find match, set infra mode\n"));
1338 if (check_fwstate(pmlmepriv, WIFI_ADHOC_STATE)) {
1339 if (pnetwork->network.InfrastructureMode != pmlmepriv->cur_network.network.InfrastructureMode)
1343 if (!rtw_set_802_11_infrastructure_mode(padapter, pnetwork->network.InfrastructureMode)) {
1345 spin_unlock_bh(&queue->lock);
1352 spin_unlock_bh(&queue->lock);
1353 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1354 ("set ssid: set_802_11_auth. mode =%d\n", authmode));
1355 rtw_set_802_11_authentication_mode(padapter, authmode);
1356 if (!rtw_set_802_11_ssid(padapter, &ndis_ssid)) {
1363 DBG_88E("<=%s, ret %d\n", __func__, ret);
1368 static int rtw_wx_get_essid(struct net_device *dev,
1369 struct iw_request_info *a,
1370 union iwreq_data *wrqu, char *extra)
1373 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1374 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1375 struct wlan_bssid_ex *pcur_bss = &pmlmepriv->cur_network.network;
1377 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, ("rtw_wx_get_essid\n"));
1379 if ((check_fwstate(pmlmepriv, _FW_LINKED)) ||
1380 (check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE))) {
1381 len = pcur_bss->ssid.ssid_length;
1382 memcpy(extra, pcur_bss->ssid.ssid, len);
1387 wrqu->essid.length = len;
1388 wrqu->essid.flags = 1;
1393 static int rtw_wx_set_rate(struct net_device *dev,
1394 struct iw_request_info *a,
1395 union iwreq_data *wrqu, char *extra)
1398 u8 datarates[NumRates];
1399 u32 target_rate = wrqu->bitrate.value;
1400 u32 fixed = wrqu->bitrate.fixed;
1402 u8 mpdatarate[NumRates] = {11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0xff};
1404 RT_TRACE(_module_rtl871x_mlme_c_, _drv_info_, (" rtw_wx_set_rate\n"));
1405 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("target_rate = %d, fixed = %d\n", target_rate, fixed));
1407 if (target_rate == -1) {
1411 target_rate = target_rate/100000;
1413 switch (target_rate) {
1457 for (i = 0; i < NumRates; i++) {
1458 if (ratevalue == mpdatarate[i]) {
1459 datarates[i] = mpdatarate[i];
1463 datarates[i] = 0xff;
1466 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_, ("datarate_inx =%d\n", datarates[i]));
1472 static int rtw_wx_get_rate(struct net_device *dev,
1473 struct iw_request_info *info,
1474 union iwreq_data *wrqu, char *extra)
1478 max_rate = rtw_get_cur_max_rate((struct adapter *)rtw_netdev_priv(dev));
1483 wrqu->bitrate.fixed = 0; /* no auto select */
1484 wrqu->bitrate.value = max_rate * 100000;
1489 static int rtw_wx_set_rts(struct net_device *dev,
1490 struct iw_request_info *info,
1491 union iwreq_data *wrqu, char *extra)
1493 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1495 if (wrqu->rts.disabled) {
1496 padapter->registrypriv.rts_thresh = 2347;
1498 if (wrqu->rts.value < 0 ||
1499 wrqu->rts.value > 2347)
1502 padapter->registrypriv.rts_thresh = wrqu->rts.value;
1505 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1510 static int rtw_wx_get_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);
1516 DBG_88E("%s, rts_thresh =%d\n", __func__, padapter->registrypriv.rts_thresh);
1518 wrqu->rts.value = padapter->registrypriv.rts_thresh;
1519 wrqu->rts.fixed = 0; /* no auto select */
1520 /* wrqu->rts.disabled = (wrqu->rts.value == DEFAULT_RTS_THRESHOLD); */
1525 static int rtw_wx_set_frag(struct net_device *dev,
1526 struct iw_request_info *info,
1527 union iwreq_data *wrqu, char *extra)
1529 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1531 if (wrqu->frag.disabled) {
1532 padapter->xmitpriv.frag_len = MAX_FRAG_THRESHOLD;
1534 if (wrqu->frag.value < MIN_FRAG_THRESHOLD ||
1535 wrqu->frag.value > MAX_FRAG_THRESHOLD)
1538 padapter->xmitpriv.frag_len = wrqu->frag.value & ~0x1;
1541 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1546 static int rtw_wx_get_frag(struct net_device *dev,
1547 struct iw_request_info *info,
1548 union iwreq_data *wrqu, char *extra)
1550 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1552 DBG_88E("%s, frag_len =%d\n", __func__, padapter->xmitpriv.frag_len);
1554 wrqu->frag.value = padapter->xmitpriv.frag_len;
1555 wrqu->frag.fixed = 0; /* no auto select */
1560 static int rtw_wx_get_retry(struct net_device *dev,
1561 struct iw_request_info *info,
1562 union iwreq_data *wrqu, char *extra)
1564 wrqu->retry.value = 7;
1565 wrqu->retry.fixed = 0; /* no auto select */
1566 wrqu->retry.disabled = 1;
1571 static int rtw_wx_set_enc(struct net_device *dev,
1572 struct iw_request_info *info,
1573 union iwreq_data *wrqu, char *keybuf)
1576 u32 keyindex_provided;
1577 struct ndis_802_11_wep wep;
1578 enum ndis_802_11_auth_mode authmode;
1580 struct iw_point *erq = &(wrqu->encoding);
1581 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1582 struct pwrctrl_priv *pwrpriv = &padapter->pwrctrlpriv;
1584 DBG_88E("+rtw_wx_set_enc, flags = 0x%x\n", erq->flags);
1586 memset(&wep, 0, sizeof(struct ndis_802_11_wep));
1588 key = erq->flags & IW_ENCODE_INDEX;
1590 if (erq->flags & IW_ENCODE_DISABLED) {
1591 DBG_88E("EncryptionDisabled\n");
1592 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1593 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1594 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1595 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1596 authmode = Ndis802_11AuthModeOpen;
1597 padapter->securitypriv.ndisauthtype = authmode;
1606 keyindex_provided = 1;
1608 keyindex_provided = 0;
1609 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1610 DBG_88E("rtw_wx_set_enc, key =%d\n", key);
1613 /* set authentication mode */
1614 if (erq->flags & IW_ENCODE_OPEN) {
1615 DBG_88E("rtw_wx_set_enc():IW_ENCODE_OPEN\n");
1616 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
1617 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1618 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1619 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1620 authmode = Ndis802_11AuthModeOpen;
1621 padapter->securitypriv.ndisauthtype = authmode;
1622 } else if (erq->flags & IW_ENCODE_RESTRICTED) {
1623 DBG_88E("rtw_wx_set_enc():IW_ENCODE_RESTRICTED\n");
1624 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;
1625 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Shared;
1626 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1627 padapter->securitypriv.dot118021XGrpPrivacy = _WEP40_;
1628 authmode = Ndis802_11AuthModeShared;
1629 padapter->securitypriv.ndisauthtype = authmode;
1631 DBG_88E("rtw_wx_set_enc():erq->flags = 0x%x\n", erq->flags);
1633 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption1Enabled;/* Ndis802_11EncryptionDisabled; */
1634 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1635 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1636 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1637 authmode = Ndis802_11AuthModeOpen;
1638 padapter->securitypriv.ndisauthtype = authmode;
1642 if (erq->length > 0) {
1643 wep.KeyLength = erq->length <= 5 ? 5 : 13;
1645 wep.Length = wep.KeyLength + offsetof(struct ndis_802_11_wep, KeyMaterial);
1649 if (keyindex_provided == 1) {
1650 /* set key_id only, no given KeyMaterial(erq->length == 0). */
1651 padapter->securitypriv.dot11PrivacyKeyIndex = key;
1653 DBG_88E("(keyindex_provided == 1), keyid =%d, key_len =%d\n", key, padapter->securitypriv.dot11DefKeylen[key]);
1655 switch (padapter->securitypriv.dot11DefKeylen[key]) {
1657 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP40_;
1660 padapter->securitypriv.dot11PrivacyAlgrthm = _WEP104_;
1663 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1671 wep.KeyIndex |= 0x80000000;
1673 memcpy(wep.KeyMaterial, keybuf, wep.KeyLength);
1675 if (!rtw_set_802_11_add_wep(padapter, &wep)) {
1676 if (rf_on == pwrpriv->rf_pwrstate)
1685 static int rtw_wx_get_enc(struct net_device *dev,
1686 struct iw_request_info *info,
1687 union iwreq_data *wrqu, char *keybuf)
1690 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1691 struct iw_point *erq = &(wrqu->encoding);
1692 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
1694 if (!check_fwstate(pmlmepriv, _FW_LINKED)) {
1695 if (!check_fwstate(pmlmepriv, WIFI_ADHOC_MASTER_STATE)) {
1697 erq->flags |= IW_ENCODE_DISABLED;
1702 key = erq->flags & IW_ENCODE_INDEX;
1709 key = padapter->securitypriv.dot11PrivacyKeyIndex;
1712 erq->flags = key + 1;
1714 switch (padapter->securitypriv.ndisencryptstatus) {
1715 case Ndis802_11EncryptionNotSupported:
1716 case Ndis802_11EncryptionDisabled:
1718 erq->flags |= IW_ENCODE_DISABLED;
1720 case Ndis802_11Encryption1Enabled:
1721 erq->length = padapter->securitypriv.dot11DefKeylen[key];
1723 memcpy(keybuf, padapter->securitypriv.dot11DefKey[key].skey, padapter->securitypriv.dot11DefKeylen[key]);
1725 erq->flags |= IW_ENCODE_ENABLED;
1727 if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeOpen)
1728 erq->flags |= IW_ENCODE_OPEN;
1729 else if (padapter->securitypriv.ndisauthtype == Ndis802_11AuthModeShared)
1730 erq->flags |= IW_ENCODE_RESTRICTED;
1733 erq->flags |= IW_ENCODE_DISABLED;
1736 case Ndis802_11Encryption2Enabled:
1737 case Ndis802_11Encryption3Enabled:
1739 erq->flags |= (IW_ENCODE_ENABLED | IW_ENCODE_OPEN | IW_ENCODE_NOKEY);
1743 erq->flags |= IW_ENCODE_DISABLED;
1750 static int rtw_wx_get_power(struct net_device *dev,
1751 struct iw_request_info *info,
1752 union iwreq_data *wrqu, char *extra)
1754 wrqu->power.value = 0;
1755 wrqu->power.fixed = 0; /* no auto select */
1756 wrqu->power.disabled = 1;
1761 static int rtw_wx_set_gen_ie(struct net_device *dev,
1762 struct iw_request_info *info,
1763 union iwreq_data *wrqu, char *extra)
1765 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1767 return rtw_set_wpa_ie(padapter, extra, wrqu->data.length);
1770 static int rtw_wx_set_auth(struct net_device *dev,
1771 struct iw_request_info *info,
1772 union iwreq_data *wrqu, char *extra)
1774 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1775 struct iw_param *param = (struct iw_param *)&(wrqu->param);
1778 switch (param->flags & IW_AUTH_INDEX) {
1779 case IW_AUTH_WPA_VERSION:
1781 case IW_AUTH_CIPHER_PAIRWISE:
1784 case IW_AUTH_CIPHER_GROUP:
1787 case IW_AUTH_KEY_MGMT:
1789 * ??? does not use these parameters
1792 case IW_AUTH_TKIP_COUNTERMEASURES:
1794 /* wpa_supplicant is enabling the tkip countermeasure. */
1795 padapter->securitypriv.btkip_countermeasure = true;
1797 /* wpa_supplicant is disabling the tkip countermeasure. */
1798 padapter->securitypriv.btkip_countermeasure = false;
1801 case IW_AUTH_DROP_UNENCRYPTED:
1804 * wpa_supplicant calls set_wpa_enabled when the driver
1805 * is loaded and unloaded, regardless of if WPA is being
1806 * used. No other calls are made which can be used to
1807 * determine if encryption will be used or not prior to
1808 * association being expected. If encryption is not being
1809 * used, drop_unencrypted is set to false, else true -- we
1810 * can use this to determine if the CAP_PRIVACY_ON bit should
1814 if (padapter->securitypriv.ndisencryptstatus == Ndis802_11Encryption1Enabled)
1815 break;/* it means init value, or using wep, ndisencryptstatus = Ndis802_11Encryption1Enabled, */
1816 /* then it needn't reset it; */
1819 padapter->securitypriv.ndisencryptstatus = Ndis802_11EncryptionDisabled;
1820 padapter->securitypriv.dot11PrivacyAlgrthm = _NO_PRIVACY_;
1821 padapter->securitypriv.dot118021XGrpPrivacy = _NO_PRIVACY_;
1822 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_Open;
1823 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeOpen;
1827 case IW_AUTH_80211_AUTH_ALG:
1829 * It's the starting point of a link layer connection using wpa_supplicant
1831 if (check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
1832 LeaveAllPowerSaveMode(padapter);
1833 rtw_disassoc_cmd(padapter, 500, false);
1834 DBG_88E("%s...call rtw_indicate_disconnect\n ", __func__);
1835 rtw_indicate_disconnect(padapter);
1836 rtw_free_assoc_resources(padapter);
1838 ret = wpa_set_auth_algs(dev, (u32)param->value);
1840 case IW_AUTH_WPA_ENABLED:
1842 case IW_AUTH_RX_UNENCRYPTED_EAPOL:
1844 case IW_AUTH_PRIVACY_INVOKED:
1853 static int rtw_wx_set_enc_ext(struct net_device *dev,
1854 struct iw_request_info *info,
1855 union iwreq_data *wrqu, char *extra)
1859 struct ieee_param *param = NULL;
1860 struct iw_point *pencoding = &wrqu->encoding;
1861 struct iw_encode_ext *pext = (struct iw_encode_ext *)extra;
1864 param_len = sizeof(struct ieee_param) + pext->key_len;
1865 param = (struct ieee_param *)rtw_malloc(param_len);
1869 memset(param, 0, param_len);
1871 param->cmd = IEEE_CMD_SET_ENCRYPTION;
1872 eth_broadcast_addr(param->sta_addr);
1874 switch (pext->alg) {
1875 case IW_ENCODE_ALG_NONE:
1876 /* todo: remove key */
1880 case IW_ENCODE_ALG_WEP:
1883 case IW_ENCODE_ALG_TKIP:
1886 case IW_ENCODE_ALG_CCMP:
1894 strlcpy((char *)param->u.crypt.alg, alg_name, IEEE_CRYPT_ALG_NAME_LEN);
1896 if (pext->ext_flags & IW_ENCODE_EXT_SET_TX_KEY)
1897 param->u.crypt.set_tx = 1;
1899 /* cliW: WEP does not have group key
1900 * just not checking GROUP key setting
1902 if ((pext->alg != IW_ENCODE_ALG_WEP) &&
1903 (pext->ext_flags & IW_ENCODE_EXT_GROUP_KEY))
1904 param->u.crypt.set_tx = 0;
1906 param->u.crypt.idx = (pencoding->flags&0x00FF) - 1;
1908 if (pext->ext_flags & IW_ENCODE_EXT_RX_SEQ_VALID)
1909 memcpy(param->u.crypt.seq, pext->rx_seq, 8);
1911 if (pext->key_len) {
1912 param->u.crypt.key_len = pext->key_len;
1913 memcpy(param->u.crypt.key, pext + 1, pext->key_len);
1916 ret = wpa_set_encryption(dev, param, param_len);
1923 static int rtw_wx_get_nick(struct net_device *dev,
1924 struct iw_request_info *info,
1925 union iwreq_data *wrqu, char *extra)
1928 wrqu->data.length = 14;
1929 wrqu->data.flags = 1;
1930 memcpy(extra, "<WIFI@REALTEK>", 14);
1933 /* dump debug info here */
1937 static int dummy(struct net_device *dev, struct iw_request_info *a,
1938 union iwreq_data *wrqu, char *b)
1943 static int wpa_set_param(struct net_device *dev, u8 name, u32 value)
1946 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
1949 case IEEE_PARAM_WPA_ENABLED:
1950 padapter->securitypriv.dot11AuthAlgrthm = dot11AuthAlgrthm_8021X; /* 802.1x */
1951 switch ((value)&0xff) {
1953 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPAPSK; /* WPA_PSK */
1954 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption2Enabled;
1957 padapter->securitypriv.ndisauthtype = Ndis802_11AuthModeWPA2PSK; /* WPA2_PSK */
1958 padapter->securitypriv.ndisencryptstatus = Ndis802_11Encryption3Enabled;
1961 RT_TRACE(_module_rtl871x_ioctl_os_c, _drv_info_,
1962 ("wpa_set_param:padapter->securitypriv.ndisauthtype =%d\n", padapter->securitypriv.ndisauthtype));
1964 case IEEE_PARAM_TKIP_COUNTERMEASURES:
1966 case IEEE_PARAM_DROP_UNENCRYPTED: {
1969 * wpa_supplicant calls set_wpa_enabled when the driver
1970 * is loaded and unloaded, regardless of if WPA is being
1971 * used. No other calls are made which can be used to
1972 * determine if encryption will be used or not prior to
1973 * association being expected. If encryption is not being
1974 * used, drop_unencrypted is set to false, else true -- we
1975 * can use this to determine if the CAP_PRIVACY_ON bit should
1981 case IEEE_PARAM_PRIVACY_INVOKED:
1984 case IEEE_PARAM_AUTH_ALGS:
1985 ret = wpa_set_auth_algs(dev, value);
1987 case IEEE_PARAM_IEEE_802_1X:
1989 case IEEE_PARAM_WPAX_SELECT:
1998 static int wpa_mlme(struct net_device *dev, u32 command, u32 reason)
2001 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2004 case IEEE_MLME_STA_DEAUTH:
2005 if (!rtw_set_802_11_disassociate(padapter))
2008 case IEEE_MLME_STA_DISASSOC:
2009 if (!rtw_set_802_11_disassociate(padapter))
2020 static int wpa_supplicant_ioctl(struct net_device *dev, struct iw_point *p)
2022 struct ieee_param *param;
2025 if (p->length < sizeof(struct ieee_param) || !p->pointer) {
2030 param = (struct ieee_param *)rtw_malloc(p->length);
2036 if (copy_from_user(param, p->pointer, p->length)) {
2042 switch (param->cmd) {
2043 case IEEE_CMD_SET_WPA_PARAM:
2044 ret = wpa_set_param(dev, param->u.wpa_param.name, param->u.wpa_param.value);
2047 case IEEE_CMD_SET_WPA_IE:
2048 ret = rtw_set_wpa_ie((struct adapter *)rtw_netdev_priv(dev),
2049 (char *)param->u.wpa_ie.data, (u16)param->u.wpa_ie.len);
2052 case IEEE_CMD_SET_ENCRYPTION:
2053 ret = wpa_set_encryption(dev, param, p->length);
2057 ret = wpa_mlme(dev, param->u.mlme.command, param->u.mlme.reason_code);
2061 DBG_88E("Unknown WPA supplicant request: %d\n", param->cmd);
2066 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2076 #ifdef CONFIG_88EU_AP_MODE
2077 static u8 set_pairwise_key(struct adapter *padapter, struct sta_info *psta)
2079 struct cmd_obj *ph2c;
2080 struct set_stakey_parm *psetstakey_para;
2081 struct cmd_priv *pcmdpriv = &padapter->cmdpriv;
2084 ph2c = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
2090 psetstakey_para = kzalloc(sizeof(struct set_stakey_parm), GFP_KERNEL);
2091 if (!psetstakey_para) {
2097 init_h2fwcmd_w_parm_no_rsp(ph2c, psetstakey_para, _SetStaKey_CMD_);
2099 psetstakey_para->algorithm = (u8)psta->dot118021XPrivacy;
2101 memcpy(psetstakey_para->addr, psta->hwaddr, ETH_ALEN);
2103 memcpy(psetstakey_para->key, &psta->dot118021x_UncstKey, 16);
2105 res = rtw_enqueue_cmd(pcmdpriv, ph2c);
2112 static int set_group_key(struct adapter *padapter, u8 *key, u8 alg, int keyid)
2115 struct cmd_obj *pcmd;
2116 struct setkey_parm *psetkeyparm;
2117 struct cmd_priv *pcmdpriv = &(padapter->cmdpriv);
2120 DBG_88E("%s\n", __func__);
2122 pcmd = kzalloc(sizeof(struct cmd_obj), GFP_KERNEL);
2127 psetkeyparm = kzalloc(sizeof(struct setkey_parm), GFP_KERNEL);
2134 psetkeyparm->keyid = (u8)keyid;
2136 psetkeyparm->algorithm = alg;
2138 psetkeyparm->set_tx = 1;
2154 memcpy(&(psetkeyparm->key[0]), key, keylen);
2156 pcmd->cmdcode = _SetKey_CMD_;
2157 pcmd->parmbuf = (u8 *)psetkeyparm;
2158 pcmd->cmdsz = (sizeof(struct setkey_parm));
2162 INIT_LIST_HEAD(&pcmd->list);
2164 res = rtw_enqueue_cmd(pcmdpriv, pcmd);
2171 static int set_wep_key(struct adapter *padapter, u8 *key, u8 keylen, int keyid)
2186 return set_group_key(padapter, key, alg, keyid);
2189 static int rtw_set_encryption(struct net_device *dev, struct ieee_param *param, u32 param_len)
2192 u32 wep_key_idx, wep_key_len, wep_total_len;
2193 struct ndis_802_11_wep *pwep = NULL;
2194 struct sta_info *psta = NULL, *pbcmc_sta = NULL;
2195 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2196 struct mlme_priv *pmlmepriv = &padapter->mlmepriv;
2197 struct security_priv *psecuritypriv = &(padapter->securitypriv);
2198 struct sta_priv *pstapriv = &padapter->stapriv;
2200 DBG_88E("%s\n", __func__);
2201 param->u.crypt.err = 0;
2202 param->u.crypt.alg[IEEE_CRYPT_ALG_NAME_LEN - 1] = '\0';
2203 if (param_len != sizeof(struct ieee_param) + param->u.crypt.key_len) {
2207 if (is_broadcast_ether_addr(param->sta_addr)) {
2208 if (param->u.crypt.idx >= WEP_KEYS) {
2213 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2215 DBG_88E("rtw_set_encryption(), sta has already been removed or never been added\n");
2220 if (strcmp(param->u.crypt.alg, "none") == 0 && (!psta)) {
2221 /* todo:clear default encryption keys */
2223 DBG_88E("clear default encryption keys, keyid =%d\n", param->u.crypt.idx);
2226 if (strcmp(param->u.crypt.alg, "WEP") == 0 && (!psta)) {
2227 DBG_88E("r871x_set_encryption, crypt.alg = WEP\n");
2228 wep_key_idx = param->u.crypt.idx;
2229 wep_key_len = param->u.crypt.key_len;
2230 DBG_88E("r871x_set_encryption, wep_key_idx=%d, len=%d\n", wep_key_idx, wep_key_len);
2231 if ((wep_key_idx >= WEP_KEYS) || (wep_key_len <= 0)) {
2236 if (wep_key_len > 0) {
2237 wep_key_len = wep_key_len <= 5 ? 5 : 13;
2238 wep_total_len = wep_key_len + offsetof(struct ndis_802_11_wep, KeyMaterial);
2239 pwep = (struct ndis_802_11_wep *)rtw_malloc(wep_total_len);
2241 DBG_88E(" r871x_set_encryption: pwep allocate fail !!!\n");
2245 memset(pwep, 0, wep_total_len);
2247 pwep->KeyLength = wep_key_len;
2248 pwep->Length = wep_total_len;
2251 pwep->KeyIndex = wep_key_idx;
2253 memcpy(pwep->KeyMaterial, param->u.crypt.key, pwep->KeyLength);
2255 if (param->u.crypt.set_tx) {
2256 DBG_88E("wep, set_tx = 1\n");
2258 psecuritypriv->ndisencryptstatus = Ndis802_11Encryption1Enabled;
2259 psecuritypriv->dot11PrivacyAlgrthm = _WEP40_;
2260 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2262 if (pwep->KeyLength == 13) {
2263 psecuritypriv->dot11PrivacyAlgrthm = _WEP104_;
2264 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2267 psecuritypriv->dot11PrivacyKeyIndex = wep_key_idx;
2269 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2271 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2273 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2275 DBG_88E("wep, set_tx = 0\n");
2277 /* don't update "psecuritypriv->dot11PrivacyAlgrthm" and */
2278 /* psecuritypriv->dot11PrivacyKeyIndex = keyid", but can rtw_set_key to cam */
2280 memcpy(&(psecuritypriv->dot11DefKey[wep_key_idx].skey[0]), pwep->KeyMaterial, pwep->KeyLength);
2282 psecuritypriv->dot11DefKeylen[wep_key_idx] = pwep->KeyLength;
2284 set_wep_key(padapter, pwep->KeyMaterial, pwep->KeyLength, wep_key_idx);
2290 if (!psta && check_fwstate(pmlmepriv, WIFI_AP_STATE)) { /* group key */
2291 if (param->u.crypt.set_tx == 1) {
2292 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2293 DBG_88E("%s, set group_key, WEP\n", __func__);
2295 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2296 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2298 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2299 if (param->u.crypt.key_len == 13)
2300 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2301 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2302 DBG_88E("%s, set group_key, TKIP\n", __func__);
2303 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2304 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2305 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2307 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2308 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2310 psecuritypriv->busetkipkey = true;
2311 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2312 DBG_88E("%s, set group_key, CCMP\n", __func__);
2313 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2314 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2315 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2317 DBG_88E("%s, set group_key, none\n", __func__);
2318 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2320 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2321 psecuritypriv->binstallGrpkey = true;
2322 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2323 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2324 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2326 pbcmc_sta->ieee8021x_blocked = false;
2327 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2333 if (psecuritypriv->dot11AuthAlgrthm == dot11AuthAlgrthm_8021X && psta) { /* psk/802_1x */
2334 if (check_fwstate(pmlmepriv, WIFI_AP_STATE)) {
2335 if (param->u.crypt.set_tx == 1) {
2336 memcpy(psta->dot118021x_UncstKey.skey, param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2338 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2339 DBG_88E("%s, set pairwise key, WEP\n", __func__);
2341 psta->dot118021XPrivacy = _WEP40_;
2342 if (param->u.crypt.key_len == 13)
2343 psta->dot118021XPrivacy = _WEP104_;
2344 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2345 DBG_88E("%s, set pairwise key, TKIP\n", __func__);
2347 psta->dot118021XPrivacy = _TKIP_;
2350 memcpy(psta->dot11tkiptxmickey.skey, &(param->u.crypt.key[16]), 8);
2351 memcpy(psta->dot11tkiprxmickey.skey, &(param->u.crypt.key[24]), 8);
2353 psecuritypriv->busetkipkey = true;
2354 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2355 DBG_88E("%s, set pairwise key, CCMP\n", __func__);
2357 psta->dot118021XPrivacy = _AES_;
2359 DBG_88E("%s, set pairwise key, none\n", __func__);
2361 psta->dot118021XPrivacy = _NO_PRIVACY_;
2364 set_pairwise_key(padapter, psta);
2366 psta->ieee8021x_blocked = false;
2367 } else { /* group key??? */
2368 if (strcmp(param->u.crypt.alg, "WEP") == 0) {
2369 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2370 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2371 psecuritypriv->dot118021XGrpPrivacy = _WEP40_;
2372 if (param->u.crypt.key_len == 13)
2373 psecuritypriv->dot118021XGrpPrivacy = _WEP104_;
2374 } else if (strcmp(param->u.crypt.alg, "TKIP") == 0) {
2375 psecuritypriv->dot118021XGrpPrivacy = _TKIP_;
2377 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2378 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2381 memcpy(psecuritypriv->dot118021XGrptxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[16]), 8);
2382 memcpy(psecuritypriv->dot118021XGrprxmickey[param->u.crypt.idx].skey, &(param->u.crypt.key[24]), 8);
2384 psecuritypriv->busetkipkey = true;
2385 } else if (strcmp(param->u.crypt.alg, "CCMP") == 0) {
2386 psecuritypriv->dot118021XGrpPrivacy = _AES_;
2388 memcpy(psecuritypriv->dot118021XGrpKey[param->u.crypt.idx].skey,
2389 param->u.crypt.key, min_t(u16, param->u.crypt.key_len, 16));
2391 psecuritypriv->dot118021XGrpPrivacy = _NO_PRIVACY_;
2394 psecuritypriv->dot118021XGrpKeyid = param->u.crypt.idx;
2396 psecuritypriv->binstallGrpkey = true;
2398 psecuritypriv->dot11PrivacyAlgrthm = psecuritypriv->dot118021XGrpPrivacy;/* */
2400 set_group_key(padapter, param->u.crypt.key, psecuritypriv->dot118021XGrpPrivacy, param->u.crypt.idx);
2402 pbcmc_sta = rtw_get_bcmc_stainfo(padapter);
2404 pbcmc_sta->ieee8021x_blocked = false;
2405 pbcmc_sta->dot118021XPrivacy = psecuritypriv->dot118021XGrpPrivacy;/* rx will use bmc_sta's dot118021XPrivacy */
2418 static int rtw_set_beacon(struct net_device *dev, struct ieee_param *param, int len)
2421 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2422 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2423 struct sta_priv *pstapriv = &padapter->stapriv;
2424 unsigned char *pbuf = param->u.bcn_ie.buf;
2426 DBG_88E("%s, len =%d\n", __func__, len);
2428 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2431 memcpy(&pstapriv->max_num_sta, param->u.bcn_ie.reserved, 2);
2433 if ((pstapriv->max_num_sta > NUM_STA) || (pstapriv->max_num_sta <= 0))
2434 pstapriv->max_num_sta = NUM_STA;
2436 if (rtw_check_beacon_data(padapter, pbuf, (len-12-2)) == _SUCCESS)/* 12 = param header, 2:no packed */
2444 static int rtw_hostapd_sta_flush(struct net_device *dev)
2446 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2448 DBG_88E("%s\n", __func__);
2450 flush_all_cam_entry(padapter); /* clear CAM */
2452 return rtw_sta_flush(padapter);
2455 static int rtw_add_sta(struct net_device *dev, struct ieee_param *param)
2458 struct sta_info *psta = NULL;
2459 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2460 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2461 struct sta_priv *pstapriv = &padapter->stapriv;
2463 DBG_88E("rtw_add_sta(aid =%d) =%pM\n", param->u.add_sta.aid, (param->sta_addr));
2465 if (!check_fwstate(pmlmepriv, (_FW_LINKED|WIFI_AP_STATE)))
2468 if (is_broadcast_ether_addr(param->sta_addr))
2471 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2473 int flags = param->u.add_sta.flags;
2475 psta->aid = param->u.add_sta.aid;/* aid = 1~2007 */
2477 memcpy(psta->bssrateset, param->u.add_sta.tx_supp_rates, 16);
2479 /* check wmm cap. */
2480 if (WLAN_STA_WME&flags)
2481 psta->qos_option = 1;
2483 psta->qos_option = 0;
2485 if (pmlmepriv->qospriv.qos_option == 0)
2486 psta->qos_option = 0;
2488 /* chec 802.11n ht cap. */
2489 if (WLAN_STA_HT&flags) {
2490 psta->htpriv.ht_option = true;
2491 psta->qos_option = 1;
2492 memcpy(&psta->htpriv.ht_cap, ¶m->u.add_sta.ht_cap,
2493 sizeof(struct ieee80211_ht_cap));
2495 psta->htpriv.ht_option = false;
2498 if (!pmlmepriv->htpriv.ht_option)
2499 psta->htpriv.ht_option = false;
2501 update_sta_info_apmode(padapter, psta);
2509 static int rtw_del_sta(struct net_device *dev, struct ieee_param *param)
2511 struct sta_info *psta = NULL;
2512 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2513 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2514 struct sta_priv *pstapriv = &padapter->stapriv;
2517 DBG_88E("rtw_del_sta =%pM\n", (param->sta_addr));
2519 if (!check_fwstate(pmlmepriv, _FW_LINKED | WIFI_AP_STATE))
2522 if (is_broadcast_ether_addr(param->sta_addr))
2525 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2527 spin_lock_bh(&pstapriv->asoc_list_lock);
2528 if (!list_empty(&psta->asoc_list)) {
2529 list_del_init(&psta->asoc_list);
2530 pstapriv->asoc_list_cnt--;
2531 updated = ap_free_sta(padapter, psta, true, WLAN_REASON_DEAUTH_LEAVING);
2533 spin_unlock_bh(&pstapriv->asoc_list_lock);
2534 associated_clients_update(padapter, updated);
2537 DBG_88E("rtw_del_sta(), sta has already been removed or never been added\n");
2543 static int rtw_ioctl_get_sta_data(struct net_device *dev, struct ieee_param *param, int len)
2546 struct sta_info *psta = NULL;
2547 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2548 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2549 struct sta_priv *pstapriv = &padapter->stapriv;
2550 struct ieee_param_ex *param_ex = (struct ieee_param_ex *)param;
2551 struct sta_data *psta_data = (struct sta_data *)param_ex->data;
2553 DBG_88E("rtw_ioctl_get_sta_info, sta_addr: %pM\n", (param_ex->sta_addr));
2555 if (!check_fwstate(pmlmepriv, _FW_LINKED | WIFI_AP_STATE))
2558 if (is_broadcast_ether_addr(param_ex->sta_addr))
2561 psta = rtw_get_stainfo(pstapriv, param_ex->sta_addr);
2563 psta_data->aid = (u16)psta->aid;
2564 psta_data->capability = psta->capability;
2565 psta_data->flags = psta->flags;
2569 no_short_slot_time_set : BIT(1)
2570 no_short_preamble_set : BIT(2)
2571 no_ht_gf_set : BIT(3)
2573 ht_20mhz_set : BIT(5)
2576 psta_data->sta_set = ((psta->nonerp_set) |
2577 (psta->no_short_slot_time_set << 1) |
2578 (psta->no_short_preamble_set << 2) |
2579 (psta->no_ht_gf_set << 3) |
2580 (psta->no_ht_set << 4) |
2581 (psta->ht_20mhz_set << 5));
2582 psta_data->tx_supp_rates_len = psta->bssratelen;
2583 memcpy(psta_data->tx_supp_rates, psta->bssrateset, psta->bssratelen);
2584 memcpy(&psta_data->ht_cap,
2585 &psta->htpriv.ht_cap, sizeof(struct ieee80211_ht_cap));
2586 psta_data->rx_pkts = psta->sta_stats.rx_data_pkts;
2587 psta_data->rx_bytes = psta->sta_stats.rx_bytes;
2588 psta_data->rx_drops = psta->sta_stats.rx_drops;
2589 psta_data->tx_pkts = psta->sta_stats.tx_pkts;
2590 psta_data->tx_bytes = psta->sta_stats.tx_bytes;
2591 psta_data->tx_drops = psta->sta_stats.tx_drops;
2599 static int rtw_get_sta_wpaie(struct net_device *dev, struct ieee_param *param)
2602 struct sta_info *psta = NULL;
2603 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2604 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2605 struct sta_priv *pstapriv = &padapter->stapriv;
2607 DBG_88E("rtw_get_sta_wpaie, sta_addr: %pM\n", (param->sta_addr));
2609 if (!check_fwstate(pmlmepriv, _FW_LINKED | WIFI_AP_STATE))
2612 if (is_broadcast_ether_addr(param->sta_addr))
2615 psta = rtw_get_stainfo(pstapriv, param->sta_addr);
2617 if (psta->wpa_ie[0] == WLAN_EID_RSN ||
2618 psta->wpa_ie[0] == WLAN_EID_VENDOR_SPECIFIC) {
2622 wpa_ie_len = psta->wpa_ie[1];
2623 copy_len = min_t(int, wpa_ie_len + 2, sizeof(psta->wpa_ie));
2624 param->u.wpa_ie.len = copy_len;
2625 memcpy(param->u.wpa_ie.reserved, psta->wpa_ie, copy_len);
2627 DBG_88E("sta's wpa_ie is NONE\n");
2636 static int rtw_set_wps_beacon(struct net_device *dev, struct ieee_param *param, int len)
2638 unsigned char wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2639 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2640 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2641 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2644 DBG_88E("%s, len =%d\n", __func__, len);
2646 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2649 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2651 kfree(pmlmepriv->wps_beacon_ie);
2652 pmlmepriv->wps_beacon_ie = NULL;
2655 pmlmepriv->wps_beacon_ie = rtw_malloc(ie_len);
2656 pmlmepriv->wps_beacon_ie_len = ie_len;
2657 if (!pmlmepriv->wps_beacon_ie) {
2658 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2662 memcpy(pmlmepriv->wps_beacon_ie, param->u.bcn_ie.buf, ie_len);
2664 update_beacon(padapter, _VENDOR_SPECIFIC_IE_, wps_oui, true);
2666 pmlmeext->bstart_bss = true;
2672 static int rtw_set_wps_probe_resp(struct net_device *dev, struct ieee_param *param, int len)
2674 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2675 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2678 DBG_88E("%s, len =%d\n", __func__, len);
2680 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2683 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2685 kfree(pmlmepriv->wps_probe_resp_ie);
2686 pmlmepriv->wps_probe_resp_ie = NULL;
2689 pmlmepriv->wps_probe_resp_ie = rtw_malloc(ie_len);
2690 pmlmepriv->wps_probe_resp_ie_len = ie_len;
2691 if (!pmlmepriv->wps_probe_resp_ie) {
2692 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2695 memcpy(pmlmepriv->wps_probe_resp_ie, param->u.bcn_ie.buf, ie_len);
2701 static int rtw_set_wps_assoc_resp(struct net_device *dev, struct ieee_param *param, int len)
2703 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2704 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2707 DBG_88E("%s, len =%d\n", __func__, len);
2709 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2712 ie_len = len-12-2;/* 12 = param header, 2:no packed */
2714 kfree(pmlmepriv->wps_assoc_resp_ie);
2715 pmlmepriv->wps_assoc_resp_ie = NULL;
2718 pmlmepriv->wps_assoc_resp_ie = rtw_malloc(ie_len);
2719 pmlmepriv->wps_assoc_resp_ie_len = ie_len;
2720 if (!pmlmepriv->wps_assoc_resp_ie) {
2721 DBG_88E("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2725 memcpy(pmlmepriv->wps_assoc_resp_ie, param->u.bcn_ie.buf, ie_len);
2731 static int rtw_set_hidden_ssid(struct net_device *dev, struct ieee_param *param, int len)
2733 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2734 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2735 struct mlme_ext_priv *pmlmeext = &(padapter->mlmeextpriv);
2736 struct mlme_ext_info *pmlmeinfo = &(pmlmeext->mlmext_info);
2740 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2743 if (param->u.wpa_param.name != 0) /* dummy test... */
2744 DBG_88E("%s name(%u) != 0\n", __func__, param->u.wpa_param.name);
2745 value = param->u.wpa_param.value;
2747 /* use the same definition of hostapd's ignore_broadcast_ssid */
2748 if (value != 1 && value != 2)
2750 DBG_88E("%s value(%u)\n", __func__, value);
2751 pmlmeinfo->hidden_ssid_mode = value;
2755 static int rtw_ioctl_acl_remove_sta(struct net_device *dev, struct ieee_param *param, int len)
2757 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2758 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2760 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2763 if (is_broadcast_ether_addr(param->sta_addr))
2766 return rtw_acl_remove_sta(padapter, param->sta_addr);
2769 static int rtw_ioctl_acl_add_sta(struct net_device *dev, struct ieee_param *param, int len)
2771 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2772 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2774 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2777 if (is_broadcast_ether_addr(param->sta_addr))
2780 return rtw_acl_add_sta(padapter, param->sta_addr);
2783 static int rtw_ioctl_set_macaddr_acl(struct net_device *dev, struct ieee_param *param, int len)
2785 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2786 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2788 if (!check_fwstate(pmlmepriv, WIFI_AP_STATE))
2791 rtw_set_macaddr_acl(padapter, param->u.mlme.command);
2796 static int rtw_hostapd_ioctl(struct net_device *dev, struct iw_point *p)
2798 struct ieee_param *param;
2800 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2803 * this function is expect to call in master mode, which allows no power saving
2804 * so, we just check hw_init_completed
2807 if (!padapter->hw_init_completed) {
2817 param = (struct ieee_param *)rtw_malloc(p->length);
2823 if (copy_from_user(param, p->pointer, p->length)) {
2829 switch (param->cmd) {
2830 case RTL871X_HOSTAPD_FLUSH:
2831 ret = rtw_hostapd_sta_flush(dev);
2833 case RTL871X_HOSTAPD_ADD_STA:
2834 ret = rtw_add_sta(dev, param);
2836 case RTL871X_HOSTAPD_REMOVE_STA:
2837 ret = rtw_del_sta(dev, param);
2839 case RTL871X_HOSTAPD_SET_BEACON:
2840 ret = rtw_set_beacon(dev, param, p->length);
2842 case RTL871X_SET_ENCRYPTION:
2843 ret = rtw_set_encryption(dev, param, p->length);
2845 case RTL871X_HOSTAPD_GET_WPAIE_STA:
2846 ret = rtw_get_sta_wpaie(dev, param);
2848 case RTL871X_HOSTAPD_SET_WPS_BEACON:
2849 ret = rtw_set_wps_beacon(dev, param, p->length);
2851 case RTL871X_HOSTAPD_SET_WPS_PROBE_RESP:
2852 ret = rtw_set_wps_probe_resp(dev, param, p->length);
2854 case RTL871X_HOSTAPD_SET_WPS_ASSOC_RESP:
2855 ret = rtw_set_wps_assoc_resp(dev, param, p->length);
2857 case RTL871X_HOSTAPD_SET_HIDDEN_SSID:
2858 ret = rtw_set_hidden_ssid(dev, param, p->length);
2860 case RTL871X_HOSTAPD_GET_INFO_STA:
2861 ret = rtw_ioctl_get_sta_data(dev, param, p->length);
2863 case RTL871X_HOSTAPD_SET_MACADDR_ACL:
2864 ret = rtw_ioctl_set_macaddr_acl(dev, param, p->length);
2866 case RTL871X_HOSTAPD_ACL_ADD_STA:
2867 ret = rtw_ioctl_acl_add_sta(dev, param, p->length);
2869 case RTL871X_HOSTAPD_ACL_REMOVE_STA:
2870 ret = rtw_ioctl_acl_remove_sta(dev, param, p->length);
2873 DBG_88E("Unknown hostapd request: %d\n", param->cmd);
2878 if (ret == 0 && copy_to_user(p->pointer, param, p->length))
2886 #include <rtw_android.h>
2887 static int rtw_wx_set_priv(struct net_device *dev,
2888 struct iw_request_info *info,
2889 union iwreq_data *awrq,
2895 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
2896 struct iw_point *dwrq = (struct iw_point *)awrq;
2898 if (dwrq->length == 0)
2906 if (copy_from_user(ext, dwrq->pointer, len)) {
2911 /* added for wps2.0 @20110524 */
2912 if (dwrq->flags == 0x8766 && len > 8) {
2914 struct mlme_priv *pmlmepriv = &(padapter->mlmepriv);
2915 u8 *probereq_wpsie = ext;
2916 int probereq_wpsie_len = len;
2917 u8 wps_oui[4] = {0x0, 0x50, 0xf2, 0x04};
2919 if ((_VENDOR_SPECIFIC_IE_ == probereq_wpsie[0]) &&
2920 (!memcmp(&probereq_wpsie[2], wps_oui, 4))) {
2921 cp_sz = min(probereq_wpsie_len, MAX_WPS_IE_LEN);
2923 pmlmepriv->wps_probe_req_ie_len = 0;
2924 kfree(pmlmepriv->wps_probe_req_ie);
2925 pmlmepriv->wps_probe_req_ie = NULL;
2927 pmlmepriv->wps_probe_req_ie = rtw_malloc(cp_sz);
2928 if (!pmlmepriv->wps_probe_req_ie) {
2929 pr_info("%s()-%d: rtw_malloc() ERROR!\n", __func__, __LINE__);
2933 memcpy(pmlmepriv->wps_probe_req_ie, probereq_wpsie, cp_sz);
2934 pmlmepriv->wps_probe_req_ie_len = cp_sz;
2939 if (len >= WEXT_CSCAN_HEADER_SIZE &&
2940 !memcmp(ext, WEXT_CSCAN_HEADER, WEXT_CSCAN_HEADER_SIZE)) {
2941 ret = rtw_wx_set_scan(dev, info, awrq, ext);
2952 static iw_handler rtw_handlers[] = {
2953 NULL, /* SIOCSIWCOMMIT */
2954 rtw_wx_get_name, /* SIOCGIWNAME */
2955 dummy, /* SIOCSIWNWID */
2956 dummy, /* SIOCGIWNWID */
2957 rtw_wx_set_freq, /* SIOCSIWFREQ */
2958 rtw_wx_get_freq, /* SIOCGIWFREQ */
2959 rtw_wx_set_mode, /* SIOCSIWMODE */
2960 rtw_wx_get_mode, /* SIOCGIWMODE */
2961 dummy, /* SIOCSIWSENS */
2962 rtw_wx_get_sens, /* SIOCGIWSENS */
2963 NULL, /* SIOCSIWRANGE */
2964 rtw_wx_get_range, /* SIOCGIWRANGE */
2965 rtw_wx_set_priv, /* SIOCSIWPRIV */
2966 NULL, /* SIOCGIWPRIV */
2967 NULL, /* SIOCSIWSTATS */
2968 NULL, /* SIOCGIWSTATS */
2969 dummy, /* SIOCSIWSPY */
2970 dummy, /* SIOCGIWSPY */
2971 NULL, /* SIOCGIWTHRSPY */
2972 NULL, /* SIOCWIWTHRSPY */
2973 rtw_wx_set_wap, /* SIOCSIWAP */
2974 rtw_wx_get_wap, /* SIOCGIWAP */
2975 rtw_wx_set_mlme, /* request MLME operation; uses struct iw_mlme */
2976 dummy, /* SIOCGIWAPLIST -- depricated */
2977 rtw_wx_set_scan, /* SIOCSIWSCAN */
2978 rtw_wx_get_scan, /* SIOCGIWSCAN */
2979 rtw_wx_set_essid, /* SIOCSIWESSID */
2980 rtw_wx_get_essid, /* SIOCGIWESSID */
2981 dummy, /* SIOCSIWNICKN */
2982 rtw_wx_get_nick, /* SIOCGIWNICKN */
2983 NULL, /* -- hole -- */
2984 NULL, /* -- hole -- */
2985 rtw_wx_set_rate, /* SIOCSIWRATE */
2986 rtw_wx_get_rate, /* SIOCGIWRATE */
2987 rtw_wx_set_rts, /* SIOCSIWRTS */
2988 rtw_wx_get_rts, /* SIOCGIWRTS */
2989 rtw_wx_set_frag, /* SIOCSIWFRAG */
2990 rtw_wx_get_frag, /* SIOCGIWFRAG */
2991 dummy, /* SIOCSIWTXPOW */
2992 dummy, /* SIOCGIWTXPOW */
2993 dummy, /* SIOCSIWRETRY */
2994 rtw_wx_get_retry, /* SIOCGIWRETRY */
2995 rtw_wx_set_enc, /* SIOCSIWENCODE */
2996 rtw_wx_get_enc, /* SIOCGIWENCODE */
2997 dummy, /* SIOCSIWPOWER */
2998 rtw_wx_get_power, /* SIOCGIWPOWER */
2999 NULL, /*---hole---*/
3000 NULL, /*---hole---*/
3001 rtw_wx_set_gen_ie, /* SIOCSIWGENIE */
3002 NULL, /* SIOCGWGENIE */
3003 rtw_wx_set_auth, /* SIOCSIWAUTH */
3004 NULL, /* SIOCGIWAUTH */
3005 rtw_wx_set_enc_ext, /* SIOCSIWENCODEEXT */
3006 NULL, /* SIOCGIWENCODEEXT */
3007 rtw_wx_set_pmkid, /* SIOCSIWPMKSA */
3008 NULL, /*---hole---*/
3011 static struct iw_statistics *rtw_get_wireless_stats(struct net_device *dev)
3013 struct adapter *padapter = (struct adapter *)rtw_netdev_priv(dev);
3014 struct iw_statistics *piwstats = &padapter->iwstats;
3019 if (!check_fwstate(&padapter->mlmepriv, _FW_LINKED)) {
3020 piwstats->qual.qual = 0;
3021 piwstats->qual.level = 0;
3022 piwstats->qual.noise = 0;
3024 tmp_level = padapter->recvpriv.signal_strength;
3025 tmp_qual = padapter->recvpriv.signal_qual;
3026 tmp_noise = padapter->recvpriv.noise;
3028 piwstats->qual.level = tmp_level;
3029 piwstats->qual.qual = tmp_qual;
3030 piwstats->qual.noise = tmp_noise;
3032 piwstats->qual.updated = IW_QUAL_ALL_UPDATED;/* IW_QUAL_DBM; */
3033 return &padapter->iwstats;
3036 struct iw_handler_def rtw_handlers_def = {
3037 .standard = rtw_handlers,
3038 .num_standard = ARRAY_SIZE(rtw_handlers),
3039 .get_wireless_stats = rtw_get_wireless_stats,
3042 int rtw_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
3044 struct iwreq *wrq = (struct iwreq *)rq;
3048 case RTL_IOCTL_WPA_SUPPLICANT:
3049 ret = wpa_supplicant_ioctl(dev, &wrq->u.data);
3051 #ifdef CONFIG_88EU_AP_MODE
3052 case RTL_IOCTL_HOSTAPD:
3053 ret = rtw_hostapd_ioctl(dev, &wrq->u.data);
3055 #endif /* CONFIG_88EU_AP_MODE */
3056 case (SIOCDEVPRIVATE+1):
3057 ret = rtw_android_priv_cmd(dev, rq, cmd);