blob: 45b0d974e4f71d3925204e95609ec9365b176880 [file] [log] [blame]
Kalle Valobdcd8172011-07-18 00:22:30 +03001/*
2 * Copyright (c) 2004-2011 Atheros Communications Inc.
3 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040017#include <linux/moduleparam.h>
Raja Manic08631c2011-12-16 14:24:24 +053018#include <linux/inetdevice.h>
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040019
Kalle Valobdcd8172011-07-18 00:22:30 +030020#include "core.h"
21#include "cfg80211.h"
22#include "debug.h"
Kalle Valoabcb3442011-07-22 08:26:20 +030023#include "hif-ops.h"
Kalle Valo003353b0d2011-09-01 10:14:21 +030024#include "testmode.h"
Kalle Valobdcd8172011-07-18 00:22:30 +030025
Jouni Malinen6bbc7c32011-09-05 17:38:47 +030026static unsigned int ath6kl_p2p;
27
28module_param(ath6kl_p2p, uint, 0644);
29
Kalle Valobdcd8172011-07-18 00:22:30 +030030#define RATETAB_ENT(_rate, _rateid, _flags) { \
31 .bitrate = (_rate), \
32 .flags = (_flags), \
33 .hw_value = (_rateid), \
34}
35
36#define CHAN2G(_channel, _freq, _flags) { \
37 .band = IEEE80211_BAND_2GHZ, \
38 .hw_value = (_channel), \
39 .center_freq = (_freq), \
40 .flags = (_flags), \
41 .max_antenna_gain = 0, \
42 .max_power = 30, \
43}
44
45#define CHAN5G(_channel, _flags) { \
46 .band = IEEE80211_BAND_5GHZ, \
47 .hw_value = (_channel), \
48 .center_freq = 5000 + (5 * (_channel)), \
49 .flags = (_flags), \
50 .max_antenna_gain = 0, \
51 .max_power = 30, \
52}
53
54static struct ieee80211_rate ath6kl_rates[] = {
55 RATETAB_ENT(10, 0x1, 0),
56 RATETAB_ENT(20, 0x2, 0),
57 RATETAB_ENT(55, 0x4, 0),
58 RATETAB_ENT(110, 0x8, 0),
59 RATETAB_ENT(60, 0x10, 0),
60 RATETAB_ENT(90, 0x20, 0),
61 RATETAB_ENT(120, 0x40, 0),
62 RATETAB_ENT(180, 0x80, 0),
63 RATETAB_ENT(240, 0x100, 0),
64 RATETAB_ENT(360, 0x200, 0),
65 RATETAB_ENT(480, 0x400, 0),
66 RATETAB_ENT(540, 0x800, 0),
67};
68
69#define ath6kl_a_rates (ath6kl_rates + 4)
70#define ath6kl_a_rates_size 8
71#define ath6kl_g_rates (ath6kl_rates + 0)
72#define ath6kl_g_rates_size 12
73
74static struct ieee80211_channel ath6kl_2ghz_channels[] = {
75 CHAN2G(1, 2412, 0),
76 CHAN2G(2, 2417, 0),
77 CHAN2G(3, 2422, 0),
78 CHAN2G(4, 2427, 0),
79 CHAN2G(5, 2432, 0),
80 CHAN2G(6, 2437, 0),
81 CHAN2G(7, 2442, 0),
82 CHAN2G(8, 2447, 0),
83 CHAN2G(9, 2452, 0),
84 CHAN2G(10, 2457, 0),
85 CHAN2G(11, 2462, 0),
86 CHAN2G(12, 2467, 0),
87 CHAN2G(13, 2472, 0),
88 CHAN2G(14, 2484, 0),
89};
90
91static struct ieee80211_channel ath6kl_5ghz_a_channels[] = {
92 CHAN5G(34, 0), CHAN5G(36, 0),
93 CHAN5G(38, 0), CHAN5G(40, 0),
94 CHAN5G(42, 0), CHAN5G(44, 0),
95 CHAN5G(46, 0), CHAN5G(48, 0),
96 CHAN5G(52, 0), CHAN5G(56, 0),
97 CHAN5G(60, 0), CHAN5G(64, 0),
98 CHAN5G(100, 0), CHAN5G(104, 0),
99 CHAN5G(108, 0), CHAN5G(112, 0),
100 CHAN5G(116, 0), CHAN5G(120, 0),
101 CHAN5G(124, 0), CHAN5G(128, 0),
102 CHAN5G(132, 0), CHAN5G(136, 0),
103 CHAN5G(140, 0), CHAN5G(149, 0),
104 CHAN5G(153, 0), CHAN5G(157, 0),
105 CHAN5G(161, 0), CHAN5G(165, 0),
106 CHAN5G(184, 0), CHAN5G(188, 0),
107 CHAN5G(192, 0), CHAN5G(196, 0),
108 CHAN5G(200, 0), CHAN5G(204, 0),
109 CHAN5G(208, 0), CHAN5G(212, 0),
110 CHAN5G(216, 0),
111};
112
113static struct ieee80211_supported_band ath6kl_band_2ghz = {
114 .n_channels = ARRAY_SIZE(ath6kl_2ghz_channels),
115 .channels = ath6kl_2ghz_channels,
116 .n_bitrates = ath6kl_g_rates_size,
117 .bitrates = ath6kl_g_rates,
118};
119
120static struct ieee80211_supported_band ath6kl_band_5ghz = {
121 .n_channels = ARRAY_SIZE(ath6kl_5ghz_a_channels),
122 .channels = ath6kl_5ghz_a_channels,
123 .n_bitrates = ath6kl_a_rates_size,
124 .bitrates = ath6kl_a_rates,
125};
126
Jouni Malinen837cb972011-10-11 17:31:57 +0300127#define CCKM_KRK_CIPHER_SUITE 0x004096ff /* use for KRK */
128
Kalle Valo10509f92011-12-13 14:52:07 +0200129/* returns true if scheduled scan was stopped */
130static bool __ath6kl_cfg80211_sscan_stop(struct ath6kl_vif *vif)
131{
132 struct ath6kl *ar = vif->ar;
133
134 if (ar->state != ATH6KL_STATE_SCHED_SCAN)
135 return false;
136
137 del_timer_sync(&vif->sched_scan_timer);
138
139 ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
140 ATH6KL_HOST_MODE_AWAKE);
141
142 ar->state = ATH6KL_STATE_ON;
143
144 return true;
145}
146
147static void ath6kl_cfg80211_sscan_disable(struct ath6kl_vif *vif)
148{
149 struct ath6kl *ar = vif->ar;
150 bool stopped;
151
152 stopped = __ath6kl_cfg80211_sscan_stop(vif);
153
154 if (!stopped)
155 return;
156
157 cfg80211_sched_scan_stopped(ar->wiphy);
158}
159
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530160static int ath6kl_set_wpa_version(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300161 enum nl80211_wpa_versions wpa_version)
162{
163 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: %u\n", __func__, wpa_version);
164
165 if (!wpa_version) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530166 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300167 } else if (wpa_version & NL80211_WPA_VERSION_2) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530168 vif->auth_mode = WPA2_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300169 } else if (wpa_version & NL80211_WPA_VERSION_1) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530170 vif->auth_mode = WPA_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300171 } else {
172 ath6kl_err("%s: %u not supported\n", __func__, wpa_version);
173 return -ENOTSUPP;
174 }
175
176 return 0;
177}
178
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530179static int ath6kl_set_auth_type(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300180 enum nl80211_auth_type auth_type)
181{
Kalle Valobdcd8172011-07-18 00:22:30 +0300182 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, auth_type);
183
184 switch (auth_type) {
185 case NL80211_AUTHTYPE_OPEN_SYSTEM:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530186 vif->dot11_auth_mode = OPEN_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300187 break;
188 case NL80211_AUTHTYPE_SHARED_KEY:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530189 vif->dot11_auth_mode = SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300190 break;
191 case NL80211_AUTHTYPE_NETWORK_EAP:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530192 vif->dot11_auth_mode = LEAP_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300193 break;
194
195 case NL80211_AUTHTYPE_AUTOMATIC:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530196 vif->dot11_auth_mode = OPEN_AUTH | SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300197 break;
198
199 default:
200 ath6kl_err("%s: 0x%x not spported\n", __func__, auth_type);
201 return -ENOTSUPP;
202 }
203
204 return 0;
205}
206
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530207static int ath6kl_set_cipher(struct ath6kl_vif *vif, u32 cipher, bool ucast)
Kalle Valobdcd8172011-07-18 00:22:30 +0300208{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530209 u8 *ar_cipher = ucast ? &vif->prwise_crypto : &vif->grp_crypto;
210 u8 *ar_cipher_len = ucast ? &vif->prwise_crypto_len :
211 &vif->grp_crypto_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300212
213 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: cipher 0x%x, ucast %u\n",
214 __func__, cipher, ucast);
215
216 switch (cipher) {
217 case 0:
218 /* our own hack to use value 0 as no crypto used */
219 *ar_cipher = NONE_CRYPT;
220 *ar_cipher_len = 0;
221 break;
222 case WLAN_CIPHER_SUITE_WEP40:
223 *ar_cipher = WEP_CRYPT;
224 *ar_cipher_len = 5;
225 break;
226 case WLAN_CIPHER_SUITE_WEP104:
227 *ar_cipher = WEP_CRYPT;
228 *ar_cipher_len = 13;
229 break;
230 case WLAN_CIPHER_SUITE_TKIP:
231 *ar_cipher = TKIP_CRYPT;
232 *ar_cipher_len = 0;
233 break;
234 case WLAN_CIPHER_SUITE_CCMP:
235 *ar_cipher = AES_CRYPT;
236 *ar_cipher_len = 0;
237 break;
Dai Shuibing5e070212011-11-03 11:39:37 +0200238 case WLAN_CIPHER_SUITE_SMS4:
239 *ar_cipher = WAPI_CRYPT;
240 *ar_cipher_len = 0;
241 break;
Kalle Valobdcd8172011-07-18 00:22:30 +0300242 default:
243 ath6kl_err("cipher 0x%x not supported\n", cipher);
244 return -ENOTSUPP;
245 }
246
247 return 0;
248}
249
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530250static void ath6kl_set_key_mgmt(struct ath6kl_vif *vif, u32 key_mgmt)
Kalle Valobdcd8172011-07-18 00:22:30 +0300251{
252 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, key_mgmt);
253
254 if (key_mgmt == WLAN_AKM_SUITE_PSK) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530255 if (vif->auth_mode == WPA_AUTH)
256 vif->auth_mode = WPA_PSK_AUTH;
257 else if (vif->auth_mode == WPA2_AUTH)
258 vif->auth_mode = WPA2_PSK_AUTH;
Jouni Malinen837cb972011-10-11 17:31:57 +0300259 } else if (key_mgmt == 0x00409600) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530260 if (vif->auth_mode == WPA_AUTH)
261 vif->auth_mode = WPA_AUTH_CCKM;
262 else if (vif->auth_mode == WPA2_AUTH)
263 vif->auth_mode = WPA2_AUTH_CCKM;
Kalle Valobdcd8172011-07-18 00:22:30 +0300264 } else if (key_mgmt != WLAN_AKM_SUITE_8021X) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530265 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300266 }
267}
268
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530269static bool ath6kl_cfg80211_ready(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +0300270{
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530271 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530272
Kalle Valobdcd8172011-07-18 00:22:30 +0300273 if (!test_bit(WMI_READY, &ar->flag)) {
274 ath6kl_err("wmi is not ready\n");
275 return false;
276 }
277
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530278 if (!test_bit(WLAN_ENABLED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300279 ath6kl_err("wlan disabled\n");
280 return false;
281 }
282
283 return true;
284}
285
Kevin Fang6981ffd2011-10-07 08:51:19 +0800286static bool ath6kl_is_wpa_ie(const u8 *pos)
287{
288 return pos[0] == WLAN_EID_WPA && pos[1] >= 4 &&
289 pos[2] == 0x00 && pos[3] == 0x50 &&
290 pos[4] == 0xf2 && pos[5] == 0x01;
291}
292
293static bool ath6kl_is_rsn_ie(const u8 *pos)
294{
295 return pos[0] == WLAN_EID_RSN;
296}
297
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700298static bool ath6kl_is_wps_ie(const u8 *pos)
299{
300 return (pos[0] == WLAN_EID_VENDOR_SPECIFIC &&
301 pos[1] >= 4 &&
302 pos[2] == 0x00 && pos[3] == 0x50 && pos[4] == 0xf2 &&
303 pos[5] == 0x04);
304}
305
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530306static int ath6kl_set_assoc_req_ies(struct ath6kl_vif *vif, const u8 *ies,
307 size_t ies_len)
Kevin Fang6981ffd2011-10-07 08:51:19 +0800308{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530309 struct ath6kl *ar = vif->ar;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800310 const u8 *pos;
311 u8 *buf = NULL;
312 size_t len = 0;
313 int ret;
314
315 /*
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700316 * Clear previously set flag
317 */
318
319 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
320
321 /*
Kevin Fang6981ffd2011-10-07 08:51:19 +0800322 * Filter out RSN/WPA IE(s)
323 */
324
325 if (ies && ies_len) {
326 buf = kmalloc(ies_len, GFP_KERNEL);
327 if (buf == NULL)
328 return -ENOMEM;
329 pos = ies;
330
331 while (pos + 1 < ies + ies_len) {
332 if (pos + 2 + pos[1] > ies + ies_len)
333 break;
334 if (!(ath6kl_is_wpa_ie(pos) || ath6kl_is_rsn_ie(pos))) {
335 memcpy(buf + len, pos, 2 + pos[1]);
336 len += 2 + pos[1];
337 }
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700338
339 if (ath6kl_is_wps_ie(pos))
340 ar->connect_ctrl_flags |= CONNECT_WPS_FLAG;
341
Kevin Fang6981ffd2011-10-07 08:51:19 +0800342 pos += 2 + pos[1];
343 }
344 }
345
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530346 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
347 WMI_FRAME_ASSOC_REQ, buf, len);
Kevin Fang6981ffd2011-10-07 08:51:19 +0800348 kfree(buf);
349 return ret;
350}
351
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530352static int ath6kl_nliftype_to_drv_iftype(enum nl80211_iftype type, u8 *nw_type)
353{
354 switch (type) {
355 case NL80211_IFTYPE_STATION:
356 *nw_type = INFRA_NETWORK;
357 break;
358 case NL80211_IFTYPE_ADHOC:
359 *nw_type = ADHOC_NETWORK;
360 break;
361 case NL80211_IFTYPE_AP:
362 *nw_type = AP_NETWORK;
363 break;
364 case NL80211_IFTYPE_P2P_CLIENT:
365 *nw_type = INFRA_NETWORK;
366 break;
367 case NL80211_IFTYPE_P2P_GO:
368 *nw_type = AP_NETWORK;
369 break;
370 default:
371 ath6kl_err("invalid interface type %u\n", type);
372 return -ENOTSUPP;
373 }
374
375 return 0;
376}
377
378static bool ath6kl_is_valid_iftype(struct ath6kl *ar, enum nl80211_iftype type,
379 u8 *if_idx, u8 *nw_type)
380{
381 int i;
382
383 if (ath6kl_nliftype_to_drv_iftype(type, nw_type))
384 return false;
385
386 if (ar->ibss_if_active || ((type == NL80211_IFTYPE_ADHOC) &&
387 ar->num_vif))
388 return false;
389
390 if (type == NL80211_IFTYPE_STATION ||
391 type == NL80211_IFTYPE_AP || type == NL80211_IFTYPE_ADHOC) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200392 for (i = 0; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530393 if ((ar->avail_idx_map >> i) & BIT(0)) {
394 *if_idx = i;
395 return true;
396 }
397 }
398 }
399
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530400 if (type == NL80211_IFTYPE_P2P_CLIENT ||
401 type == NL80211_IFTYPE_P2P_GO) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200402 for (i = ar->max_norm_iface; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530403 if ((ar->avail_idx_map >> i) & BIT(0)) {
404 *if_idx = i;
405 return true;
406 }
407 }
408 }
409
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530410 return false;
411}
412
Kalle Valobdcd8172011-07-18 00:22:30 +0300413static int ath6kl_cfg80211_connect(struct wiphy *wiphy, struct net_device *dev,
414 struct cfg80211_connect_params *sme)
415{
416 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530417 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300418 int status;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800419 u8 nw_subtype = (ar->p2p) ? SUBTYPE_P2PDEV : SUBTYPE_NONE;
Kalle Valobdcd8172011-07-18 00:22:30 +0300420
Kalle Valo10509f92011-12-13 14:52:07 +0200421 ath6kl_cfg80211_sscan_disable(vif);
422
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530423 vif->sme_state = SME_CONNECTING;
Kalle Valobdcd8172011-07-18 00:22:30 +0300424
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530425 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300426 return -EIO;
427
428 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
429 ath6kl_err("destroy in progress\n");
430 return -EBUSY;
431 }
432
433 if (test_bit(SKIP_SCAN, &ar->flag) &&
434 ((sme->channel && sme->channel->center_freq == 0) ||
435 (sme->bssid && is_zero_ether_addr(sme->bssid)))) {
436 ath6kl_err("SkipScan: channel or bssid invalid\n");
437 return -EINVAL;
438 }
439
440 if (down_interruptible(&ar->sem)) {
441 ath6kl_err("busy, couldn't get access\n");
442 return -ERESTARTSYS;
443 }
444
445 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
446 ath6kl_err("busy, destroy in progress\n");
447 up(&ar->sem);
448 return -EBUSY;
449 }
450
451 if (ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)]) {
452 /*
453 * sleep until the command queue drains
454 */
455 wait_event_interruptible_timeout(ar->event_wq,
456 ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)] == 0,
457 WMI_TIMEOUT);
458 if (signal_pending(current)) {
459 ath6kl_err("cmd queue drain timeout\n");
460 up(&ar->sem);
461 return -EINTR;
462 }
463 }
464
Jouni Malinen6e786cb2011-12-15 14:16:00 +0200465 status = ath6kl_set_assoc_req_ies(vif, sme->ie, sme->ie_len);
466 if (status) {
467 up(&ar->sem);
468 return status;
469 }
470
471 if (sme->ie == NULL || sme->ie_len == 0)
Raja Mani542c5192011-11-15 14:14:56 +0530472 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800473
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530474 if (test_bit(CONNECTED, &vif->flags) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530475 vif->ssid_len == sme->ssid_len &&
476 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530477 vif->reconnect_flag = true;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530478 status = ath6kl_wmi_reconnect_cmd(ar->wmi, vif->fw_vif_idx,
479 vif->req_bssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530480 vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300481
482 up(&ar->sem);
483 if (status) {
484 ath6kl_err("wmi_reconnect_cmd failed\n");
485 return -EIO;
486 }
487 return 0;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530488 } else if (vif->ssid_len == sme->ssid_len &&
489 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530490 ath6kl_disconnect(vif);
Kalle Valobdcd8172011-07-18 00:22:30 +0300491 }
492
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530493 memset(vif->ssid, 0, sizeof(vif->ssid));
494 vif->ssid_len = sme->ssid_len;
495 memcpy(vif->ssid, sme->ssid, sme->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +0300496
497 if (sme->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530498 vif->ch_hint = sme->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +0300499
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530500 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300501 if (sme->bssid && !is_broadcast_ether_addr(sme->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530502 memcpy(vif->req_bssid, sme->bssid, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300503
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530504 ath6kl_set_wpa_version(vif, sme->crypto.wpa_versions);
Kalle Valobdcd8172011-07-18 00:22:30 +0300505
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530506 status = ath6kl_set_auth_type(vif, sme->auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300507 if (status) {
508 up(&ar->sem);
509 return status;
510 }
511
512 if (sme->crypto.n_ciphers_pairwise)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530513 ath6kl_set_cipher(vif, sme->crypto.ciphers_pairwise[0], true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300514 else
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530515 ath6kl_set_cipher(vif, 0, true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300516
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530517 ath6kl_set_cipher(vif, sme->crypto.cipher_group, false);
Kalle Valobdcd8172011-07-18 00:22:30 +0300518
519 if (sme->crypto.n_akm_suites)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530520 ath6kl_set_key_mgmt(vif, sme->crypto.akm_suites[0]);
Kalle Valobdcd8172011-07-18 00:22:30 +0300521
522 if ((sme->key_len) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530523 (vif->auth_mode == NONE_AUTH) &&
524 (vif->prwise_crypto == WEP_CRYPT)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300525 struct ath6kl_key *key = NULL;
526
527 if (sme->key_idx < WMI_MIN_KEY_INDEX ||
528 sme->key_idx > WMI_MAX_KEY_INDEX) {
529 ath6kl_err("key index %d out of bounds\n",
530 sme->key_idx);
531 up(&ar->sem);
532 return -ENOENT;
533 }
534
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530535 key = &vif->keys[sme->key_idx];
Kalle Valobdcd8172011-07-18 00:22:30 +0300536 key->key_len = sme->key_len;
537 memcpy(key->key, sme->key, key->key_len);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530538 key->cipher = vif->prwise_crypto;
539 vif->def_txkey_index = sme->key_idx;
Kalle Valobdcd8172011-07-18 00:22:30 +0300540
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530541 ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, sme->key_idx,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530542 vif->prwise_crypto,
Kalle Valobdcd8172011-07-18 00:22:30 +0300543 GROUP_USAGE | TX_USAGE,
544 key->key_len,
Jouni Malinenf4bb9a62011-11-02 23:45:55 +0200545 NULL, 0,
Kalle Valobdcd8172011-07-18 00:22:30 +0300546 key->key, KEY_OP_INIT_VAL, NULL,
547 NO_SYNC_WMIFLAG);
548 }
549
550 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530551 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530552 if (ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
553 ALL_BSS_FILTER, 0) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300554 ath6kl_err("couldn't set bss filtering\n");
555 up(&ar->sem);
556 return -EIO;
557 }
558 }
559
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530560 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +0300561
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800562 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_CLIENT)
563 nw_subtype = SUBTYPE_P2PCLIENT;
564
Kalle Valobdcd8172011-07-18 00:22:30 +0300565 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
566 "%s: connect called with authmode %d dot11 auth %d"
567 " PW crypto %d PW crypto len %d GRP crypto %d"
568 " GRP crypto len %d channel hint %u\n",
569 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530570 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
571 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530572 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300573
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530574 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530575 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530576 vif->dot11_auth_mode, vif->auth_mode,
577 vif->prwise_crypto,
578 vif->prwise_crypto_len,
579 vif->grp_crypto, vif->grp_crypto_len,
580 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530581 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800582 ar->connect_ctrl_flags, nw_subtype);
Kalle Valobdcd8172011-07-18 00:22:30 +0300583
584 up(&ar->sem);
585
586 if (status == -EINVAL) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530587 memset(vif->ssid, 0, sizeof(vif->ssid));
588 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300589 ath6kl_err("invalid request\n");
590 return -ENOENT;
591 } else if (status) {
592 ath6kl_err("ath6kl_wmi_connect_cmd failed\n");
593 return -EIO;
594 }
595
596 if ((!(ar->connect_ctrl_flags & CONNECT_DO_WPA_OFFLOAD)) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530597 ((vif->auth_mode == WPA_PSK_AUTH)
598 || (vif->auth_mode == WPA2_PSK_AUTH))) {
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +0530599 mod_timer(&vif->disconnect_timer,
Kalle Valobdcd8172011-07-18 00:22:30 +0300600 jiffies + msecs_to_jiffies(DISCON_TIMER_INTVAL));
601 }
602
603 ar->connect_ctrl_flags &= ~CONNECT_DO_WPA_OFFLOAD;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530604 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300605
606 return 0;
607}
608
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530609static struct cfg80211_bss *
610ath6kl_add_bss_if_needed(struct ath6kl_vif *vif,
611 enum network_type nw_type,
612 const u8 *bssid,
613 struct ieee80211_channel *chan,
614 const u8 *beacon_ie,
615 size_t beacon_ie_len)
Jouni Malinen01cac472011-09-19 19:14:59 +0300616{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530617 struct ath6kl *ar = vif->ar;
Jouni Malinen01cac472011-09-19 19:14:59 +0300618 struct cfg80211_bss *bss;
Raja Mani4eab6f42011-11-09 17:02:23 +0530619 u16 cap_mask, cap_val;
Jouni Malinen01cac472011-09-19 19:14:59 +0300620 u8 *ie;
621
Raja Mani4eab6f42011-11-09 17:02:23 +0530622 if (nw_type & ADHOC_NETWORK) {
623 cap_mask = WLAN_CAPABILITY_IBSS;
624 cap_val = WLAN_CAPABILITY_IBSS;
625 } else {
626 cap_mask = WLAN_CAPABILITY_ESS;
627 cap_val = WLAN_CAPABILITY_ESS;
628 }
629
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530630 bss = cfg80211_get_bss(ar->wiphy, chan, bssid,
Raja Mani4eab6f42011-11-09 17:02:23 +0530631 vif->ssid, vif->ssid_len,
632 cap_mask, cap_val);
Jouni Malinen01cac472011-09-19 19:14:59 +0300633 if (bss == NULL) {
634 /*
635 * Since cfg80211 may not yet know about the BSS,
636 * generate a partial entry until the first BSS info
637 * event becomes available.
638 *
639 * Prepend SSID element since it is not included in the Beacon
640 * IEs from the target.
641 */
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530642 ie = kmalloc(2 + vif->ssid_len + beacon_ie_len, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300643 if (ie == NULL)
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530644 return NULL;
Jouni Malinen01cac472011-09-19 19:14:59 +0300645 ie[0] = WLAN_EID_SSID;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530646 ie[1] = vif->ssid_len;
647 memcpy(ie + 2, vif->ssid, vif->ssid_len);
648 memcpy(ie + 2 + vif->ssid_len, beacon_ie, beacon_ie_len);
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530649 bss = cfg80211_inform_bss(ar->wiphy, chan,
Raja Mani4eab6f42011-11-09 17:02:23 +0530650 bssid, 0, cap_val, 100,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530651 ie, 2 + vif->ssid_len + beacon_ie_len,
Jouni Malinen01cac472011-09-19 19:14:59 +0300652 0, GFP_KERNEL);
653 if (bss)
Raja Mani4eab6f42011-11-09 17:02:23 +0530654 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "added bss %pM to "
655 "cfg80211\n", bssid);
Jouni Malinen01cac472011-09-19 19:14:59 +0300656 kfree(ie);
657 } else
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530658 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "cfg80211 already has a bss\n");
Jouni Malinen01cac472011-09-19 19:14:59 +0300659
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530660 return bss;
Jouni Malinen01cac472011-09-19 19:14:59 +0300661}
662
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530663void ath6kl_cfg80211_connect_event(struct ath6kl_vif *vif, u16 channel,
Kalle Valobdcd8172011-07-18 00:22:30 +0300664 u8 *bssid, u16 listen_intvl,
665 u16 beacon_intvl,
666 enum network_type nw_type,
667 u8 beacon_ie_len, u8 assoc_req_len,
668 u8 assoc_resp_len, u8 *assoc_info)
669{
Jouni Malinen01cac472011-09-19 19:14:59 +0300670 struct ieee80211_channel *chan;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530671 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530672 struct cfg80211_bss *bss;
Kalle Valobdcd8172011-07-18 00:22:30 +0300673
674 /* capinfo + listen interval */
675 u8 assoc_req_ie_offset = sizeof(u16) + sizeof(u16);
676
677 /* capinfo + status code + associd */
678 u8 assoc_resp_ie_offset = sizeof(u16) + sizeof(u16) + sizeof(u16);
679
680 u8 *assoc_req_ie = assoc_info + beacon_ie_len + assoc_req_ie_offset;
681 u8 *assoc_resp_ie = assoc_info + beacon_ie_len + assoc_req_len +
682 assoc_resp_ie_offset;
683
684 assoc_req_len -= assoc_req_ie_offset;
685 assoc_resp_len -= assoc_resp_ie_offset;
686
Jouni Malinen32c10872011-09-19 19:15:07 +0300687 /*
688 * Store Beacon interval here; DTIM period will be available only once
689 * a Beacon frame from the AP is seen.
690 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530691 vif->assoc_bss_beacon_int = beacon_intvl;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530692 clear_bit(DTIM_PERIOD_AVAIL, &vif->flags);
Jouni Malinen32c10872011-09-19 19:15:07 +0300693
Kalle Valobdcd8172011-07-18 00:22:30 +0300694 if (nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530695 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300696 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
697 "%s: ath6k not in ibss mode\n", __func__);
698 return;
699 }
700 }
701
702 if (nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530703 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
704 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300705 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
706 "%s: ath6k not in station mode\n", __func__);
707 return;
708 }
709 }
710
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530711 chan = ieee80211_get_channel(ar->wiphy, (int) channel);
Kalle Valobdcd8172011-07-18 00:22:30 +0300712
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530713 bss = ath6kl_add_bss_if_needed(vif, nw_type, bssid, chan,
714 assoc_info, beacon_ie_len);
715 if (!bss) {
Raja Mani4eab6f42011-11-09 17:02:23 +0530716 ath6kl_err("could not add cfg80211 bss entry\n");
Kalle Valobdcd8172011-07-18 00:22:30 +0300717 return;
718 }
719
Raja Mani4eab6f42011-11-09 17:02:23 +0530720 if (nw_type & ADHOC_NETWORK) {
721 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "ad-hoc %s selected\n",
722 nw_type & ADHOC_CREATOR ? "creator" : "joiner");
723 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530724 cfg80211_put_bss(bss);
Jouni Malinen01cac472011-09-19 19:14:59 +0300725 return;
726 }
727
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530728 if (vif->sme_state == SME_CONNECTING) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300729 /* inform connect result to cfg80211 */
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530730 vif->sme_state = SME_CONNECTED;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530731 cfg80211_connect_result(vif->ndev, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300732 assoc_req_ie, assoc_req_len,
733 assoc_resp_ie, assoc_resp_len,
734 WLAN_STATUS_SUCCESS, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530735 cfg80211_put_bss(bss);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530736 } else if (vif->sme_state == SME_CONNECTED) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300737 /* inform roam event to cfg80211 */
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530738 cfg80211_roamed_bss(vif->ndev, bss, assoc_req_ie, assoc_req_len,
739 assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300740 }
741}
742
743static int ath6kl_cfg80211_disconnect(struct wiphy *wiphy,
744 struct net_device *dev, u16 reason_code)
745{
Kalle Valod6d5c062011-11-25 13:17:37 +0200746 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530747 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300748
749 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: reason=%u\n", __func__,
750 reason_code);
751
Kalle Valo10509f92011-12-13 14:52:07 +0200752 ath6kl_cfg80211_sscan_disable(vif);
753
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530754 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300755 return -EIO;
756
757 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
758 ath6kl_err("busy, destroy in progress\n");
759 return -EBUSY;
760 }
761
762 if (down_interruptible(&ar->sem)) {
763 ath6kl_err("busy, couldn't get access\n");
764 return -ERESTARTSYS;
765 }
766
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530767 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530768 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530769 memset(vif->ssid, 0, sizeof(vif->ssid));
770 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300771
772 if (!test_bit(SKIP_SCAN, &ar->flag))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530773 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300774
775 up(&ar->sem);
776
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530777 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan170826d2011-09-10 15:26:35 +0530778
Kalle Valobdcd8172011-07-18 00:22:30 +0300779 return 0;
780}
781
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530782void ath6kl_cfg80211_disconnect_event(struct ath6kl_vif *vif, u8 reason,
Kalle Valobdcd8172011-07-18 00:22:30 +0300783 u8 *bssid, u8 assoc_resp_len,
784 u8 *assoc_info, u16 proto_reason)
785{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530786 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530787
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530788 if (vif->scan_req) {
789 cfg80211_scan_done(vif->scan_req, true);
790 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300791 }
792
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530793 if (vif->nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530794 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300795 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
796 "%s: ath6k not in ibss mode\n", __func__);
797 return;
798 }
799 memset(bssid, 0, ETH_ALEN);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530800 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300801 return;
802 }
803
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530804 if (vif->nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530805 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
806 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300807 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
808 "%s: ath6k not in station mode\n", __func__);
809 return;
810 }
811 }
812
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530813 /*
814 * Send a disconnect command to target when a disconnect event is
815 * received with reason code other than 3 (DISCONNECT_CMD - disconnect
816 * request from host) to make the firmware stop trying to connect even
817 * after giving disconnect event. There will be one more disconnect
818 * event for this disconnect command with reason code DISCONNECT_CMD
819 * which will be notified to cfg80211.
820 */
Kalle Valobdcd8172011-07-18 00:22:30 +0300821
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530822 if (reason != DISCONNECT_CMD) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530823 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +0300824 return;
825 }
826
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530827 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300828
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530829 if (vif->sme_state == SME_CONNECTING) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530830 cfg80211_connect_result(vif->ndev,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530831 bssid, NULL, 0,
832 NULL, 0,
833 WLAN_STATUS_UNSPECIFIED_FAILURE,
834 GFP_KERNEL);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530835 } else if (vif->sme_state == SME_CONNECTED) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530836 cfg80211_disconnected(vif->ndev, reason,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530837 NULL, 0, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300838 }
839
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530840 vif->sme_state = SME_DISCONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300841}
842
Kalle Valobdcd8172011-07-18 00:22:30 +0300843static int ath6kl_cfg80211_scan(struct wiphy *wiphy, struct net_device *ndev,
844 struct cfg80211_scan_request *request)
845{
Kalle Valod6d5c062011-11-25 13:17:37 +0200846 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530847 struct ath6kl_vif *vif = netdev_priv(ndev);
Edward Lu1276c9e2011-08-30 21:58:00 +0300848 s8 n_channels = 0;
849 u16 *channels = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300850 int ret = 0;
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530851 u32 force_fg_scan = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300852
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530853 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300854 return -EIO;
855
Kalle Valo10509f92011-12-13 14:52:07 +0200856 ath6kl_cfg80211_sscan_disable(vif);
857
Kalle Valobdcd8172011-07-18 00:22:30 +0300858 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530859 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300860 ret = ath6kl_wmi_bssfilter_cmd(
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530861 ar->wmi, vif->fw_vif_idx,
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530862 (test_bit(CONNECTED, &vif->flags) ?
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300863 ALL_BUT_BSS_FILTER : ALL_BSS_FILTER), 0);
864 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300865 ath6kl_err("couldn't set bss filtering\n");
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300866 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +0300867 }
868 }
869
870 if (request->n_ssids && request->ssids[0].ssid_len) {
871 u8 i;
872
873 if (request->n_ssids > (MAX_PROBED_SSID_INDEX - 1))
874 request->n_ssids = MAX_PROBED_SSID_INDEX - 1;
875
876 for (i = 0; i < request->n_ssids; i++)
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530877 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
878 i + 1, SPECIFIC_SSID_FLAG,
Kalle Valobdcd8172011-07-18 00:22:30 +0300879 request->ssids[i].ssid_len,
880 request->ssids[i].ssid);
881 }
882
Kalle Valo10509f92011-12-13 14:52:07 +0200883 /*
884 * FIXME: we should clear the IE in fw if it's not set so just
885 * remove the check altogether
886 */
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300887 if (request->ie) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530888 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
889 WMI_FRAME_PROBE_REQ,
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300890 request->ie, request->ie_len);
891 if (ret) {
892 ath6kl_err("failed to set Probe Request appie for "
893 "scan");
894 return ret;
895 }
896 }
897
Jouni Malinen11869be2011-09-02 20:07:06 +0300898 /*
899 * Scan only the requested channels if the request specifies a set of
900 * channels. If the list is longer than the target supports, do not
901 * configure the list and instead, scan all available channels.
902 */
903 if (request->n_channels > 0 &&
904 request->n_channels <= WMI_MAX_CHANNELS) {
Edward Lu1276c9e2011-08-30 21:58:00 +0300905 u8 i;
906
Jouni Malinen11869be2011-09-02 20:07:06 +0300907 n_channels = request->n_channels;
Edward Lu1276c9e2011-08-30 21:58:00 +0300908
909 channels = kzalloc(n_channels * sizeof(u16), GFP_KERNEL);
910 if (channels == NULL) {
911 ath6kl_warn("failed to set scan channels, "
912 "scan all channels");
913 n_channels = 0;
914 }
915
916 for (i = 0; i < n_channels; i++)
917 channels[i] = request->channels[i]->center_freq;
918 }
919
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530920 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530921 force_fg_scan = 1;
922
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800923 if (test_bit(ATH6KL_FW_CAPABILITY_STA_P2PDEV_DUPLEX,
924 ar->fw_capabilities)) {
925 /*
926 * If capable of doing P2P mgmt operations using
927 * station interface, send additional information like
928 * supported rates to advertise and xmit rates for
929 * probe requests
930 */
931 ret = ath6kl_wmi_beginscan_cmd(ar->wmi, vif->fw_vif_idx,
932 WMI_LONG_SCAN, force_fg_scan,
933 false, 0, 0, n_channels,
934 channels, request->no_cck,
935 request->rates);
936 } else {
937 ret = ath6kl_wmi_startscan_cmd(ar->wmi, vif->fw_vif_idx,
938 WMI_LONG_SCAN, force_fg_scan,
939 false, 0, 0, n_channels,
940 channels);
941 }
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300942 if (ret)
Kalle Valobdcd8172011-07-18 00:22:30 +0300943 ath6kl_err("wmi_startscan_cmd failed\n");
Jouni Malinen11869be2011-09-02 20:07:06 +0300944 else
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530945 vif->scan_req = request;
Kalle Valobdcd8172011-07-18 00:22:30 +0300946
Edward Lu1276c9e2011-08-30 21:58:00 +0300947 kfree(channels);
948
Kalle Valobdcd8172011-07-18 00:22:30 +0300949 return ret;
950}
951
Kalle Valo1c17d312011-11-01 08:43:56 +0200952void ath6kl_cfg80211_scan_complete_event(struct ath6kl_vif *vif, bool aborted)
Kalle Valobdcd8172011-07-18 00:22:30 +0300953{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530954 struct ath6kl *ar = vif->ar;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300955 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +0300956
Kalle Valo1c17d312011-11-01 08:43:56 +0200957 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: status%s\n", __func__,
958 aborted ? " aborted" : "");
Kalle Valobdcd8172011-07-18 00:22:30 +0300959
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530960 if (!vif->scan_req)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300961 return;
Kalle Valobdcd8172011-07-18 00:22:30 +0300962
Kalle Valo1c17d312011-11-01 08:43:56 +0200963 if (aborted)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300964 goto out;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300965
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530966 if (vif->scan_req->n_ssids && vif->scan_req->ssids[0].ssid_len) {
967 for (i = 0; i < vif->scan_req->n_ssids; i++) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530968 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
969 i + 1, DISABLE_SSID_FLAG,
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300970 0, NULL);
971 }
972 }
973
974out:
Kalle Valocb938212011-10-27 18:47:46 +0300975 cfg80211_scan_done(vif->scan_req, aborted);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530976 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300977}
978
979static int ath6kl_cfg80211_add_key(struct wiphy *wiphy, struct net_device *ndev,
980 u8 key_index, bool pairwise,
981 const u8 *mac_addr,
982 struct key_params *params)
983{
Kalle Valod6d5c062011-11-25 13:17:37 +0200984 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530985 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300986 struct ath6kl_key *key = NULL;
987 u8 key_usage;
988 u8 key_type;
Kalle Valobdcd8172011-07-18 00:22:30 +0300989
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530990 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300991 return -EIO;
992
Jouni Malinen837cb972011-10-11 17:31:57 +0300993 if (params->cipher == CCKM_KRK_CIPHER_SUITE) {
994 if (params->key_len != WMI_KRK_LEN)
995 return -EINVAL;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530996 return ath6kl_wmi_add_krk_cmd(ar->wmi, vif->fw_vif_idx,
997 params->key);
Jouni Malinen837cb972011-10-11 17:31:57 +0300998 }
999
Kalle Valobdcd8172011-07-18 00:22:30 +03001000 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1001 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1002 "%s: key index %d out of bounds\n", __func__,
1003 key_index);
1004 return -ENOENT;
1005 }
1006
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301007 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001008 memset(key, 0, sizeof(struct ath6kl_key));
1009
1010 if (pairwise)
1011 key_usage = PAIRWISE_USAGE;
1012 else
1013 key_usage = GROUP_USAGE;
1014
1015 if (params) {
Dai Shuibing5e070212011-11-03 11:39:37 +02001016 int seq_len = params->seq_len;
1017 if (params->cipher == WLAN_CIPHER_SUITE_SMS4 &&
1018 seq_len > ATH6KL_KEY_SEQ_LEN) {
1019 /* Only first half of the WPI PN is configured */
1020 seq_len = ATH6KL_KEY_SEQ_LEN;
1021 }
Kalle Valobdcd8172011-07-18 00:22:30 +03001022 if (params->key_len > WLAN_MAX_KEY_LEN ||
Dai Shuibing5e070212011-11-03 11:39:37 +02001023 seq_len > sizeof(key->seq))
Kalle Valobdcd8172011-07-18 00:22:30 +03001024 return -EINVAL;
1025
1026 key->key_len = params->key_len;
1027 memcpy(key->key, params->key, key->key_len);
Dai Shuibing5e070212011-11-03 11:39:37 +02001028 key->seq_len = seq_len;
Kalle Valobdcd8172011-07-18 00:22:30 +03001029 memcpy(key->seq, params->seq, key->seq_len);
1030 key->cipher = params->cipher;
1031 }
1032
1033 switch (key->cipher) {
1034 case WLAN_CIPHER_SUITE_WEP40:
1035 case WLAN_CIPHER_SUITE_WEP104:
1036 key_type = WEP_CRYPT;
1037 break;
1038
1039 case WLAN_CIPHER_SUITE_TKIP:
1040 key_type = TKIP_CRYPT;
1041 break;
1042
1043 case WLAN_CIPHER_SUITE_CCMP:
1044 key_type = AES_CRYPT;
1045 break;
Dai Shuibing5e070212011-11-03 11:39:37 +02001046 case WLAN_CIPHER_SUITE_SMS4:
1047 key_type = WAPI_CRYPT;
1048 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001049
1050 default:
1051 return -ENOTSUPP;
1052 }
1053
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301054 if (((vif->auth_mode == WPA_PSK_AUTH)
1055 || (vif->auth_mode == WPA2_PSK_AUTH))
Kalle Valobdcd8172011-07-18 00:22:30 +03001056 && (key_usage & GROUP_USAGE))
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05301057 del_timer(&vif->disconnect_timer);
Kalle Valobdcd8172011-07-18 00:22:30 +03001058
1059 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1060 "%s: index %d, key_len %d, key_type 0x%x, key_usage 0x%x, seq_len %d\n",
1061 __func__, key_index, key->key_len, key_type,
1062 key_usage, key->seq_len);
1063
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301064 if (vif->nw_type == AP_NETWORK && !pairwise &&
Jouni Malinen47032902011-12-08 16:50:30 +02001065 (key_type == TKIP_CRYPT || key_type == AES_CRYPT ||
1066 key_type == WAPI_CRYPT) && params) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001067 ar->ap_mode_bkey.valid = true;
1068 ar->ap_mode_bkey.key_index = key_index;
1069 ar->ap_mode_bkey.key_type = key_type;
1070 ar->ap_mode_bkey.key_len = key->key_len;
1071 memcpy(ar->ap_mode_bkey.key, key->key, key->key_len);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301072 if (!test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001073 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay initial group "
1074 "key configuration until AP mode has been "
1075 "started\n");
1076 /*
1077 * The key will be set in ath6kl_connect_ap_mode() once
1078 * the connected event is received from the target.
1079 */
1080 return 0;
1081 }
1082 }
1083
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301084 if (vif->next_mode == AP_NETWORK && key_type == WEP_CRYPT &&
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301085 !test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen151411e2011-09-15 15:10:16 +03001086 /*
1087 * Store the key locally so that it can be re-configured after
1088 * the AP mode has properly started
1089 * (ath6kl_install_statioc_wep_keys).
1090 */
1091 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay WEP key configuration "
1092 "until AP mode has been started\n");
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301093 vif->wep_key_list[key_index].key_len = key->key_len;
1094 memcpy(vif->wep_key_list[key_index].key, key->key,
1095 key->key_len);
Jouni Malinen151411e2011-09-15 15:10:16 +03001096 return 0;
1097 }
1098
Vasanthakumar Thiagarajan7cefa442011-11-11 20:33:00 +05301099 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, key_index,
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001100 key_type, key_usage, key->key_len,
1101 key->seq, key->seq_len, key->key,
1102 KEY_OP_INIT_VAL,
1103 (u8 *) mac_addr, SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001104}
1105
1106static int ath6kl_cfg80211_del_key(struct wiphy *wiphy, struct net_device *ndev,
1107 u8 key_index, bool pairwise,
1108 const u8 *mac_addr)
1109{
Kalle Valod6d5c062011-11-25 13:17:37 +02001110 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301111 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001112
1113 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1114
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301115 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001116 return -EIO;
1117
1118 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1119 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1120 "%s: key index %d out of bounds\n", __func__,
1121 key_index);
1122 return -ENOENT;
1123 }
1124
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301125 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001126 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1127 "%s: index %d is empty\n", __func__, key_index);
1128 return 0;
1129 }
1130
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301131 vif->keys[key_index].key_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001132
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301133 return ath6kl_wmi_deletekey_cmd(ar->wmi, vif->fw_vif_idx, key_index);
Kalle Valobdcd8172011-07-18 00:22:30 +03001134}
1135
1136static int ath6kl_cfg80211_get_key(struct wiphy *wiphy, struct net_device *ndev,
1137 u8 key_index, bool pairwise,
1138 const u8 *mac_addr, void *cookie,
1139 void (*callback) (void *cookie,
1140 struct key_params *))
1141{
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301142 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001143 struct ath6kl_key *key = NULL;
1144 struct key_params params;
1145
1146 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1147
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301148 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001149 return -EIO;
1150
1151 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1152 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1153 "%s: key index %d out of bounds\n", __func__,
1154 key_index);
1155 return -ENOENT;
1156 }
1157
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301158 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001159 memset(&params, 0, sizeof(params));
1160 params.cipher = key->cipher;
1161 params.key_len = key->key_len;
1162 params.seq_len = key->seq_len;
1163 params.seq = key->seq;
1164 params.key = key->key;
1165
1166 callback(cookie, &params);
1167
1168 return key->key_len ? 0 : -ENOENT;
1169}
1170
1171static int ath6kl_cfg80211_set_default_key(struct wiphy *wiphy,
1172 struct net_device *ndev,
1173 u8 key_index, bool unicast,
1174 bool multicast)
1175{
Kalle Valod6d5c062011-11-25 13:17:37 +02001176 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301177 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001178 struct ath6kl_key *key = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +03001179 u8 key_usage;
Edward Lu229ed6b2011-08-30 21:58:07 +03001180 enum crypto_type key_type = NONE_CRYPT;
Kalle Valobdcd8172011-07-18 00:22:30 +03001181
1182 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1183
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301184 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001185 return -EIO;
1186
1187 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1188 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1189 "%s: key index %d out of bounds\n",
1190 __func__, key_index);
1191 return -ENOENT;
1192 }
1193
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301194 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001195 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: invalid key index %d\n",
1196 __func__, key_index);
1197 return -EINVAL;
1198 }
1199
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301200 vif->def_txkey_index = key_index;
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301201 key = &vif->keys[vif->def_txkey_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001202 key_usage = GROUP_USAGE;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301203 if (vif->prwise_crypto == WEP_CRYPT)
Kalle Valobdcd8172011-07-18 00:22:30 +03001204 key_usage |= TX_USAGE;
Edward Lu229ed6b2011-08-30 21:58:07 +03001205 if (unicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301206 key_type = vif->prwise_crypto;
Edward Lu229ed6b2011-08-30 21:58:07 +03001207 if (multicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301208 key_type = vif->grp_crypto;
Kalle Valobdcd8172011-07-18 00:22:30 +03001209
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301210 if (vif->next_mode == AP_NETWORK && !test_bit(CONNECTED, &vif->flags))
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001211 return 0; /* Delay until AP mode has been started */
1212
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001213 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx,
1214 vif->def_txkey_index,
1215 key_type, key_usage,
1216 key->key_len, key->seq, key->seq_len,
1217 key->key,
1218 KEY_OP_INIT_VAL, NULL,
1219 SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001220}
1221
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301222void ath6kl_cfg80211_tkip_micerr_event(struct ath6kl_vif *vif, u8 keyid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001223 bool ismcast)
1224{
1225 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1226 "%s: keyid %d, ismcast %d\n", __func__, keyid, ismcast);
1227
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301228 cfg80211_michael_mic_failure(vif->ndev, vif->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001229 (ismcast ? NL80211_KEYTYPE_GROUP :
1230 NL80211_KEYTYPE_PAIRWISE), keyid, NULL,
1231 GFP_KERNEL);
1232}
1233
1234static int ath6kl_cfg80211_set_wiphy_params(struct wiphy *wiphy, u32 changed)
1235{
1236 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301237 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001238 int ret;
1239
1240 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: changed 0x%x\n", __func__,
1241 changed);
1242
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301243 vif = ath6kl_vif_first(ar);
1244 if (!vif)
1245 return -EIO;
1246
1247 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001248 return -EIO;
1249
1250 if (changed & WIPHY_PARAM_RTS_THRESHOLD) {
1251 ret = ath6kl_wmi_set_rts_cmd(ar->wmi, wiphy->rts_threshold);
1252 if (ret != 0) {
1253 ath6kl_err("ath6kl_wmi_set_rts_cmd failed\n");
1254 return -EIO;
1255 }
1256 }
1257
1258 return 0;
1259}
1260
1261/*
1262 * The type nl80211_tx_power_setting replaces the following
1263 * data type from 2.6.36 onwards
1264*/
1265static int ath6kl_cfg80211_set_txpower(struct wiphy *wiphy,
1266 enum nl80211_tx_power_setting type,
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001267 int mbm)
Kalle Valobdcd8172011-07-18 00:22:30 +03001268{
1269 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301270 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001271 u8 ath6kl_dbm;
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001272 int dbm = MBM_TO_DBM(mbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001273
1274 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x, dbm %d\n", __func__,
1275 type, dbm);
1276
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301277 vif = ath6kl_vif_first(ar);
1278 if (!vif)
1279 return -EIO;
1280
1281 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001282 return -EIO;
1283
1284 switch (type) {
1285 case NL80211_TX_POWER_AUTOMATIC:
1286 return 0;
1287 case NL80211_TX_POWER_LIMITED:
1288 ar->tx_pwr = ath6kl_dbm = dbm;
1289 break;
1290 default:
1291 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x not supported\n",
1292 __func__, type);
1293 return -EOPNOTSUPP;
1294 }
1295
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301296 ath6kl_wmi_set_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx, ath6kl_dbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001297
1298 return 0;
1299}
1300
1301static int ath6kl_cfg80211_get_txpower(struct wiphy *wiphy, int *dbm)
1302{
1303 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301304 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001305
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301306 vif = ath6kl_vif_first(ar);
1307 if (!vif)
1308 return -EIO;
1309
1310 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001311 return -EIO;
1312
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301313 if (test_bit(CONNECTED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001314 ar->tx_pwr = 0;
1315
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301316 if (ath6kl_wmi_get_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001317 ath6kl_err("ath6kl_wmi_get_tx_pwr_cmd failed\n");
1318 return -EIO;
1319 }
1320
1321 wait_event_interruptible_timeout(ar->event_wq, ar->tx_pwr != 0,
1322 5 * HZ);
1323
1324 if (signal_pending(current)) {
1325 ath6kl_err("target did not respond\n");
1326 return -EINTR;
1327 }
1328 }
1329
1330 *dbm = ar->tx_pwr;
1331 return 0;
1332}
1333
1334static int ath6kl_cfg80211_set_power_mgmt(struct wiphy *wiphy,
1335 struct net_device *dev,
1336 bool pmgmt, int timeout)
1337{
1338 struct ath6kl *ar = ath6kl_priv(dev);
1339 struct wmi_power_mode_cmd mode;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301340 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001341
1342 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: pmgmt %d, timeout %d\n",
1343 __func__, pmgmt, timeout);
1344
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301345 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001346 return -EIO;
1347
1348 if (pmgmt) {
1349 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: max perf\n", __func__);
1350 mode.pwr_mode = REC_POWER;
1351 } else {
1352 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: rec power\n", __func__);
1353 mode.pwr_mode = MAX_PERF_POWER;
1354 }
1355
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301356 if (ath6kl_wmi_powermode_cmd(ar->wmi, vif->fw_vif_idx,
1357 mode.pwr_mode) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001358 ath6kl_err("wmi_powermode_cmd failed\n");
1359 return -EIO;
1360 }
1361
1362 return 0;
1363}
1364
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301365static struct net_device *ath6kl_cfg80211_add_iface(struct wiphy *wiphy,
1366 char *name,
1367 enum nl80211_iftype type,
1368 u32 *flags,
1369 struct vif_params *params)
1370{
1371 struct ath6kl *ar = wiphy_priv(wiphy);
1372 struct net_device *ndev;
1373 u8 if_idx, nw_type;
1374
Kalle Valo71f96ee2011-11-14 19:31:30 +02001375 if (ar->num_vif == ar->vif_max) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301376 ath6kl_err("Reached maximum number of supported vif\n");
1377 return ERR_PTR(-EINVAL);
1378 }
1379
1380 if (!ath6kl_is_valid_iftype(ar, type, &if_idx, &nw_type)) {
1381 ath6kl_err("Not a supported interface type\n");
1382 return ERR_PTR(-EINVAL);
1383 }
1384
1385 ndev = ath6kl_interface_add(ar, name, type, if_idx, nw_type);
1386 if (!ndev)
1387 return ERR_PTR(-ENOMEM);
1388
1389 ar->num_vif++;
1390
1391 return ndev;
1392}
1393
1394static int ath6kl_cfg80211_del_iface(struct wiphy *wiphy,
1395 struct net_device *ndev)
1396{
1397 struct ath6kl *ar = wiphy_priv(wiphy);
1398 struct ath6kl_vif *vif = netdev_priv(ndev);
1399
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301400 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301401 list_del(&vif->list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301402 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301403
1404 ath6kl_cleanup_vif(vif, test_bit(WMI_READY, &ar->flag));
1405
1406 ath6kl_deinit_if_data(vif);
1407
1408 return 0;
1409}
1410
Kalle Valobdcd8172011-07-18 00:22:30 +03001411static int ath6kl_cfg80211_change_iface(struct wiphy *wiphy,
1412 struct net_device *ndev,
1413 enum nl80211_iftype type, u32 *flags,
1414 struct vif_params *params)
1415{
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301416 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001417
1418 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type %u\n", __func__, type);
1419
Kalle Valobdcd8172011-07-18 00:22:30 +03001420 switch (type) {
1421 case NL80211_IFTYPE_STATION:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301422 vif->next_mode = INFRA_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001423 break;
1424 case NL80211_IFTYPE_ADHOC:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301425 vif->next_mode = ADHOC_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001426 break;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001427 case NL80211_IFTYPE_AP:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301428 vif->next_mode = AP_NETWORK;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001429 break;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001430 case NL80211_IFTYPE_P2P_CLIENT:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301431 vif->next_mode = INFRA_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001432 break;
1433 case NL80211_IFTYPE_P2P_GO:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301434 vif->next_mode = AP_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001435 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001436 default:
1437 ath6kl_err("invalid interface type %u\n", type);
1438 return -EOPNOTSUPP;
1439 }
1440
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +05301441 vif->wdev.iftype = type;
Kalle Valobdcd8172011-07-18 00:22:30 +03001442
1443 return 0;
1444}
1445
1446static int ath6kl_cfg80211_join_ibss(struct wiphy *wiphy,
1447 struct net_device *dev,
1448 struct cfg80211_ibss_params *ibss_param)
1449{
1450 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301451 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001452 int status;
1453
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301454 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001455 return -EIO;
1456
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301457 vif->ssid_len = ibss_param->ssid_len;
1458 memcpy(vif->ssid, ibss_param->ssid, vif->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +03001459
1460 if (ibss_param->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301461 vif->ch_hint = ibss_param->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +03001462
1463 if (ibss_param->channel_fixed) {
1464 /*
1465 * TODO: channel_fixed: The channel should be fixed, do not
1466 * search for IBSSs to join on other channels. Target
1467 * firmware does not support this feature, needs to be
1468 * updated.
1469 */
1470 return -EOPNOTSUPP;
1471 }
1472
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301473 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001474 if (ibss_param->bssid && !is_broadcast_ether_addr(ibss_param->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301475 memcpy(vif->req_bssid, ibss_param->bssid,
1476 sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001477
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301478 ath6kl_set_wpa_version(vif, 0);
Kalle Valobdcd8172011-07-18 00:22:30 +03001479
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301480 status = ath6kl_set_auth_type(vif, NL80211_AUTHTYPE_OPEN_SYSTEM);
Kalle Valobdcd8172011-07-18 00:22:30 +03001481 if (status)
1482 return status;
1483
1484 if (ibss_param->privacy) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301485 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, true);
1486 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001487 } else {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301488 ath6kl_set_cipher(vif, 0, true);
1489 ath6kl_set_cipher(vif, 0, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001490 }
1491
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301492 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +03001493
1494 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1495 "%s: connect called with authmode %d dot11 auth %d"
1496 " PW crypto %d PW crypto len %d GRP crypto %d"
1497 " GRP crypto len %d channel hint %u\n",
1498 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301499 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
1500 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301501 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +03001502
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301503 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301504 vif->dot11_auth_mode, vif->auth_mode,
1505 vif->prwise_crypto,
1506 vif->prwise_crypto_len,
1507 vif->grp_crypto, vif->grp_crypto_len,
1508 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301509 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001510 ar->connect_ctrl_flags, SUBTYPE_NONE);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301511 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001512
1513 return 0;
1514}
1515
1516static int ath6kl_cfg80211_leave_ibss(struct wiphy *wiphy,
1517 struct net_device *dev)
1518{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301519 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001520
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301521 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001522 return -EIO;
1523
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301524 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301525 memset(vif->ssid, 0, sizeof(vif->ssid));
1526 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001527
1528 return 0;
1529}
1530
1531static const u32 cipher_suites[] = {
1532 WLAN_CIPHER_SUITE_WEP40,
1533 WLAN_CIPHER_SUITE_WEP104,
1534 WLAN_CIPHER_SUITE_TKIP,
1535 WLAN_CIPHER_SUITE_CCMP,
Jouni Malinen837cb972011-10-11 17:31:57 +03001536 CCKM_KRK_CIPHER_SUITE,
Dai Shuibing5e070212011-11-03 11:39:37 +02001537 WLAN_CIPHER_SUITE_SMS4,
Kalle Valobdcd8172011-07-18 00:22:30 +03001538};
1539
1540static bool is_rate_legacy(s32 rate)
1541{
1542 static const s32 legacy[] = { 1000, 2000, 5500, 11000,
1543 6000, 9000, 12000, 18000, 24000,
1544 36000, 48000, 54000
1545 };
1546 u8 i;
1547
1548 for (i = 0; i < ARRAY_SIZE(legacy); i++)
1549 if (rate == legacy[i])
1550 return true;
1551
1552 return false;
1553}
1554
1555static bool is_rate_ht20(s32 rate, u8 *mcs, bool *sgi)
1556{
1557 static const s32 ht20[] = { 6500, 13000, 19500, 26000, 39000,
1558 52000, 58500, 65000, 72200
1559 };
1560 u8 i;
1561
1562 for (i = 0; i < ARRAY_SIZE(ht20); i++) {
1563 if (rate == ht20[i]) {
1564 if (i == ARRAY_SIZE(ht20) - 1)
1565 /* last rate uses sgi */
1566 *sgi = true;
1567 else
1568 *sgi = false;
1569
1570 *mcs = i;
1571 return true;
1572 }
1573 }
1574 return false;
1575}
1576
1577static bool is_rate_ht40(s32 rate, u8 *mcs, bool *sgi)
1578{
1579 static const s32 ht40[] = { 13500, 27000, 40500, 54000,
1580 81000, 108000, 121500, 135000,
1581 150000
1582 };
1583 u8 i;
1584
1585 for (i = 0; i < ARRAY_SIZE(ht40); i++) {
1586 if (rate == ht40[i]) {
1587 if (i == ARRAY_SIZE(ht40) - 1)
1588 /* last rate uses sgi */
1589 *sgi = true;
1590 else
1591 *sgi = false;
1592
1593 *mcs = i;
1594 return true;
1595 }
1596 }
1597
1598 return false;
1599}
1600
1601static int ath6kl_get_station(struct wiphy *wiphy, struct net_device *dev,
1602 u8 *mac, struct station_info *sinfo)
1603{
1604 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301605 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001606 long left;
1607 bool sgi;
1608 s32 rate;
1609 int ret;
1610 u8 mcs;
1611
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301612 if (memcmp(mac, vif->bssid, ETH_ALEN) != 0)
Kalle Valobdcd8172011-07-18 00:22:30 +03001613 return -ENOENT;
1614
1615 if (down_interruptible(&ar->sem))
1616 return -EBUSY;
1617
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301618 set_bit(STATS_UPDATE_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001619
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301620 ret = ath6kl_wmi_get_stats_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +03001621
1622 if (ret != 0) {
1623 up(&ar->sem);
1624 return -EIO;
1625 }
1626
1627 left = wait_event_interruptible_timeout(ar->event_wq,
1628 !test_bit(STATS_UPDATE_PEND,
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301629 &vif->flags),
Kalle Valobdcd8172011-07-18 00:22:30 +03001630 WMI_TIMEOUT);
1631
1632 up(&ar->sem);
1633
1634 if (left == 0)
1635 return -ETIMEDOUT;
1636 else if (left < 0)
1637 return left;
1638
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301639 if (vif->target_stats.rx_byte) {
1640 sinfo->rx_bytes = vif->target_stats.rx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001641 sinfo->filled |= STATION_INFO_RX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301642 sinfo->rx_packets = vif->target_stats.rx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001643 sinfo->filled |= STATION_INFO_RX_PACKETS;
1644 }
1645
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301646 if (vif->target_stats.tx_byte) {
1647 sinfo->tx_bytes = vif->target_stats.tx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001648 sinfo->filled |= STATION_INFO_TX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301649 sinfo->tx_packets = vif->target_stats.tx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001650 sinfo->filled |= STATION_INFO_TX_PACKETS;
1651 }
1652
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301653 sinfo->signal = vif->target_stats.cs_rssi;
Kalle Valobdcd8172011-07-18 00:22:30 +03001654 sinfo->filled |= STATION_INFO_SIGNAL;
1655
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301656 rate = vif->target_stats.tx_ucast_rate;
Kalle Valobdcd8172011-07-18 00:22:30 +03001657
1658 if (is_rate_legacy(rate)) {
1659 sinfo->txrate.legacy = rate / 100;
1660 } else if (is_rate_ht20(rate, &mcs, &sgi)) {
1661 if (sgi) {
1662 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1663 sinfo->txrate.mcs = mcs - 1;
1664 } else {
1665 sinfo->txrate.mcs = mcs;
1666 }
1667
1668 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1669 } else if (is_rate_ht40(rate, &mcs, &sgi)) {
1670 if (sgi) {
1671 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1672 sinfo->txrate.mcs = mcs - 1;
1673 } else {
1674 sinfo->txrate.mcs = mcs;
1675 }
1676
1677 sinfo->txrate.flags |= RATE_INFO_FLAGS_40_MHZ_WIDTH;
1678 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1679 } else {
Kalle Valo9a730832011-09-27 23:33:28 +03001680 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1681 "invalid rate from stats: %d\n", rate);
1682 ath6kl_debug_war(ar, ATH6KL_WAR_INVALID_RATE);
Kalle Valobdcd8172011-07-18 00:22:30 +03001683 return 0;
1684 }
1685
1686 sinfo->filled |= STATION_INFO_TX_BITRATE;
1687
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301688 if (test_bit(CONNECTED, &vif->flags) &&
1689 test_bit(DTIM_PERIOD_AVAIL, &vif->flags) &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301690 vif->nw_type == INFRA_NETWORK) {
Jouni Malinen32c10872011-09-19 19:15:07 +03001691 sinfo->filled |= STATION_INFO_BSS_PARAM;
1692 sinfo->bss_param.flags = 0;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301693 sinfo->bss_param.dtim_period = vif->assoc_bss_dtim_period;
1694 sinfo->bss_param.beacon_interval = vif->assoc_bss_beacon_int;
Jouni Malinen32c10872011-09-19 19:15:07 +03001695 }
1696
Kalle Valobdcd8172011-07-18 00:22:30 +03001697 return 0;
1698}
1699
1700static int ath6kl_set_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1701 struct cfg80211_pmksa *pmksa)
1702{
1703 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301704 struct ath6kl_vif *vif = netdev_priv(netdev);
1705
1706 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001707 pmksa->pmkid, true);
1708}
1709
1710static int ath6kl_del_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1711 struct cfg80211_pmksa *pmksa)
1712{
1713 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301714 struct ath6kl_vif *vif = netdev_priv(netdev);
1715
1716 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001717 pmksa->pmkid, false);
1718}
1719
1720static int ath6kl_flush_pmksa(struct wiphy *wiphy, struct net_device *netdev)
1721{
1722 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301723 struct ath6kl_vif *vif = netdev_priv(netdev);
1724
1725 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301726 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx,
1727 vif->bssid, NULL, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001728 return 0;
1729}
1730
Raja Mani6cb3c712011-11-07 22:52:45 +02001731static int ath6kl_wow_suspend(struct ath6kl *ar, struct cfg80211_wowlan *wow)
1732{
Raja Manic08631c2011-12-16 14:24:24 +05301733 struct in_device *in_dev;
1734 struct in_ifaddr *ifa;
Raja Mani6cb3c712011-11-07 22:52:45 +02001735 struct ath6kl_vif *vif;
1736 int ret, pos, left;
1737 u32 filter = 0;
1738 u16 i;
Raja Manic08631c2011-12-16 14:24:24 +05301739 u8 mask[WOW_MASK_SIZE], index = 0;
1740 __be32 ips[MAX_IP_ADDRS];
Raja Mani6cb3c712011-11-07 22:52:45 +02001741
1742 vif = ath6kl_vif_first(ar);
1743 if (!vif)
1744 return -EIO;
1745
1746 if (!ath6kl_cfg80211_ready(vif))
1747 return -EIO;
1748
1749 if (!test_bit(CONNECTED, &vif->flags))
1750 return -EINVAL;
1751
1752 /* Clear existing WOW patterns */
1753 for (i = 0; i < WOW_MAX_FILTERS_PER_LIST; i++)
1754 ath6kl_wmi_del_wow_pattern_cmd(ar->wmi, vif->fw_vif_idx,
1755 WOW_LIST_ID, i);
1756 /* Configure new WOW patterns */
1757 for (i = 0; i < wow->n_patterns; i++) {
1758
1759 /*
1760 * Convert given nl80211 specific mask value to equivalent
1761 * driver specific mask value and send it to the chip along
1762 * with patterns. For example, If the mask value defined in
1763 * struct cfg80211_wowlan is 0xA (equivalent binary is 1010),
1764 * then equivalent driver specific mask value is
1765 * "0xFF 0x00 0xFF 0x00".
1766 */
1767 memset(&mask, 0, sizeof(mask));
1768 for (pos = 0; pos < wow->patterns[i].pattern_len; pos++) {
1769 if (wow->patterns[i].mask[pos / 8] & (0x1 << (pos % 8)))
1770 mask[pos] = 0xFF;
1771 }
1772 /*
1773 * Note: Pattern's offset is not passed as part of wowlan
1774 * parameter from CFG layer. So it's always passed as ZERO
1775 * to the firmware. It means, given WOW patterns are always
1776 * matched from the first byte of received pkt in the firmware.
1777 */
1778 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1779 vif->fw_vif_idx, WOW_LIST_ID,
1780 wow->patterns[i].pattern_len,
1781 0 /* pattern offset */,
1782 wow->patterns[i].pattern, mask);
1783 if (ret)
1784 return ret;
1785 }
1786
Raja Manic08631c2011-12-16 14:24:24 +05301787 /* Setup own IP addr for ARP agent. */
1788 in_dev = __in_dev_get_rtnl(vif->ndev);
1789 if (!in_dev)
1790 goto skip_arp;
1791
1792 ifa = in_dev->ifa_list;
1793 memset(&ips, 0, sizeof(ips));
1794
1795 /* Configure IP addr only if IP address count < MAX_IP_ADDRS */
1796 while (index < MAX_IP_ADDRS && ifa) {
1797 ips[index] = ifa->ifa_local;
1798 ifa = ifa->ifa_next;
1799 index++;
1800 }
1801
1802 if (ifa) {
1803 ath6kl_err("total IP addr count is exceeding fw limit\n");
1804 return -EINVAL;
1805 }
1806
1807 ret = ath6kl_wmi_set_ip_cmd(ar->wmi, vif->fw_vif_idx, ips[0], ips[1]);
1808 if (ret) {
1809 ath6kl_err("fail to setup ip for arp agent\n");
1810 return ret;
1811 }
1812
1813skip_arp:
Raja Mani6cb3c712011-11-07 22:52:45 +02001814 if (wow->disconnect)
1815 filter |= WOW_FILTER_OPTION_NWK_DISASSOC;
1816
1817 if (wow->magic_pkt)
1818 filter |= WOW_FILTER_OPTION_MAGIC_PACKET;
1819
1820 if (wow->gtk_rekey_failure)
1821 filter |= WOW_FILTER_OPTION_GTK_ERROR;
1822
1823 if (wow->eap_identity_req)
1824 filter |= WOW_FILTER_OPTION_EAP_REQ;
1825
1826 if (wow->four_way_handshake)
1827 filter |= WOW_FILTER_OPTION_8021X_4WAYHS;
1828
1829 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
1830 ATH6KL_WOW_MODE_ENABLE,
1831 filter,
1832 WOW_HOST_REQ_DELAY);
1833 if (ret)
1834 return ret;
1835
1836 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
1837 ATH6KL_HOST_MODE_ASLEEP);
1838 if (ret)
1839 return ret;
1840
1841 if (ar->tx_pending[ar->ctrl_ep]) {
1842 left = wait_event_interruptible_timeout(ar->event_wq,
1843 ar->tx_pending[ar->ctrl_ep] == 0, WMI_TIMEOUT);
1844 if (left == 0) {
1845 ath6kl_warn("clear wmi ctrl data timeout\n");
1846 ret = -ETIMEDOUT;
1847 } else if (left < 0) {
1848 ath6kl_warn("clear wmi ctrl data failed: %d\n", left);
1849 ret = left;
1850 }
1851 }
1852
1853 return ret;
1854}
1855
1856static int ath6kl_wow_resume(struct ath6kl *ar)
1857{
1858 struct ath6kl_vif *vif;
1859 int ret;
1860
1861 vif = ath6kl_vif_first(ar);
1862 if (!vif)
1863 return -EIO;
1864
1865 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
1866 ATH6KL_HOST_MODE_AWAKE);
1867 return ret;
1868}
1869
Kalle Valo52d81a62011-11-01 08:44:21 +02001870int ath6kl_cfg80211_suspend(struct ath6kl *ar,
Raja Mani0f60e9f2011-11-07 22:52:45 +02001871 enum ath6kl_cfg_suspend_mode mode,
1872 struct cfg80211_wowlan *wow)
Kalle Valo52d81a62011-11-01 08:44:21 +02001873{
1874 int ret;
1875
Kalle Valo52d81a62011-11-01 08:44:21 +02001876 switch (mode) {
Raja Manid7c44e02011-11-07 22:52:46 +02001877 case ATH6KL_CFG_SUSPEND_WOW:
1878
1879 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode suspend\n");
1880
1881 /* Flush all non control pkts in TX path */
1882 ath6kl_tx_data_cleanup(ar);
1883
1884 ret = ath6kl_wow_suspend(ar, wow);
1885 if (ret) {
1886 ath6kl_err("wow suspend failed: %d\n", ret);
1887 return ret;
1888 }
1889 ar->state = ATH6KL_STATE_WOW;
1890 break;
1891
Kalle Valo52d81a62011-11-01 08:44:21 +02001892 case ATH6KL_CFG_SUSPEND_DEEPSLEEP:
Raja Mani524441e2011-11-07 22:52:46 +02001893
Kalle Valo7125f012011-12-13 14:51:37 +02001894 ath6kl_cfg80211_stop_all(ar);
Raja Mani524441e2011-11-07 22:52:46 +02001895
Kalle Valo52d81a62011-11-01 08:44:21 +02001896 /* save the current power mode before enabling power save */
1897 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
1898
1899 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER);
1900 if (ret) {
1901 ath6kl_warn("wmi powermode command failed during suspend: %d\n",
1902 ret);
1903 }
1904
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001905 ar->state = ATH6KL_STATE_DEEPSLEEP;
1906
Kalle Valo52d81a62011-11-01 08:44:21 +02001907 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001908
1909 case ATH6KL_CFG_SUSPEND_CUTPOWER:
Raja Mani524441e2011-11-07 22:52:46 +02001910
Kalle Valo7125f012011-12-13 14:51:37 +02001911 ath6kl_cfg80211_stop_all(ar);
Raja Mani524441e2011-11-07 22:52:46 +02001912
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001913 if (ar->state == ATH6KL_STATE_OFF) {
1914 ath6kl_dbg(ATH6KL_DBG_SUSPEND,
1915 "suspend hw off, no action for cutpower\n");
1916 break;
1917 }
1918
1919 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "suspend cutting power\n");
1920
1921 ret = ath6kl_init_hw_stop(ar);
1922 if (ret) {
1923 ath6kl_warn("failed to stop hw during suspend: %d\n",
1924 ret);
1925 }
1926
1927 ar->state = ATH6KL_STATE_CUTPOWER;
1928
1929 break;
1930
Kalle Valo10509f92011-12-13 14:52:07 +02001931 case ATH6KL_CFG_SUSPEND_SCHED_SCAN:
1932 /*
1933 * Nothing needed for schedule scan, firmware is already in
1934 * wow mode and sleeping most of the time.
1935 */
1936 break;
1937
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001938 default:
1939 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02001940 }
1941
1942 return 0;
1943}
1944
1945int ath6kl_cfg80211_resume(struct ath6kl *ar)
1946{
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001947 int ret;
1948
1949 switch (ar->state) {
Raja Manid7c44e02011-11-07 22:52:46 +02001950 case ATH6KL_STATE_WOW:
1951 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode resume\n");
1952
1953 ret = ath6kl_wow_resume(ar);
1954 if (ret) {
1955 ath6kl_warn("wow mode resume failed: %d\n", ret);
1956 return ret;
1957 }
1958
1959 ar->state = ATH6KL_STATE_ON;
1960 break;
1961
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001962 case ATH6KL_STATE_DEEPSLEEP:
1963 if (ar->wmi->pwr_mode != ar->wmi->saved_pwr_mode) {
1964 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0,
1965 ar->wmi->saved_pwr_mode);
1966 if (ret) {
1967 ath6kl_warn("wmi powermode command failed during resume: %d\n",
1968 ret);
1969 }
1970 }
1971
1972 ar->state = ATH6KL_STATE_ON;
1973
1974 break;
1975
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001976 case ATH6KL_STATE_CUTPOWER:
1977 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "resume restoring power\n");
1978
1979 ret = ath6kl_init_hw_start(ar);
1980 if (ret) {
1981 ath6kl_warn("Failed to boot hw in resume: %d\n", ret);
1982 return ret;
1983 }
Raja Manid7c44e02011-11-07 22:52:46 +02001984 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001985
Kalle Valo10509f92011-12-13 14:52:07 +02001986 case ATH6KL_STATE_SCHED_SCAN:
1987 break;
1988
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001989 default:
1990 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02001991 }
1992
1993 return 0;
1994}
1995
Kalle Valoabcb3442011-07-22 08:26:20 +03001996#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02001997
1998/* hif layer decides what suspend mode to use */
1999static int __ath6kl_cfg80211_suspend(struct wiphy *wiphy,
Kalle Valoabcb3442011-07-22 08:26:20 +03002000 struct cfg80211_wowlan *wow)
2001{
2002 struct ath6kl *ar = wiphy_priv(wiphy);
2003
Raja Mani0f60e9f2011-11-07 22:52:45 +02002004 return ath6kl_hif_suspend(ar, wow);
Kalle Valoabcb3442011-07-22 08:26:20 +03002005}
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002006
Kalle Valo52d81a62011-11-01 08:44:21 +02002007static int __ath6kl_cfg80211_resume(struct wiphy *wiphy)
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002008{
2009 struct ath6kl *ar = wiphy_priv(wiphy);
2010
2011 return ath6kl_hif_resume(ar);
2012}
Raja Mania918fb32011-11-07 22:52:46 +02002013
2014/*
2015 * FIXME: WOW suspend mode is selected if the host sdio controller supports
2016 * both sdio irq wake up and keep power. The target pulls sdio data line to
2017 * wake up the host when WOW pattern matches. This causes sdio irq handler
2018 * is being called in the host side which internally hits ath6kl's RX path.
2019 *
2020 * Since sdio interrupt is not disabled, RX path executes even before
2021 * the host executes the actual resume operation from PM module.
2022 *
2023 * In the current scenario, WOW resume should happen before start processing
2024 * any data from the target. So It's required to perform WOW resume in RX path.
2025 * Ideally we should perform WOW resume only in the actual platform
2026 * resume path. This area needs bit rework to avoid WOW resume in RX path.
2027 *
2028 * ath6kl_check_wow_status() is called from ath6kl_rx().
2029 */
2030void ath6kl_check_wow_status(struct ath6kl *ar)
2031{
2032 if (ar->state == ATH6KL_STATE_WOW)
2033 ath6kl_cfg80211_resume(ar);
2034}
2035
2036#else
2037
2038void ath6kl_check_wow_status(struct ath6kl *ar)
2039{
2040}
Kalle Valoabcb3442011-07-22 08:26:20 +03002041#endif
2042
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002043static int ath6kl_set_channel(struct wiphy *wiphy, struct net_device *dev,
2044 struct ieee80211_channel *chan,
2045 enum nl80211_channel_type channel_type)
2046{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302047 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002048
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302049 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002050 return -EIO;
2051
2052 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: center_freq=%u hw_value=%u\n",
2053 __func__, chan->center_freq, chan->hw_value);
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302054 vif->next_chan = chan->center_freq;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002055
2056 return 0;
2057}
2058
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002059static bool ath6kl_is_p2p_ie(const u8 *pos)
2060{
2061 return pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
2062 pos[2] == 0x50 && pos[3] == 0x6f &&
2063 pos[4] == 0x9a && pos[5] == 0x09;
2064}
2065
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302066static int ath6kl_set_ap_probe_resp_ies(struct ath6kl_vif *vif,
2067 const u8 *ies, size_t ies_len)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002068{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302069 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002070 const u8 *pos;
2071 u8 *buf = NULL;
2072 size_t len = 0;
2073 int ret;
2074
2075 /*
2076 * Filter out P2P IE(s) since they will be included depending on
2077 * the Probe Request frame in ath6kl_send_go_probe_resp().
2078 */
2079
2080 if (ies && ies_len) {
2081 buf = kmalloc(ies_len, GFP_KERNEL);
2082 if (buf == NULL)
2083 return -ENOMEM;
2084 pos = ies;
2085 while (pos + 1 < ies + ies_len) {
2086 if (pos + 2 + pos[1] > ies + ies_len)
2087 break;
2088 if (!ath6kl_is_p2p_ie(pos)) {
2089 memcpy(buf + len, pos, 2 + pos[1]);
2090 len += 2 + pos[1];
2091 }
2092 pos += 2 + pos[1];
2093 }
2094 }
2095
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302096 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2097 WMI_FRAME_PROBE_RESP, buf, len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002098 kfree(buf);
2099 return ret;
2100}
2101
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002102static int ath6kl_ap_beacon(struct wiphy *wiphy, struct net_device *dev,
2103 struct beacon_parameters *info, bool add)
2104{
2105 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302106 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002107 struct ieee80211_mgmt *mgmt;
2108 u8 *ies;
2109 int ies_len;
2110 struct wmi_connect_cmd p;
2111 int res;
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302112 int i, ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002113
2114 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: add=%d\n", __func__, add);
2115
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302116 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002117 return -EIO;
2118
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302119 if (vif->next_mode != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002120 return -EOPNOTSUPP;
2121
2122 if (info->beacon_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302123 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2124 WMI_FRAME_BEACON,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002125 info->beacon_ies,
2126 info->beacon_ies_len);
2127 if (res)
2128 return res;
2129 }
2130 if (info->proberesp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302131 res = ath6kl_set_ap_probe_resp_ies(vif, info->proberesp_ies,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002132 info->proberesp_ies_len);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002133 if (res)
2134 return res;
2135 }
2136 if (info->assocresp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302137 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2138 WMI_FRAME_ASSOC_RESP,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002139 info->assocresp_ies,
2140 info->assocresp_ies_len);
2141 if (res)
2142 return res;
2143 }
2144
2145 if (!add)
2146 return 0;
2147
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002148 ar->ap_mode_bkey.valid = false;
2149
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002150 /* TODO:
2151 * info->interval
2152 * info->dtim_period
2153 */
2154
2155 if (info->head == NULL)
2156 return -EINVAL;
2157 mgmt = (struct ieee80211_mgmt *) info->head;
2158 ies = mgmt->u.beacon.variable;
2159 if (ies > info->head + info->head_len)
2160 return -EINVAL;
2161 ies_len = info->head + info->head_len - ies;
2162
2163 if (info->ssid == NULL)
2164 return -EINVAL;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302165 memcpy(vif->ssid, info->ssid, info->ssid_len);
2166 vif->ssid_len = info->ssid_len;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002167 if (info->hidden_ssid != NL80211_HIDDEN_SSID_NOT_IN_USE)
2168 return -EOPNOTSUPP; /* TODO */
2169
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302170 ret = ath6kl_set_auth_type(vif, info->auth_type);
2171 if (ret)
2172 return ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002173
2174 memset(&p, 0, sizeof(p));
2175
2176 for (i = 0; i < info->crypto.n_akm_suites; i++) {
2177 switch (info->crypto.akm_suites[i]) {
2178 case WLAN_AKM_SUITE_8021X:
2179 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2180 p.auth_mode |= WPA_AUTH;
2181 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2182 p.auth_mode |= WPA2_AUTH;
2183 break;
2184 case WLAN_AKM_SUITE_PSK:
2185 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2186 p.auth_mode |= WPA_PSK_AUTH;
2187 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2188 p.auth_mode |= WPA2_PSK_AUTH;
2189 break;
2190 }
2191 }
2192 if (p.auth_mode == 0)
2193 p.auth_mode = NONE_AUTH;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302194 vif->auth_mode = p.auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002195
2196 for (i = 0; i < info->crypto.n_ciphers_pairwise; i++) {
2197 switch (info->crypto.ciphers_pairwise[i]) {
2198 case WLAN_CIPHER_SUITE_WEP40:
2199 case WLAN_CIPHER_SUITE_WEP104:
2200 p.prwise_crypto_type |= WEP_CRYPT;
2201 break;
2202 case WLAN_CIPHER_SUITE_TKIP:
2203 p.prwise_crypto_type |= TKIP_CRYPT;
2204 break;
2205 case WLAN_CIPHER_SUITE_CCMP:
2206 p.prwise_crypto_type |= AES_CRYPT;
2207 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002208 case WLAN_CIPHER_SUITE_SMS4:
2209 p.prwise_crypto_type |= WAPI_CRYPT;
2210 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002211 }
2212 }
Edward Lu229ed6b2011-08-30 21:58:07 +03002213 if (p.prwise_crypto_type == 0) {
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002214 p.prwise_crypto_type = NONE_CRYPT;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302215 ath6kl_set_cipher(vif, 0, true);
Edward Lu229ed6b2011-08-30 21:58:07 +03002216 } else if (info->crypto.n_ciphers_pairwise == 1)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302217 ath6kl_set_cipher(vif, info->crypto.ciphers_pairwise[0], true);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002218
2219 switch (info->crypto.cipher_group) {
2220 case WLAN_CIPHER_SUITE_WEP40:
2221 case WLAN_CIPHER_SUITE_WEP104:
2222 p.grp_crypto_type = WEP_CRYPT;
2223 break;
2224 case WLAN_CIPHER_SUITE_TKIP:
2225 p.grp_crypto_type = TKIP_CRYPT;
2226 break;
2227 case WLAN_CIPHER_SUITE_CCMP:
2228 p.grp_crypto_type = AES_CRYPT;
2229 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002230 case WLAN_CIPHER_SUITE_SMS4:
2231 p.grp_crypto_type = WAPI_CRYPT;
2232 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002233 default:
2234 p.grp_crypto_type = NONE_CRYPT;
2235 break;
2236 }
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302237 ath6kl_set_cipher(vif, info->crypto.cipher_group, false);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002238
2239 p.nw_type = AP_NETWORK;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302240 vif->nw_type = vif->next_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002241
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302242 p.ssid_len = vif->ssid_len;
2243 memcpy(p.ssid, vif->ssid, vif->ssid_len);
2244 p.dot11_auth_mode = vif->dot11_auth_mode;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302245 p.ch = cpu_to_le16(vif->next_chan);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002246
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002247 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_GO) {
2248 p.nw_subtype = SUBTYPE_P2PGO;
2249 } else {
2250 /*
2251 * Due to firmware limitation, it is not possible to
2252 * do P2P mgmt operations in AP mode
2253 */
2254 p.nw_subtype = SUBTYPE_NONE;
2255 }
2256
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302257 res = ath6kl_wmi_ap_profile_commit(ar->wmi, vif->fw_vif_idx, &p);
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002258 if (res < 0)
2259 return res;
2260
2261 return 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002262}
2263
2264static int ath6kl_add_beacon(struct wiphy *wiphy, struct net_device *dev,
2265 struct beacon_parameters *info)
2266{
2267 return ath6kl_ap_beacon(wiphy, dev, info, true);
2268}
2269
2270static int ath6kl_set_beacon(struct wiphy *wiphy, struct net_device *dev,
2271 struct beacon_parameters *info)
2272{
2273 return ath6kl_ap_beacon(wiphy, dev, info, false);
2274}
2275
2276static int ath6kl_del_beacon(struct wiphy *wiphy, struct net_device *dev)
2277{
2278 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302279 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002280
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302281 if (vif->nw_type != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002282 return -EOPNOTSUPP;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302283 if (!test_bit(CONNECTED, &vif->flags))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002284 return -ENOTCONN;
2285
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302286 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302287 clear_bit(CONNECTED, &vif->flags);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002288
2289 return 0;
2290}
2291
Jouni Malinen23875132011-08-30 21:57:53 +03002292static int ath6kl_change_station(struct wiphy *wiphy, struct net_device *dev,
2293 u8 *mac, struct station_parameters *params)
2294{
2295 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302296 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen23875132011-08-30 21:57:53 +03002297
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302298 if (vif->nw_type != AP_NETWORK)
Jouni Malinen23875132011-08-30 21:57:53 +03002299 return -EOPNOTSUPP;
2300
2301 /* Use this only for authorizing/unauthorizing a station */
2302 if (!(params->sta_flags_mask & BIT(NL80211_STA_FLAG_AUTHORIZED)))
2303 return -EOPNOTSUPP;
2304
2305 if (params->sta_flags_set & BIT(NL80211_STA_FLAG_AUTHORIZED))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302306 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2307 WMI_AP_MLME_AUTHORIZE, mac, 0);
2308 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2309 WMI_AP_MLME_UNAUTHORIZE, mac, 0);
Jouni Malinen23875132011-08-30 21:57:53 +03002310}
2311
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002312static int ath6kl_remain_on_channel(struct wiphy *wiphy,
2313 struct net_device *dev,
2314 struct ieee80211_channel *chan,
2315 enum nl80211_channel_type channel_type,
2316 unsigned int duration,
2317 u64 *cookie)
2318{
2319 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302320 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen10522612011-10-27 16:00:13 +03002321 u32 id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002322
2323 /* TODO: if already pending or ongoing remain-on-channel,
2324 * return -EBUSY */
Jouni Malinen10522612011-10-27 16:00:13 +03002325 id = ++vif->last_roc_id;
2326 if (id == 0) {
2327 /* Do not use 0 as the cookie value */
2328 id = ++vif->last_roc_id;
2329 }
2330 *cookie = id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002331
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302332 return ath6kl_wmi_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx,
2333 chan->center_freq, duration);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002334}
2335
2336static int ath6kl_cancel_remain_on_channel(struct wiphy *wiphy,
2337 struct net_device *dev,
2338 u64 cookie)
2339{
2340 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302341 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002342
Jouni Malinen10522612011-10-27 16:00:13 +03002343 if (cookie != vif->last_roc_id)
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002344 return -ENOENT;
Jouni Malinen10522612011-10-27 16:00:13 +03002345 vif->last_cancel_roc_id = cookie;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002346
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302347 return ath6kl_wmi_cancel_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002348}
2349
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302350static int ath6kl_send_go_probe_resp(struct ath6kl_vif *vif,
2351 const u8 *buf, size_t len,
2352 unsigned int freq)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002353{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302354 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002355 const u8 *pos;
2356 u8 *p2p;
2357 int p2p_len;
2358 int ret;
2359 const struct ieee80211_mgmt *mgmt;
2360
2361 mgmt = (const struct ieee80211_mgmt *) buf;
2362
2363 /* Include P2P IE(s) from the frame generated in user space. */
2364
2365 p2p = kmalloc(len, GFP_KERNEL);
2366 if (p2p == NULL)
2367 return -ENOMEM;
2368 p2p_len = 0;
2369
2370 pos = mgmt->u.probe_resp.variable;
2371 while (pos + 1 < buf + len) {
2372 if (pos + 2 + pos[1] > buf + len)
2373 break;
2374 if (ath6kl_is_p2p_ie(pos)) {
2375 memcpy(p2p + p2p_len, pos, 2 + pos[1]);
2376 p2p_len += 2 + pos[1];
2377 }
2378 pos += 2 + pos[1];
2379 }
2380
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302381 ret = ath6kl_wmi_send_probe_response_cmd(ar->wmi, vif->fw_vif_idx, freq,
2382 mgmt->da, p2p, p2p_len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002383 kfree(p2p);
2384 return ret;
2385}
2386
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002387static int ath6kl_mgmt_tx(struct wiphy *wiphy, struct net_device *dev,
2388 struct ieee80211_channel *chan, bool offchan,
2389 enum nl80211_channel_type channel_type,
2390 bool channel_type_valid, unsigned int wait,
Johannes Berge247bd902011-11-04 11:18:21 +01002391 const u8 *buf, size_t len, bool no_cck,
2392 bool dont_wait_for_ack, u64 *cookie)
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002393{
2394 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302395 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002396 u32 id;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002397 const struct ieee80211_mgmt *mgmt;
2398
2399 mgmt = (const struct ieee80211_mgmt *) buf;
2400 if (buf + len >= mgmt->u.probe_resp.variable &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302401 vif->nw_type == AP_NETWORK && test_bit(CONNECTED, &vif->flags) &&
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002402 ieee80211_is_probe_resp(mgmt->frame_control)) {
2403 /*
2404 * Send Probe Response frame in AP mode using a separate WMI
2405 * command to allow the target to fill in the generic IEs.
2406 */
2407 *cookie = 0; /* TX status not supported */
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302408 return ath6kl_send_go_probe_resp(vif, buf, len,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002409 chan->center_freq);
2410 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002411
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302412 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002413 if (id == 0) {
2414 /*
2415 * 0 is a reserved value in the WMI command and shall not be
2416 * used for the command.
2417 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302418 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002419 }
2420
2421 *cookie = id;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002422
2423 if (test_bit(ATH6KL_FW_CAPABILITY_STA_P2PDEV_DUPLEX,
2424 ar->fw_capabilities)) {
2425 /*
2426 * If capable of doing P2P mgmt operations using
2427 * station interface, send additional information like
2428 * supported rates to advertise and xmit rates for
2429 * probe requests
2430 */
2431 return ath6kl_wmi_send_mgmt_cmd(ar->wmi, vif->fw_vif_idx, id,
2432 chan->center_freq, wait,
2433 buf, len, no_cck);
2434 } else {
2435 return ath6kl_wmi_send_action_cmd(ar->wmi, vif->fw_vif_idx, id,
2436 chan->center_freq, wait,
2437 buf, len);
2438 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002439}
2440
Jouni Malinenae32c302011-08-30 21:58:01 +03002441static void ath6kl_mgmt_frame_register(struct wiphy *wiphy,
2442 struct net_device *dev,
2443 u16 frame_type, bool reg)
2444{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302445 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinenae32c302011-08-30 21:58:01 +03002446
2447 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: frame_type=0x%x reg=%d\n",
2448 __func__, frame_type, reg);
2449 if (frame_type == IEEE80211_STYPE_PROBE_REQ) {
2450 /*
2451 * Note: This notification callback is not allowed to sleep, so
2452 * we cannot send WMI_PROBE_REQ_REPORT_CMD here. Instead, we
2453 * hardcode target to report Probe Request frames all the time.
2454 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302455 vif->probe_req_report = reg;
Jouni Malinenae32c302011-08-30 21:58:01 +03002456 }
2457}
2458
Kalle Valo10509f92011-12-13 14:52:07 +02002459static int ath6kl_cfg80211_sscan_start(struct wiphy *wiphy,
2460 struct net_device *dev,
2461 struct cfg80211_sched_scan_request *request)
2462{
2463 struct ath6kl *ar = ath6kl_priv(dev);
2464 struct ath6kl_vif *vif = netdev_priv(dev);
2465 u16 interval;
2466 int ret;
2467 u8 i;
2468
2469 if (ar->state != ATH6KL_STATE_ON)
2470 return -EIO;
2471
2472 if (vif->sme_state != SME_DISCONNECTED)
2473 return -EBUSY;
2474
2475 for (i = 0; i < ar->wiphy->max_sched_scan_ssids; i++) {
2476 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
2477 i, DISABLE_SSID_FLAG,
2478 0, NULL);
2479 }
2480
2481 /* fw uses seconds, also make sure that it's >0 */
2482 interval = max_t(u16, 1, request->interval / 1000);
2483
2484 ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
2485 interval, interval,
2486 10, 0, 0, 0, 3, 0, 0, 0);
2487
2488 if (request->n_ssids && request->ssids[0].ssid_len) {
2489 for (i = 0; i < request->n_ssids; i++) {
2490 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
2491 i, SPECIFIC_SSID_FLAG,
2492 request->ssids[i].ssid_len,
2493 request->ssids[i].ssid);
2494 }
2495 }
2496
2497 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
2498 ATH6KL_WOW_MODE_ENABLE,
2499 WOW_FILTER_SSID,
2500 WOW_HOST_REQ_DELAY);
2501 if (ret) {
2502 ath6kl_warn("Failed to enable wow with ssid filter: %d\n", ret);
2503 return ret;
2504 }
2505
2506 /* this also clears IE in fw if it's not set */
2507 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2508 WMI_FRAME_PROBE_REQ,
2509 request->ie, request->ie_len);
2510 if (ret) {
2511 ath6kl_warn("Failed to set probe request IE for scheduled scan: %d",
2512 ret);
2513 return ret;
2514 }
2515
2516 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2517 ATH6KL_HOST_MODE_ASLEEP);
2518 if (ret) {
2519 ath6kl_warn("Failed to enable host sleep mode for sched scan: %d\n",
2520 ret);
2521 return ret;
2522 }
2523
2524 ar->state = ATH6KL_STATE_SCHED_SCAN;
2525
2526 return ret;
2527}
2528
2529static int ath6kl_cfg80211_sscan_stop(struct wiphy *wiphy,
2530 struct net_device *dev)
2531{
2532 struct ath6kl_vif *vif = netdev_priv(dev);
2533 bool stopped;
2534
2535 stopped = __ath6kl_cfg80211_sscan_stop(vif);
2536
2537 if (!stopped)
2538 return -EIO;
2539
2540 return 0;
2541}
2542
Jouni Malinenf80574a2011-08-30 21:58:04 +03002543static const struct ieee80211_txrx_stypes
2544ath6kl_mgmt_stypes[NUM_NL80211_IFTYPES] = {
2545 [NL80211_IFTYPE_STATION] = {
2546 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2547 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2548 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2549 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2550 },
2551 [NL80211_IFTYPE_P2P_CLIENT] = {
2552 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2553 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2554 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2555 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2556 },
2557 [NL80211_IFTYPE_P2P_GO] = {
2558 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2559 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2560 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2561 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2562 },
2563};
2564
Kalle Valobdcd8172011-07-18 00:22:30 +03002565static struct cfg80211_ops ath6kl_cfg80211_ops = {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302566 .add_virtual_intf = ath6kl_cfg80211_add_iface,
2567 .del_virtual_intf = ath6kl_cfg80211_del_iface,
Kalle Valobdcd8172011-07-18 00:22:30 +03002568 .change_virtual_intf = ath6kl_cfg80211_change_iface,
2569 .scan = ath6kl_cfg80211_scan,
2570 .connect = ath6kl_cfg80211_connect,
2571 .disconnect = ath6kl_cfg80211_disconnect,
2572 .add_key = ath6kl_cfg80211_add_key,
2573 .get_key = ath6kl_cfg80211_get_key,
2574 .del_key = ath6kl_cfg80211_del_key,
2575 .set_default_key = ath6kl_cfg80211_set_default_key,
2576 .set_wiphy_params = ath6kl_cfg80211_set_wiphy_params,
2577 .set_tx_power = ath6kl_cfg80211_set_txpower,
2578 .get_tx_power = ath6kl_cfg80211_get_txpower,
2579 .set_power_mgmt = ath6kl_cfg80211_set_power_mgmt,
2580 .join_ibss = ath6kl_cfg80211_join_ibss,
2581 .leave_ibss = ath6kl_cfg80211_leave_ibss,
2582 .get_station = ath6kl_get_station,
2583 .set_pmksa = ath6kl_set_pmksa,
2584 .del_pmksa = ath6kl_del_pmksa,
2585 .flush_pmksa = ath6kl_flush_pmksa,
Kalle Valo003353b0d2011-09-01 10:14:21 +03002586 CFG80211_TESTMODE_CMD(ath6kl_tm_cmd)
Kalle Valoabcb3442011-07-22 08:26:20 +03002587#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02002588 .suspend = __ath6kl_cfg80211_suspend,
2589 .resume = __ath6kl_cfg80211_resume,
Kalle Valoabcb3442011-07-22 08:26:20 +03002590#endif
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002591 .set_channel = ath6kl_set_channel,
2592 .add_beacon = ath6kl_add_beacon,
2593 .set_beacon = ath6kl_set_beacon,
2594 .del_beacon = ath6kl_del_beacon,
Jouni Malinen23875132011-08-30 21:57:53 +03002595 .change_station = ath6kl_change_station,
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002596 .remain_on_channel = ath6kl_remain_on_channel,
2597 .cancel_remain_on_channel = ath6kl_cancel_remain_on_channel,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002598 .mgmt_tx = ath6kl_mgmt_tx,
Jouni Malinenae32c302011-08-30 21:58:01 +03002599 .mgmt_frame_register = ath6kl_mgmt_frame_register,
Kalle Valo10509f92011-12-13 14:52:07 +02002600 .sched_scan_start = ath6kl_cfg80211_sscan_start,
2601 .sched_scan_stop = ath6kl_cfg80211_sscan_stop,
Kalle Valobdcd8172011-07-18 00:22:30 +03002602};
2603
Kalle Valo7125f012011-12-13 14:51:37 +02002604void ath6kl_cfg80211_stop(struct ath6kl_vif *vif)
Kalle Valoec4b7f62011-11-01 08:44:04 +02002605{
Kalle Valo10509f92011-12-13 14:52:07 +02002606 ath6kl_cfg80211_sscan_disable(vif);
2607
Kalle Valoec4b7f62011-11-01 08:44:04 +02002608 switch (vif->sme_state) {
Kalle Valoc97a31b2011-12-13 14:51:10 +02002609 case SME_DISCONNECTED:
2610 break;
Kalle Valoec4b7f62011-11-01 08:44:04 +02002611 case SME_CONNECTING:
2612 cfg80211_connect_result(vif->ndev, vif->bssid, NULL, 0,
2613 NULL, 0,
2614 WLAN_STATUS_UNSPECIFIED_FAILURE,
2615 GFP_KERNEL);
2616 break;
2617 case SME_CONNECTED:
Kalle Valoec4b7f62011-11-01 08:44:04 +02002618 cfg80211_disconnected(vif->ndev, 0, NULL, 0, GFP_KERNEL);
2619 break;
2620 }
2621
2622 if (test_bit(CONNECTED, &vif->flags) ||
2623 test_bit(CONNECT_PEND, &vif->flags))
Kalle Valo7125f012011-12-13 14:51:37 +02002624 ath6kl_wmi_disconnect_cmd(vif->ar->wmi, vif->fw_vif_idx);
Kalle Valoec4b7f62011-11-01 08:44:04 +02002625
2626 vif->sme_state = SME_DISCONNECTED;
Kalle Valo1f40525512011-11-01 08:44:13 +02002627 clear_bit(CONNECTED, &vif->flags);
2628 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valoec4b7f62011-11-01 08:44:04 +02002629
2630 /* disable scanning */
Kalle Valo7125f012011-12-13 14:51:37 +02002631 if (ath6kl_wmi_scanparams_cmd(vif->ar->wmi, vif->fw_vif_idx, 0xFFFF,
2632 0, 0, 0, 0, 0, 0, 0, 0, 0) != 0)
2633 ath6kl_warn("failed to disable scan during stop\n");
Kalle Valoec4b7f62011-11-01 08:44:04 +02002634
2635 ath6kl_cfg80211_scan_complete_event(vif, true);
2636}
2637
Kalle Valo7125f012011-12-13 14:51:37 +02002638void ath6kl_cfg80211_stop_all(struct ath6kl *ar)
2639{
2640 struct ath6kl_vif *vif;
2641
2642 vif = ath6kl_vif_first(ar);
2643 if (!vif) {
2644 /* save the current power mode before enabling power save */
2645 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
2646
2647 if (ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER) != 0)
2648 ath6kl_warn("ath6kl_deep_sleep_enable: "
2649 "wmi_powermode_cmd failed\n");
2650 return;
2651 }
2652
2653 /*
2654 * FIXME: we should take ar->list_lock to protect changes in the
2655 * vif_list, but that's not trivial to do as ath6kl_cfg80211_stop()
2656 * sleeps.
2657 */
2658 list_for_each_entry(vif, &ar->vif_list, list)
2659 ath6kl_cfg80211_stop(vif);
2660}
2661
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302662struct ath6kl *ath6kl_core_alloc(struct device *dev)
Kalle Valobdcd8172011-07-18 00:22:30 +03002663{
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002664 struct ath6kl *ar;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302665 struct wiphy *wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302666 u8 ctr;
Kalle Valobdcd8172011-07-18 00:22:30 +03002667
2668 /* create a new wiphy for use with cfg80211 */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302669 wiphy = wiphy_new(&ath6kl_cfg80211_ops, sizeof(struct ath6kl));
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302670
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302671 if (!wiphy) {
Kalle Valobdcd8172011-07-18 00:22:30 +03002672 ath6kl_err("couldn't allocate wiphy device\n");
Kalle Valobdcd8172011-07-18 00:22:30 +03002673 return NULL;
2674 }
2675
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302676 ar = wiphy_priv(wiphy);
Vasanthakumar Thiagarajan774439a2011-11-18 10:05:26 +05302677 ar->p2p = !!ath6kl_p2p;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302678 ar->wiphy = wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302679 ar->dev = dev;
2680
Kalle Valo71f96ee2011-11-14 19:31:30 +02002681 ar->vif_max = 1;
2682
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +05302683 ar->max_norm_iface = 1;
2684
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302685 spin_lock_init(&ar->lock);
2686 spin_lock_init(&ar->mcastpsq_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302687 spin_lock_init(&ar->list_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302688
2689 init_waitqueue_head(&ar->event_wq);
2690 sema_init(&ar->sem, 1);
2691
2692 INIT_LIST_HEAD(&ar->amsdu_rx_buffer_queue);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302693 INIT_LIST_HEAD(&ar->vif_list);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302694
2695 clear_bit(WMI_ENABLED, &ar->flag);
2696 clear_bit(SKIP_SCAN, &ar->flag);
2697 clear_bit(DESTROY_IN_PROGRESS, &ar->flag);
2698
2699 ar->listen_intvl_t = A_DEFAULT_LISTEN_INTERVAL;
2700 ar->listen_intvl_b = 0;
2701 ar->tx_pwr = 0;
2702
2703 ar->intra_bss = 1;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302704 ar->lrssi_roam_threshold = DEF_LRSSI_ROAM_THRESHOLD;
2705
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002706 ar->state = ATH6KL_STATE_OFF;
2707
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302708 memset((u8 *)ar->sta_list, 0,
2709 AP_MAX_NUM_STA * sizeof(struct ath6kl_sta));
2710
2711 /* Init the PS queues */
2712 for (ctr = 0; ctr < AP_MAX_NUM_STA; ctr++) {
2713 spin_lock_init(&ar->sta_list[ctr].psq_lock);
2714 skb_queue_head_init(&ar->sta_list[ctr].psq);
2715 }
2716
2717 skb_queue_head_init(&ar->mcastpsq);
2718
2719 memcpy(ar->ap_country_code, DEF_AP_COUNTRY_CODE, 3);
2720
2721 return ar;
2722}
2723
2724int ath6kl_register_ieee80211_hw(struct ath6kl *ar)
2725{
2726 struct wiphy *wiphy = ar->wiphy;
2727 int ret;
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002728
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302729 wiphy->mgmt_stypes = ath6kl_mgmt_stypes;
Jouni Malinenf80574a2011-08-30 21:58:04 +03002730
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302731 wiphy->max_remain_on_channel_duration = 5000;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002732
Kalle Valobdcd8172011-07-18 00:22:30 +03002733 /* set device pointer for wiphy */
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302734 set_wiphy_dev(wiphy, ar->dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03002735
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302736 wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302737 BIT(NL80211_IFTYPE_ADHOC) |
2738 BIT(NL80211_IFTYPE_AP);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002739 if (ar->p2p) {
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302740 wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_GO) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302741 BIT(NL80211_IFTYPE_P2P_CLIENT);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002742 }
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302743
Kalle Valobdcd8172011-07-18 00:22:30 +03002744 /* max num of ssids that can be probed during scanning */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302745 wiphy->max_scan_ssids = MAX_PROBED_SSID_INDEX;
2746 wiphy->max_scan_ie_len = 1000; /* FIX: what is correct limit? */
2747 wiphy->bands[IEEE80211_BAND_2GHZ] = &ath6kl_band_2ghz;
2748 wiphy->bands[IEEE80211_BAND_5GHZ] = &ath6kl_band_5ghz;
2749 wiphy->signal_type = CFG80211_SIGNAL_TYPE_MBM;
Kalle Valobdcd8172011-07-18 00:22:30 +03002750
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302751 wiphy->cipher_suites = cipher_suites;
2752 wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
Kalle Valobdcd8172011-07-18 00:22:30 +03002753
Raja Manieae9e062011-11-07 22:52:46 +02002754 wiphy->wowlan.flags = WIPHY_WOWLAN_MAGIC_PKT |
2755 WIPHY_WOWLAN_DISCONNECT |
2756 WIPHY_WOWLAN_GTK_REKEY_FAILURE |
2757 WIPHY_WOWLAN_SUPPORTS_GTK_REKEY |
2758 WIPHY_WOWLAN_EAP_IDENTITY_REQ |
2759 WIPHY_WOWLAN_4WAY_HANDSHAKE;
2760 wiphy->wowlan.n_patterns = WOW_MAX_FILTERS_PER_LIST;
2761 wiphy->wowlan.pattern_min_len = 1;
2762 wiphy->wowlan.pattern_max_len = WOW_PATTERN_SIZE;
2763
Kalle Valo10509f92011-12-13 14:52:07 +02002764 wiphy->max_sched_scan_ssids = 10;
2765
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302766 ret = wiphy_register(wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002767 if (ret < 0) {
2768 ath6kl_err("couldn't register wiphy device\n");
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302769 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +03002770 }
2771
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302772 return 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03002773}
2774
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302775static int ath6kl_init_if_data(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +03002776{
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302777 vif->aggr_cntxt = aggr_init(vif->ndev);
2778 if (!vif->aggr_cntxt) {
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302779 ath6kl_err("failed to initialize aggr\n");
2780 return -ENOMEM;
2781 }
Kalle Valobdcd8172011-07-18 00:22:30 +03002782
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302783 setup_timer(&vif->disconnect_timer, disconnect_timer_handler,
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302784 (unsigned long) vif->ndev);
Kalle Valo10509f92011-12-13 14:52:07 +02002785 setup_timer(&vif->sched_scan_timer, ath6kl_wmi_sscan_timer,
2786 (unsigned long) vif);
2787
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302788 set_bit(WMM_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan478ac022011-10-25 19:34:19 +05302789 spin_lock_init(&vif->if_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302790
2791 return 0;
2792}
2793
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302794void ath6kl_deinit_if_data(struct ath6kl_vif *vif)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302795{
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302796 struct ath6kl *ar = vif->ar;
2797
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302798 aggr_module_destroy(vif->aggr_cntxt);
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302799
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302800 ar->avail_idx_map |= BIT(vif->fw_vif_idx);
2801
2802 if (vif->nw_type == ADHOC_NETWORK)
2803 ar->ibss_if_active = false;
2804
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302805 unregister_netdevice(vif->ndev);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302806
2807 ar->num_vif--;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302808}
2809
2810struct net_device *ath6kl_interface_add(struct ath6kl *ar, char *name,
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302811 enum nl80211_iftype type, u8 fw_vif_idx,
2812 u8 nw_type)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302813{
2814 struct net_device *ndev;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302815 struct ath6kl_vif *vif;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302816
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302817 ndev = alloc_netdev(sizeof(*vif), name, ether_setup);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302818 if (!ndev)
2819 return NULL;
2820
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302821 vif = netdev_priv(ndev);
2822 ndev->ieee80211_ptr = &vif->wdev;
2823 vif->wdev.wiphy = ar->wiphy;
2824 vif->ar = ar;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302825 vif->ndev = ndev;
2826 SET_NETDEV_DEV(ndev, wiphy_dev(vif->wdev.wiphy));
2827 vif->wdev.netdev = ndev;
2828 vif->wdev.iftype = type;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302829 vif->fw_vif_idx = fw_vif_idx;
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302830 vif->nw_type = vif->next_mode = nw_type;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302831
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302832 memcpy(ndev->dev_addr, ar->mac_addr, ETH_ALEN);
2833 if (fw_vif_idx != 0)
2834 ndev->dev_addr[0] = (ndev->dev_addr[0] ^ (1 << fw_vif_idx)) |
2835 0x2;
2836
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302837 init_netdev(ndev);
2838
Vasanthakumar Thiagarajane29f25f2011-10-25 19:34:15 +05302839 ath6kl_init_control_info(vif);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302840
2841 /* TODO: Pass interface specific pointer instead of ar */
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302842 if (ath6kl_init_if_data(vif))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302843 goto err;
2844
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302845 if (register_netdevice(ndev))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302846 goto err;
2847
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302848 ar->avail_idx_map &= ~BIT(fw_vif_idx);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05302849 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302850 set_bit(WLAN_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302851 ar->wlan_pwr_state = WLAN_POWER_STATE_ON;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302852 set_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302853
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302854 if (type == NL80211_IFTYPE_ADHOC)
2855 ar->ibss_if_active = true;
2856
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05302857 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302858 list_add_tail(&vif->list, &ar->vif_list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05302859 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302860
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302861 return ndev;
2862
2863err:
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302864 aggr_module_destroy(vif->aggr_cntxt);
2865 free_netdev(ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302866 return NULL;
2867}
2868
2869void ath6kl_deinit_ieee80211_hw(struct ath6kl *ar)
2870{
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302871 wiphy_unregister(ar->wiphy);
2872 wiphy_free(ar->wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002873}