blob: f3a6cfc0ddc818ff891ae65370885d8dd3ccc377 [file] [log] [blame]
Kalle Valobdcd8172011-07-18 00:22:30 +03001/*
2 * Copyright (c) 2004-2011 Atheros Communications Inc.
Vasanthakumar Thiagarajan1b2df402012-02-06 20:15:53 +05303 * Copyright (c) 2011-2012 Qualcomm Atheros, Inc.
Kalle Valobdcd8172011-07-18 00:22:30 +03004 *
5 * Permission to use, copy, modify, and/or distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
8 *
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16 */
17
Joe Perches516304b2012-03-18 17:30:52 -070018#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
19
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040020#include <linux/moduleparam.h>
Raja Manic08631c2011-12-16 14:24:24 +053021#include <linux/inetdevice.h>
Kalle Valod6a434d2012-01-17 20:09:36 +020022#include <linux/export.h>
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040023
Kalle Valobdcd8172011-07-18 00:22:30 +030024#include "core.h"
25#include "cfg80211.h"
26#include "debug.h"
Kalle Valoabcb3442011-07-22 08:26:20 +030027#include "hif-ops.h"
Kalle Valo003353b0d2011-09-01 10:14:21 +030028#include "testmode.h"
Kalle Valobdcd8172011-07-18 00:22:30 +030029
30#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
Bala Shanmugamf5993592012-03-27 12:17:32 +053054#define DEFAULT_BG_SCAN_PERIOD 60
55
Naveen Singhdd45b752012-05-16 13:29:00 +030056struct ath6kl_cfg80211_match_probe_ssid {
57 struct cfg80211_ssid ssid;
58 u8 flag;
59};
60
Kalle Valobdcd8172011-07-18 00:22:30 +030061static struct ieee80211_rate ath6kl_rates[] = {
62 RATETAB_ENT(10, 0x1, 0),
63 RATETAB_ENT(20, 0x2, 0),
64 RATETAB_ENT(55, 0x4, 0),
65 RATETAB_ENT(110, 0x8, 0),
66 RATETAB_ENT(60, 0x10, 0),
67 RATETAB_ENT(90, 0x20, 0),
68 RATETAB_ENT(120, 0x40, 0),
69 RATETAB_ENT(180, 0x80, 0),
70 RATETAB_ENT(240, 0x100, 0),
71 RATETAB_ENT(360, 0x200, 0),
72 RATETAB_ENT(480, 0x400, 0),
73 RATETAB_ENT(540, 0x800, 0),
74};
75
76#define ath6kl_a_rates (ath6kl_rates + 4)
77#define ath6kl_a_rates_size 8
78#define ath6kl_g_rates (ath6kl_rates + 0)
79#define ath6kl_g_rates_size 12
80
Vasanthakumar Thiagarajanbed56e32012-04-09 19:03:57 +053081#define ath6kl_g_htcap IEEE80211_HT_CAP_SGI_20
82#define ath6kl_a_htcap (IEEE80211_HT_CAP_SUP_WIDTH_20_40 | \
Vasanthakumar Thiagarajanfaaf1922012-02-27 11:44:19 +053083 IEEE80211_HT_CAP_SGI_20 | \
84 IEEE80211_HT_CAP_SGI_40)
85
Kalle Valobdcd8172011-07-18 00:22:30 +030086static struct ieee80211_channel ath6kl_2ghz_channels[] = {
87 CHAN2G(1, 2412, 0),
88 CHAN2G(2, 2417, 0),
89 CHAN2G(3, 2422, 0),
90 CHAN2G(4, 2427, 0),
91 CHAN2G(5, 2432, 0),
92 CHAN2G(6, 2437, 0),
93 CHAN2G(7, 2442, 0),
94 CHAN2G(8, 2447, 0),
95 CHAN2G(9, 2452, 0),
96 CHAN2G(10, 2457, 0),
97 CHAN2G(11, 2462, 0),
98 CHAN2G(12, 2467, 0),
99 CHAN2G(13, 2472, 0),
100 CHAN2G(14, 2484, 0),
101};
102
103static struct ieee80211_channel ath6kl_5ghz_a_channels[] = {
104 CHAN5G(34, 0), CHAN5G(36, 0),
105 CHAN5G(38, 0), CHAN5G(40, 0),
106 CHAN5G(42, 0), CHAN5G(44, 0),
107 CHAN5G(46, 0), CHAN5G(48, 0),
108 CHAN5G(52, 0), CHAN5G(56, 0),
109 CHAN5G(60, 0), CHAN5G(64, 0),
110 CHAN5G(100, 0), CHAN5G(104, 0),
111 CHAN5G(108, 0), CHAN5G(112, 0),
112 CHAN5G(116, 0), CHAN5G(120, 0),
113 CHAN5G(124, 0), CHAN5G(128, 0),
114 CHAN5G(132, 0), CHAN5G(136, 0),
115 CHAN5G(140, 0), CHAN5G(149, 0),
116 CHAN5G(153, 0), CHAN5G(157, 0),
117 CHAN5G(161, 0), CHAN5G(165, 0),
118 CHAN5G(184, 0), CHAN5G(188, 0),
119 CHAN5G(192, 0), CHAN5G(196, 0),
120 CHAN5G(200, 0), CHAN5G(204, 0),
121 CHAN5G(208, 0), CHAN5G(212, 0),
122 CHAN5G(216, 0),
123};
124
125static struct ieee80211_supported_band ath6kl_band_2ghz = {
126 .n_channels = ARRAY_SIZE(ath6kl_2ghz_channels),
127 .channels = ath6kl_2ghz_channels,
128 .n_bitrates = ath6kl_g_rates_size,
129 .bitrates = ath6kl_g_rates,
Vasanthakumar Thiagarajanfaaf1922012-02-27 11:44:19 +0530130 .ht_cap.cap = ath6kl_g_htcap,
131 .ht_cap.ht_supported = true,
Kalle Valobdcd8172011-07-18 00:22:30 +0300132};
133
134static struct ieee80211_supported_band ath6kl_band_5ghz = {
135 .n_channels = ARRAY_SIZE(ath6kl_5ghz_a_channels),
136 .channels = ath6kl_5ghz_a_channels,
137 .n_bitrates = ath6kl_a_rates_size,
138 .bitrates = ath6kl_a_rates,
Vasanthakumar Thiagarajanbed56e32012-04-09 19:03:57 +0530139 .ht_cap.cap = ath6kl_a_htcap,
Vasanthakumar Thiagarajanfaaf1922012-02-27 11:44:19 +0530140 .ht_cap.ht_supported = true,
Kalle Valobdcd8172011-07-18 00:22:30 +0300141};
142
Jouni Malinen837cb972011-10-11 17:31:57 +0300143#define CCKM_KRK_CIPHER_SUITE 0x004096ff /* use for KRK */
144
Kalle Valo10509f92011-12-13 14:52:07 +0200145/* returns true if scheduled scan was stopped */
146static bool __ath6kl_cfg80211_sscan_stop(struct ath6kl_vif *vif)
147{
148 struct ath6kl *ar = vif->ar;
149
150 if (ar->state != ATH6KL_STATE_SCHED_SCAN)
151 return false;
152
153 del_timer_sync(&vif->sched_scan_timer);
154
155 ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
156 ATH6KL_HOST_MODE_AWAKE);
157
158 ar->state = ATH6KL_STATE_ON;
159
160 return true;
161}
162
163static void ath6kl_cfg80211_sscan_disable(struct ath6kl_vif *vif)
164{
165 struct ath6kl *ar = vif->ar;
166 bool stopped;
167
168 stopped = __ath6kl_cfg80211_sscan_stop(vif);
169
170 if (!stopped)
171 return;
172
173 cfg80211_sched_scan_stopped(ar->wiphy);
174}
175
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530176static int ath6kl_set_wpa_version(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300177 enum nl80211_wpa_versions wpa_version)
178{
179 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: %u\n", __func__, wpa_version);
180
181 if (!wpa_version) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530182 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300183 } else if (wpa_version & NL80211_WPA_VERSION_2) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530184 vif->auth_mode = WPA2_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300185 } else if (wpa_version & NL80211_WPA_VERSION_1) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530186 vif->auth_mode = WPA_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300187 } else {
188 ath6kl_err("%s: %u not supported\n", __func__, wpa_version);
189 return -ENOTSUPP;
190 }
191
192 return 0;
193}
194
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530195static int ath6kl_set_auth_type(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300196 enum nl80211_auth_type auth_type)
197{
Kalle Valobdcd8172011-07-18 00:22:30 +0300198 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, auth_type);
199
200 switch (auth_type) {
201 case NL80211_AUTHTYPE_OPEN_SYSTEM:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530202 vif->dot11_auth_mode = OPEN_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300203 break;
204 case NL80211_AUTHTYPE_SHARED_KEY:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530205 vif->dot11_auth_mode = SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300206 break;
207 case NL80211_AUTHTYPE_NETWORK_EAP:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530208 vif->dot11_auth_mode = LEAP_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300209 break;
210
211 case NL80211_AUTHTYPE_AUTOMATIC:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530212 vif->dot11_auth_mode = OPEN_AUTH | SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300213 break;
214
215 default:
Masanari Iida3c325fb2012-01-31 23:32:55 +0900216 ath6kl_err("%s: 0x%x not supported\n", __func__, auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300217 return -ENOTSUPP;
218 }
219
220 return 0;
221}
222
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530223static int ath6kl_set_cipher(struct ath6kl_vif *vif, u32 cipher, bool ucast)
Kalle Valobdcd8172011-07-18 00:22:30 +0300224{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530225 u8 *ar_cipher = ucast ? &vif->prwise_crypto : &vif->grp_crypto;
226 u8 *ar_cipher_len = ucast ? &vif->prwise_crypto_len :
227 &vif->grp_crypto_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300228
229 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: cipher 0x%x, ucast %u\n",
230 __func__, cipher, ucast);
231
232 switch (cipher) {
233 case 0:
234 /* our own hack to use value 0 as no crypto used */
235 *ar_cipher = NONE_CRYPT;
236 *ar_cipher_len = 0;
237 break;
238 case WLAN_CIPHER_SUITE_WEP40:
239 *ar_cipher = WEP_CRYPT;
240 *ar_cipher_len = 5;
241 break;
242 case WLAN_CIPHER_SUITE_WEP104:
243 *ar_cipher = WEP_CRYPT;
244 *ar_cipher_len = 13;
245 break;
246 case WLAN_CIPHER_SUITE_TKIP:
247 *ar_cipher = TKIP_CRYPT;
248 *ar_cipher_len = 0;
249 break;
250 case WLAN_CIPHER_SUITE_CCMP:
251 *ar_cipher = AES_CRYPT;
252 *ar_cipher_len = 0;
253 break;
Dai Shuibing5e070212011-11-03 11:39:37 +0200254 case WLAN_CIPHER_SUITE_SMS4:
255 *ar_cipher = WAPI_CRYPT;
256 *ar_cipher_len = 0;
257 break;
Kalle Valobdcd8172011-07-18 00:22:30 +0300258 default:
259 ath6kl_err("cipher 0x%x not supported\n", cipher);
260 return -ENOTSUPP;
261 }
262
263 return 0;
264}
265
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530266static void ath6kl_set_key_mgmt(struct ath6kl_vif *vif, u32 key_mgmt)
Kalle Valobdcd8172011-07-18 00:22:30 +0300267{
268 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, key_mgmt);
269
270 if (key_mgmt == WLAN_AKM_SUITE_PSK) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530271 if (vif->auth_mode == WPA_AUTH)
272 vif->auth_mode = WPA_PSK_AUTH;
273 else if (vif->auth_mode == WPA2_AUTH)
274 vif->auth_mode = WPA2_PSK_AUTH;
Jouni Malinen837cb972011-10-11 17:31:57 +0300275 } else if (key_mgmt == 0x00409600) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530276 if (vif->auth_mode == WPA_AUTH)
277 vif->auth_mode = WPA_AUTH_CCKM;
278 else if (vif->auth_mode == WPA2_AUTH)
279 vif->auth_mode = WPA2_AUTH_CCKM;
Kalle Valobdcd8172011-07-18 00:22:30 +0300280 } else if (key_mgmt != WLAN_AKM_SUITE_8021X) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530281 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300282 }
283}
284
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530285static bool ath6kl_cfg80211_ready(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +0300286{
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530287 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530288
Kalle Valobdcd8172011-07-18 00:22:30 +0300289 if (!test_bit(WMI_READY, &ar->flag)) {
290 ath6kl_err("wmi is not ready\n");
291 return false;
292 }
293
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530294 if (!test_bit(WLAN_ENABLED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300295 ath6kl_err("wlan disabled\n");
296 return false;
297 }
298
299 return true;
300}
301
Kevin Fang6981ffd2011-10-07 08:51:19 +0800302static bool ath6kl_is_wpa_ie(const u8 *pos)
303{
304 return pos[0] == WLAN_EID_WPA && pos[1] >= 4 &&
305 pos[2] == 0x00 && pos[3] == 0x50 &&
306 pos[4] == 0xf2 && pos[5] == 0x01;
307}
308
309static bool ath6kl_is_rsn_ie(const u8 *pos)
310{
311 return pos[0] == WLAN_EID_RSN;
312}
313
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700314static bool ath6kl_is_wps_ie(const u8 *pos)
315{
316 return (pos[0] == WLAN_EID_VENDOR_SPECIFIC &&
317 pos[1] >= 4 &&
318 pos[2] == 0x00 && pos[3] == 0x50 && pos[4] == 0xf2 &&
319 pos[5] == 0x04);
320}
321
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530322static int ath6kl_set_assoc_req_ies(struct ath6kl_vif *vif, const u8 *ies,
323 size_t ies_len)
Kevin Fang6981ffd2011-10-07 08:51:19 +0800324{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530325 struct ath6kl *ar = vif->ar;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800326 const u8 *pos;
327 u8 *buf = NULL;
328 size_t len = 0;
329 int ret;
330
331 /*
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700332 * Clear previously set flag
333 */
334
335 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
336
337 /*
Kevin Fang6981ffd2011-10-07 08:51:19 +0800338 * Filter out RSN/WPA IE(s)
339 */
340
341 if (ies && ies_len) {
342 buf = kmalloc(ies_len, GFP_KERNEL);
343 if (buf == NULL)
344 return -ENOMEM;
345 pos = ies;
346
347 while (pos + 1 < ies + ies_len) {
348 if (pos + 2 + pos[1] > ies + ies_len)
349 break;
350 if (!(ath6kl_is_wpa_ie(pos) || ath6kl_is_rsn_ie(pos))) {
351 memcpy(buf + len, pos, 2 + pos[1]);
352 len += 2 + pos[1];
353 }
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700354
355 if (ath6kl_is_wps_ie(pos))
356 ar->connect_ctrl_flags |= CONNECT_WPS_FLAG;
357
Kevin Fang6981ffd2011-10-07 08:51:19 +0800358 pos += 2 + pos[1];
359 }
360 }
361
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530362 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
363 WMI_FRAME_ASSOC_REQ, buf, len);
Kevin Fang6981ffd2011-10-07 08:51:19 +0800364 kfree(buf);
365 return ret;
366}
367
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530368static int ath6kl_nliftype_to_drv_iftype(enum nl80211_iftype type, u8 *nw_type)
369{
370 switch (type) {
371 case NL80211_IFTYPE_STATION:
372 *nw_type = INFRA_NETWORK;
373 break;
374 case NL80211_IFTYPE_ADHOC:
375 *nw_type = ADHOC_NETWORK;
376 break;
377 case NL80211_IFTYPE_AP:
378 *nw_type = AP_NETWORK;
379 break;
380 case NL80211_IFTYPE_P2P_CLIENT:
381 *nw_type = INFRA_NETWORK;
382 break;
383 case NL80211_IFTYPE_P2P_GO:
384 *nw_type = AP_NETWORK;
385 break;
386 default:
387 ath6kl_err("invalid interface type %u\n", type);
388 return -ENOTSUPP;
389 }
390
391 return 0;
392}
393
394static bool ath6kl_is_valid_iftype(struct ath6kl *ar, enum nl80211_iftype type,
395 u8 *if_idx, u8 *nw_type)
396{
397 int i;
398
399 if (ath6kl_nliftype_to_drv_iftype(type, nw_type))
400 return false;
401
402 if (ar->ibss_if_active || ((type == NL80211_IFTYPE_ADHOC) &&
Kalle Valo96f1fad2012-03-07 20:03:57 +0200403 ar->num_vif))
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530404 return false;
405
406 if (type == NL80211_IFTYPE_STATION ||
407 type == NL80211_IFTYPE_AP || type == NL80211_IFTYPE_ADHOC) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200408 for (i = 0; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530409 if ((ar->avail_idx_map >> i) & BIT(0)) {
410 *if_idx = i;
411 return true;
412 }
413 }
414 }
415
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530416 if (type == NL80211_IFTYPE_P2P_CLIENT ||
417 type == NL80211_IFTYPE_P2P_GO) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200418 for (i = ar->max_norm_iface; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530419 if ((ar->avail_idx_map >> i) & BIT(0)) {
420 *if_idx = i;
421 return true;
422 }
423 }
424 }
425
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530426 return false;
427}
428
Kalle Valo8c9bb052012-03-07 20:04:00 +0200429static bool ath6kl_is_tx_pending(struct ath6kl *ar)
430{
431 return ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)] == 0;
432}
433
434
Kalle Valobdcd8172011-07-18 00:22:30 +0300435static int ath6kl_cfg80211_connect(struct wiphy *wiphy, struct net_device *dev,
436 struct cfg80211_connect_params *sme)
437{
438 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530439 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300440 int status;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800441 u8 nw_subtype = (ar->p2p) ? SUBTYPE_P2PDEV : SUBTYPE_NONE;
Raja Manice0dc0c2012-02-20 19:08:08 +0530442 u16 interval;
Kalle Valobdcd8172011-07-18 00:22:30 +0300443
Kalle Valo10509f92011-12-13 14:52:07 +0200444 ath6kl_cfg80211_sscan_disable(vif);
445
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530446 vif->sme_state = SME_CONNECTING;
Kalle Valobdcd8172011-07-18 00:22:30 +0300447
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530448 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300449 return -EIO;
450
451 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
452 ath6kl_err("destroy in progress\n");
453 return -EBUSY;
454 }
455
456 if (test_bit(SKIP_SCAN, &ar->flag) &&
457 ((sme->channel && sme->channel->center_freq == 0) ||
458 (sme->bssid && is_zero_ether_addr(sme->bssid)))) {
459 ath6kl_err("SkipScan: channel or bssid invalid\n");
460 return -EINVAL;
461 }
462
463 if (down_interruptible(&ar->sem)) {
464 ath6kl_err("busy, couldn't get access\n");
465 return -ERESTARTSYS;
466 }
467
468 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
469 ath6kl_err("busy, destroy in progress\n");
470 up(&ar->sem);
471 return -EBUSY;
472 }
473
474 if (ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)]) {
475 /*
476 * sleep until the command queue drains
477 */
478 wait_event_interruptible_timeout(ar->event_wq,
Kalle Valo8c9bb052012-03-07 20:04:00 +0200479 ath6kl_is_tx_pending(ar),
480 WMI_TIMEOUT);
Kalle Valobdcd8172011-07-18 00:22:30 +0300481 if (signal_pending(current)) {
482 ath6kl_err("cmd queue drain timeout\n");
483 up(&ar->sem);
484 return -EINTR;
485 }
486 }
487
Jouni Malinen6e786cb2011-12-15 14:16:00 +0200488 status = ath6kl_set_assoc_req_ies(vif, sme->ie, sme->ie_len);
489 if (status) {
490 up(&ar->sem);
491 return status;
492 }
493
494 if (sme->ie == NULL || sme->ie_len == 0)
Raja Mani542c5192011-11-15 14:14:56 +0530495 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800496
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530497 if (test_bit(CONNECTED, &vif->flags) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530498 vif->ssid_len == sme->ssid_len &&
499 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530500 vif->reconnect_flag = true;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530501 status = ath6kl_wmi_reconnect_cmd(ar->wmi, vif->fw_vif_idx,
502 vif->req_bssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530503 vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300504
505 up(&ar->sem);
506 if (status) {
507 ath6kl_err("wmi_reconnect_cmd failed\n");
508 return -EIO;
509 }
510 return 0;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530511 } else if (vif->ssid_len == sme->ssid_len &&
512 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530513 ath6kl_disconnect(vif);
Kalle Valobdcd8172011-07-18 00:22:30 +0300514 }
515
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530516 memset(vif->ssid, 0, sizeof(vif->ssid));
517 vif->ssid_len = sme->ssid_len;
518 memcpy(vif->ssid, sme->ssid, sme->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +0300519
520 if (sme->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530521 vif->ch_hint = sme->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +0300522
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530523 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300524 if (sme->bssid && !is_broadcast_ether_addr(sme->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530525 memcpy(vif->req_bssid, sme->bssid, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300526
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530527 ath6kl_set_wpa_version(vif, sme->crypto.wpa_versions);
Kalle Valobdcd8172011-07-18 00:22:30 +0300528
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530529 status = ath6kl_set_auth_type(vif, sme->auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300530 if (status) {
531 up(&ar->sem);
532 return status;
533 }
534
535 if (sme->crypto.n_ciphers_pairwise)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530536 ath6kl_set_cipher(vif, sme->crypto.ciphers_pairwise[0], true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300537 else
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530538 ath6kl_set_cipher(vif, 0, true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300539
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530540 ath6kl_set_cipher(vif, sme->crypto.cipher_group, false);
Kalle Valobdcd8172011-07-18 00:22:30 +0300541
542 if (sme->crypto.n_akm_suites)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530543 ath6kl_set_key_mgmt(vif, sme->crypto.akm_suites[0]);
Kalle Valobdcd8172011-07-18 00:22:30 +0300544
545 if ((sme->key_len) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530546 (vif->auth_mode == NONE_AUTH) &&
547 (vif->prwise_crypto == WEP_CRYPT)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300548 struct ath6kl_key *key = NULL;
549
Vivek Natarajan792ecb32011-12-29 16:18:39 +0530550 if (sme->key_idx > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300551 ath6kl_err("key index %d out of bounds\n",
552 sme->key_idx);
553 up(&ar->sem);
554 return -ENOENT;
555 }
556
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530557 key = &vif->keys[sme->key_idx];
Kalle Valobdcd8172011-07-18 00:22:30 +0300558 key->key_len = sme->key_len;
559 memcpy(key->key, sme->key, key->key_len);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530560 key->cipher = vif->prwise_crypto;
561 vif->def_txkey_index = sme->key_idx;
Kalle Valobdcd8172011-07-18 00:22:30 +0300562
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530563 ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, sme->key_idx,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530564 vif->prwise_crypto,
Kalle Valobdcd8172011-07-18 00:22:30 +0300565 GROUP_USAGE | TX_USAGE,
566 key->key_len,
Jouni Malinenf4bb9a62011-11-02 23:45:55 +0200567 NULL, 0,
Kalle Valobdcd8172011-07-18 00:22:30 +0300568 key->key, KEY_OP_INIT_VAL, NULL,
569 NO_SYNC_WMIFLAG);
570 }
571
572 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530573 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530574 if (ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
Kalle Valo96f1fad2012-03-07 20:03:57 +0200575 ALL_BSS_FILTER, 0) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300576 ath6kl_err("couldn't set bss filtering\n");
577 up(&ar->sem);
578 return -EIO;
579 }
580 }
581
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530582 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +0300583
Thomas Pedersenc422d52d2012-05-15 00:09:23 -0700584 /* enable enhanced bmiss detection if applicable */
585 ath6kl_cfg80211_sta_bmiss_enhance(vif, true);
586
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800587 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_CLIENT)
588 nw_subtype = SUBTYPE_P2PCLIENT;
589
Kalle Valobdcd8172011-07-18 00:22:30 +0300590 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
591 "%s: connect called with authmode %d dot11 auth %d"
592 " PW crypto %d PW crypto len %d GRP crypto %d"
593 " GRP crypto len %d channel hint %u\n",
594 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530595 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
596 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530597 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300598
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530599 vif->reconnect_flag = 0;
Raja Manice0dc0c2012-02-20 19:08:08 +0530600
601 if (vif->nw_type == INFRA_NETWORK) {
Kalle Valob5283872012-03-12 13:23:23 +0200602 interval = max_t(u16, vif->listen_intvl_t,
603 ATH6KL_MAX_WOW_LISTEN_INTL);
Raja Manice0dc0c2012-02-20 19:08:08 +0530604 status = ath6kl_wmi_listeninterval_cmd(ar->wmi, vif->fw_vif_idx,
605 interval,
606 0);
607 if (status) {
608 ath6kl_err("couldn't set listen intervel\n");
609 up(&ar->sem);
610 return status;
611 }
612 }
613
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530614 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530615 vif->dot11_auth_mode, vif->auth_mode,
616 vif->prwise_crypto,
617 vif->prwise_crypto_len,
618 vif->grp_crypto, vif->grp_crypto_len,
619 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530620 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800621 ar->connect_ctrl_flags, nw_subtype);
Kalle Valobdcd8172011-07-18 00:22:30 +0300622
Bala Shanmugamf5993592012-03-27 12:17:32 +0530623 /* disable background scan if period is 0 */
624 if (sme->bg_scan_period == 0)
625 sme->bg_scan_period = 0xffff;
626
627 /* configure default value if not specified */
628 if (sme->bg_scan_period == -1)
629 sme->bg_scan_period = DEFAULT_BG_SCAN_PERIOD;
630
631 ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx, 0, 0,
632 sme->bg_scan_period, 0, 0, 0, 3, 0, 0, 0);
633
Kalle Valobdcd8172011-07-18 00:22:30 +0300634 up(&ar->sem);
635
636 if (status == -EINVAL) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530637 memset(vif->ssid, 0, sizeof(vif->ssid));
638 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300639 ath6kl_err("invalid request\n");
640 return -ENOENT;
641 } else if (status) {
642 ath6kl_err("ath6kl_wmi_connect_cmd failed\n");
643 return -EIO;
644 }
645
646 if ((!(ar->connect_ctrl_flags & CONNECT_DO_WPA_OFFLOAD)) &&
Kalle Valoddc3d772012-03-07 20:03:58 +0200647 ((vif->auth_mode == WPA_PSK_AUTH) ||
648 (vif->auth_mode == WPA2_PSK_AUTH))) {
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +0530649 mod_timer(&vif->disconnect_timer,
Kalle Valobdcd8172011-07-18 00:22:30 +0300650 jiffies + msecs_to_jiffies(DISCON_TIMER_INTVAL));
651 }
652
653 ar->connect_ctrl_flags &= ~CONNECT_DO_WPA_OFFLOAD;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530654 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300655
656 return 0;
657}
658
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530659static struct cfg80211_bss *
660ath6kl_add_bss_if_needed(struct ath6kl_vif *vif,
661 enum network_type nw_type,
662 const u8 *bssid,
663 struct ieee80211_channel *chan,
664 const u8 *beacon_ie,
665 size_t beacon_ie_len)
Jouni Malinen01cac472011-09-19 19:14:59 +0300666{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530667 struct ath6kl *ar = vif->ar;
Jouni Malinen01cac472011-09-19 19:14:59 +0300668 struct cfg80211_bss *bss;
Raja Mani4eab6f42011-11-09 17:02:23 +0530669 u16 cap_mask, cap_val;
Jouni Malinen01cac472011-09-19 19:14:59 +0300670 u8 *ie;
671
Raja Mani4eab6f42011-11-09 17:02:23 +0530672 if (nw_type & ADHOC_NETWORK) {
673 cap_mask = WLAN_CAPABILITY_IBSS;
674 cap_val = WLAN_CAPABILITY_IBSS;
675 } else {
676 cap_mask = WLAN_CAPABILITY_ESS;
677 cap_val = WLAN_CAPABILITY_ESS;
678 }
679
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530680 bss = cfg80211_get_bss(ar->wiphy, chan, bssid,
Raja Mani4eab6f42011-11-09 17:02:23 +0530681 vif->ssid, vif->ssid_len,
682 cap_mask, cap_val);
Jouni Malinen01cac472011-09-19 19:14:59 +0300683 if (bss == NULL) {
684 /*
685 * Since cfg80211 may not yet know about the BSS,
686 * generate a partial entry until the first BSS info
687 * event becomes available.
688 *
689 * Prepend SSID element since it is not included in the Beacon
690 * IEs from the target.
691 */
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530692 ie = kmalloc(2 + vif->ssid_len + beacon_ie_len, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300693 if (ie == NULL)
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530694 return NULL;
Jouni Malinen01cac472011-09-19 19:14:59 +0300695 ie[0] = WLAN_EID_SSID;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530696 ie[1] = vif->ssid_len;
697 memcpy(ie + 2, vif->ssid, vif->ssid_len);
698 memcpy(ie + 2 + vif->ssid_len, beacon_ie, beacon_ie_len);
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530699 bss = cfg80211_inform_bss(ar->wiphy, chan,
Raja Mani4eab6f42011-11-09 17:02:23 +0530700 bssid, 0, cap_val, 100,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530701 ie, 2 + vif->ssid_len + beacon_ie_len,
Jouni Malinen01cac472011-09-19 19:14:59 +0300702 0, GFP_KERNEL);
703 if (bss)
Kalle Valocdeb8602012-04-12 11:02:18 +0300704 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
705 "added bss %pM to cfg80211\n", bssid);
Jouni Malinen01cac472011-09-19 19:14:59 +0300706 kfree(ie);
707 } else
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530708 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "cfg80211 already has a bss\n");
Jouni Malinen01cac472011-09-19 19:14:59 +0300709
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530710 return bss;
Jouni Malinen01cac472011-09-19 19:14:59 +0300711}
712
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530713void ath6kl_cfg80211_connect_event(struct ath6kl_vif *vif, u16 channel,
Kalle Valobdcd8172011-07-18 00:22:30 +0300714 u8 *bssid, u16 listen_intvl,
715 u16 beacon_intvl,
716 enum network_type nw_type,
717 u8 beacon_ie_len, u8 assoc_req_len,
718 u8 assoc_resp_len, u8 *assoc_info)
719{
Jouni Malinen01cac472011-09-19 19:14:59 +0300720 struct ieee80211_channel *chan;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530721 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530722 struct cfg80211_bss *bss;
Kalle Valobdcd8172011-07-18 00:22:30 +0300723
724 /* capinfo + listen interval */
725 u8 assoc_req_ie_offset = sizeof(u16) + sizeof(u16);
726
727 /* capinfo + status code + associd */
728 u8 assoc_resp_ie_offset = sizeof(u16) + sizeof(u16) + sizeof(u16);
729
730 u8 *assoc_req_ie = assoc_info + beacon_ie_len + assoc_req_ie_offset;
731 u8 *assoc_resp_ie = assoc_info + beacon_ie_len + assoc_req_len +
732 assoc_resp_ie_offset;
733
734 assoc_req_len -= assoc_req_ie_offset;
735 assoc_resp_len -= assoc_resp_ie_offset;
736
Jouni Malinen32c10872011-09-19 19:15:07 +0300737 /*
738 * Store Beacon interval here; DTIM period will be available only once
739 * a Beacon frame from the AP is seen.
740 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530741 vif->assoc_bss_beacon_int = beacon_intvl;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530742 clear_bit(DTIM_PERIOD_AVAIL, &vif->flags);
Jouni Malinen32c10872011-09-19 19:15:07 +0300743
Kalle Valobdcd8172011-07-18 00:22:30 +0300744 if (nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530745 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300746 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
747 "%s: ath6k not in ibss mode\n", __func__);
748 return;
749 }
750 }
751
752 if (nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530753 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
754 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300755 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
756 "%s: ath6k not in station mode\n", __func__);
757 return;
758 }
759 }
760
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530761 chan = ieee80211_get_channel(ar->wiphy, (int) channel);
Kalle Valobdcd8172011-07-18 00:22:30 +0300762
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530763 bss = ath6kl_add_bss_if_needed(vif, nw_type, bssid, chan,
764 assoc_info, beacon_ie_len);
765 if (!bss) {
Raja Mani4eab6f42011-11-09 17:02:23 +0530766 ath6kl_err("could not add cfg80211 bss entry\n");
Kalle Valobdcd8172011-07-18 00:22:30 +0300767 return;
768 }
769
Raja Mani4eab6f42011-11-09 17:02:23 +0530770 if (nw_type & ADHOC_NETWORK) {
771 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "ad-hoc %s selected\n",
772 nw_type & ADHOC_CREATOR ? "creator" : "joiner");
773 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530774 cfg80211_put_bss(bss);
Jouni Malinen01cac472011-09-19 19:14:59 +0300775 return;
776 }
777
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530778 if (vif->sme_state == SME_CONNECTING) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300779 /* inform connect result to cfg80211 */
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530780 vif->sme_state = SME_CONNECTED;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530781 cfg80211_connect_result(vif->ndev, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300782 assoc_req_ie, assoc_req_len,
783 assoc_resp_ie, assoc_resp_len,
784 WLAN_STATUS_SUCCESS, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530785 cfg80211_put_bss(bss);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530786 } else if (vif->sme_state == SME_CONNECTED) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300787 /* inform roam event to cfg80211 */
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530788 cfg80211_roamed_bss(vif->ndev, bss, assoc_req_ie, assoc_req_len,
789 assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300790 }
791}
792
793static int ath6kl_cfg80211_disconnect(struct wiphy *wiphy,
794 struct net_device *dev, u16 reason_code)
795{
Kalle Valod6d5c062011-11-25 13:17:37 +0200796 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530797 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300798
799 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: reason=%u\n", __func__,
800 reason_code);
801
Kalle Valo10509f92011-12-13 14:52:07 +0200802 ath6kl_cfg80211_sscan_disable(vif);
803
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530804 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300805 return -EIO;
806
807 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
808 ath6kl_err("busy, destroy in progress\n");
809 return -EBUSY;
810 }
811
812 if (down_interruptible(&ar->sem)) {
813 ath6kl_err("busy, couldn't get access\n");
814 return -ERESTARTSYS;
815 }
816
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530817 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530818 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530819 memset(vif->ssid, 0, sizeof(vif->ssid));
820 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300821
822 if (!test_bit(SKIP_SCAN, &ar->flag))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530823 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300824
825 up(&ar->sem);
826
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530827 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan170826d2011-09-10 15:26:35 +0530828
Kalle Valobdcd8172011-07-18 00:22:30 +0300829 return 0;
830}
831
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530832void ath6kl_cfg80211_disconnect_event(struct ath6kl_vif *vif, u8 reason,
Kalle Valobdcd8172011-07-18 00:22:30 +0300833 u8 *bssid, u8 assoc_resp_len,
834 u8 *assoc_info, u16 proto_reason)
835{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530836 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530837
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530838 if (vif->scan_req) {
839 cfg80211_scan_done(vif->scan_req, true);
840 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300841 }
842
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530843 if (vif->nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530844 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300845 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
846 "%s: ath6k not in ibss mode\n", __func__);
847 return;
848 }
849 memset(bssid, 0, ETH_ALEN);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530850 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300851 return;
852 }
853
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530854 if (vif->nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530855 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
856 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300857 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
858 "%s: ath6k not in station mode\n", __func__);
859 return;
860 }
861 }
862
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530863 /*
864 * Send a disconnect command to target when a disconnect event is
865 * received with reason code other than 3 (DISCONNECT_CMD - disconnect
866 * request from host) to make the firmware stop trying to connect even
867 * after giving disconnect event. There will be one more disconnect
868 * event for this disconnect command with reason code DISCONNECT_CMD
869 * which will be notified to cfg80211.
870 */
Kalle Valobdcd8172011-07-18 00:22:30 +0300871
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530872 if (reason != DISCONNECT_CMD) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530873 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +0300874 return;
875 }
876
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530877 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300878
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530879 if (vif->sme_state == SME_CONNECTING) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530880 cfg80211_connect_result(vif->ndev,
Kalle Valo96f1fad2012-03-07 20:03:57 +0200881 bssid, NULL, 0,
882 NULL, 0,
883 WLAN_STATUS_UNSPECIFIED_FAILURE,
884 GFP_KERNEL);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530885 } else if (vif->sme_state == SME_CONNECTED) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530886 cfg80211_disconnected(vif->ndev, reason,
Kalle Valo96f1fad2012-03-07 20:03:57 +0200887 NULL, 0, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300888 }
889
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530890 vif->sme_state = SME_DISCONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300891}
892
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300893static int ath6kl_set_probed_ssids(struct ath6kl *ar,
894 struct ath6kl_vif *vif,
Naveen Singhdd45b752012-05-16 13:29:00 +0300895 struct cfg80211_ssid *ssids, int n_ssids,
896 struct cfg80211_match_set *match_set,
897 int n_match_ssid)
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300898{
Naveen Singhdd45b752012-05-16 13:29:00 +0300899 u8 i, j, index_to_add, ssid_found = false;
900 struct ath6kl_cfg80211_match_probe_ssid ssid_list[MAX_PROBED_SSIDS];
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300901
Naveen Singhdd45b752012-05-16 13:29:00 +0300902 memset(ssid_list, 0, sizeof(ssid_list));
903
904 if (n_ssids > MAX_PROBED_SSIDS ||
905 n_match_ssid > MAX_PROBED_SSIDS)
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300906 return -EINVAL;
907
908 for (i = 0; i < n_ssids; i++) {
Naveen Singhdd45b752012-05-16 13:29:00 +0300909 memcpy(ssid_list[i].ssid.ssid,
910 ssids[i].ssid,
911 ssids[i].ssid_len);
912 ssid_list[i].ssid.ssid_len = ssids[i].ssid_len;
913
914 if (ssids[i].ssid_len)
915 ssid_list[i].flag = SPECIFIC_SSID_FLAG;
916 else
917 ssid_list[i].flag = ANY_SSID_FLAG;
918
919 if (n_match_ssid == 0)
920 ssid_list[i].flag |= MATCH_SSID_FLAG;
921 }
922
923 index_to_add = i;
924
925 for (i = 0; i < n_match_ssid; i++) {
926 ssid_found = false;
927
928 for (j = 0; j < n_ssids; j++) {
929 if ((match_set[i].ssid.ssid_len ==
930 ssid_list[j].ssid.ssid_len) &&
931 (!memcmp(ssid_list[j].ssid.ssid,
932 match_set[i].ssid.ssid,
933 match_set[i].ssid.ssid_len))) {
934 ssid_list[j].flag |= MATCH_SSID_FLAG;
935 ssid_found = true;
936 break;
937 }
938 }
939
940 if (ssid_found)
941 continue;
942
943 if (index_to_add >= MAX_PROBED_SSIDS)
944 continue;
945
946 ssid_list[index_to_add].ssid.ssid_len =
947 match_set[i].ssid.ssid_len;
948 memcpy(ssid_list[index_to_add].ssid.ssid,
949 match_set[i].ssid.ssid,
950 match_set[i].ssid.ssid_len);
951 ssid_list[index_to_add].flag |= MATCH_SSID_FLAG;
952 index_to_add++;
953 }
954
955 for (i = 0; i < index_to_add; i++) {
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300956 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx, i,
Naveen Singhdd45b752012-05-16 13:29:00 +0300957 ssid_list[i].flag,
958 ssid_list[i].ssid.ssid_len,
959 ssid_list[i].ssid.ssid);
960
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300961 }
962
963 /* Make sure no old entries are left behind */
Naveen Singhdd45b752012-05-16 13:29:00 +0300964 for (i = index_to_add; i < MAX_PROBED_SSIDS; i++) {
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300965 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx, i,
966 DISABLE_SSID_FLAG, 0, NULL);
967 }
968
969 return 0;
970}
971
Kalle Valobdcd8172011-07-18 00:22:30 +0300972static int ath6kl_cfg80211_scan(struct wiphy *wiphy, struct net_device *ndev,
973 struct cfg80211_scan_request *request)
974{
Kalle Valod6d5c062011-11-25 13:17:37 +0200975 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530976 struct ath6kl_vif *vif = netdev_priv(ndev);
Edward Lu1276c9e2011-08-30 21:58:00 +0300977 s8 n_channels = 0;
978 u16 *channels = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300979 int ret = 0;
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530980 u32 force_fg_scan = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300981
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530982 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300983 return -EIO;
984
Kalle Valo10509f92011-12-13 14:52:07 +0200985 ath6kl_cfg80211_sscan_disable(vif);
986
Kalle Valobdcd8172011-07-18 00:22:30 +0300987 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530988 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Vasanthakumar Thiagarajan954e6ce2012-04-25 12:39:06 +0530989 ret = ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
990 ALL_BSS_FILTER, 0);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300991 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300992 ath6kl_err("couldn't set bss filtering\n");
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300993 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +0300994 }
995 }
996
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300997 ret = ath6kl_set_probed_ssids(ar, vif, request->ssids,
Naveen Singhdd45b752012-05-16 13:29:00 +0300998 request->n_ssids, NULL, 0);
Jouni Malinen3b8ffc62012-04-16 19:28:03 +0300999 if (ret < 0)
1000 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +03001001
Aarthi Thiruvengadam080eec42012-02-28 09:17:04 -08001002 /* this also clears IE in fw if it's not set */
1003 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
1004 WMI_FRAME_PROBE_REQ,
1005 request->ie, request->ie_len);
1006 if (ret) {
Kalle Valocdeb8602012-04-12 11:02:18 +03001007 ath6kl_err("failed to set Probe Request appie for scan");
Aarthi Thiruvengadam080eec42012-02-28 09:17:04 -08001008 return ret;
Jouni Malinenb84da8c2011-08-30 21:57:59 +03001009 }
1010
Jouni Malinen11869be2011-09-02 20:07:06 +03001011 /*
1012 * Scan only the requested channels if the request specifies a set of
1013 * channels. If the list is longer than the target supports, do not
1014 * configure the list and instead, scan all available channels.
1015 */
1016 if (request->n_channels > 0 &&
1017 request->n_channels <= WMI_MAX_CHANNELS) {
Edward Lu1276c9e2011-08-30 21:58:00 +03001018 u8 i;
1019
Jouni Malinen11869be2011-09-02 20:07:06 +03001020 n_channels = request->n_channels;
Edward Lu1276c9e2011-08-30 21:58:00 +03001021
1022 channels = kzalloc(n_channels * sizeof(u16), GFP_KERNEL);
1023 if (channels == NULL) {
Kalle Valocdeb8602012-04-12 11:02:18 +03001024 ath6kl_warn("failed to set scan channels, scan all channels");
Edward Lu1276c9e2011-08-30 21:58:00 +03001025 n_channels = 0;
1026 }
1027
1028 for (i = 0; i < n_channels; i++)
1029 channels[i] = request->channels[i]->center_freq;
1030 }
1031
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301032 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +05301033 force_fg_scan = 1;
1034
Raja Mani5b35dff2012-03-28 18:50:35 +05301035 vif->scan_req = request;
1036
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001037 if (test_bit(ATH6KL_FW_CAPABILITY_STA_P2PDEV_DUPLEX,
Kalle Valo96f1fad2012-03-07 20:03:57 +02001038 ar->fw_capabilities)) {
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001039 /*
1040 * If capable of doing P2P mgmt operations using
1041 * station interface, send additional information like
1042 * supported rates to advertise and xmit rates for
1043 * probe requests
1044 */
1045 ret = ath6kl_wmi_beginscan_cmd(ar->wmi, vif->fw_vif_idx,
1046 WMI_LONG_SCAN, force_fg_scan,
Vasanthakumar Thiagarajan13423c32012-02-21 15:30:42 +05301047 false, 0,
1048 ATH6KL_FG_SCAN_INTERVAL,
1049 n_channels, channels,
1050 request->no_cck,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001051 request->rates);
1052 } else {
1053 ret = ath6kl_wmi_startscan_cmd(ar->wmi, vif->fw_vif_idx,
1054 WMI_LONG_SCAN, force_fg_scan,
Vasanthakumar Thiagarajan13423c32012-02-21 15:30:42 +05301055 false, 0,
1056 ATH6KL_FG_SCAN_INTERVAL,
1057 n_channels, channels);
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001058 }
Raja Mani5b35dff2012-03-28 18:50:35 +05301059 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001060 ath6kl_err("wmi_startscan_cmd failed\n");
Raja Mani5b35dff2012-03-28 18:50:35 +05301061 vif->scan_req = NULL;
1062 }
Kalle Valobdcd8172011-07-18 00:22:30 +03001063
Edward Lu1276c9e2011-08-30 21:58:00 +03001064 kfree(channels);
1065
Kalle Valobdcd8172011-07-18 00:22:30 +03001066 return ret;
1067}
1068
Kalle Valo1c17d312011-11-01 08:43:56 +02001069void ath6kl_cfg80211_scan_complete_event(struct ath6kl_vif *vif, bool aborted)
Kalle Valobdcd8172011-07-18 00:22:30 +03001070{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301071 struct ath6kl *ar = vif->ar;
Kalle Valo6fd1eac2011-07-21 10:22:50 +03001072 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +03001073
Kalle Valo1c17d312011-11-01 08:43:56 +02001074 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: status%s\n", __func__,
1075 aborted ? " aborted" : "");
Kalle Valobdcd8172011-07-18 00:22:30 +03001076
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05301077 if (!vif->scan_req)
Kalle Valo6fd1eac2011-07-21 10:22:50 +03001078 return;
Kalle Valobdcd8172011-07-18 00:22:30 +03001079
Kalle Valo1c17d312011-11-01 08:43:56 +02001080 if (aborted)
Kalle Valo6fd1eac2011-07-21 10:22:50 +03001081 goto out;
Kalle Valo6fd1eac2011-07-21 10:22:50 +03001082
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05301083 if (vif->scan_req->n_ssids && vif->scan_req->ssids[0].ssid_len) {
1084 for (i = 0; i < vif->scan_req->n_ssids; i++) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301085 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
1086 i + 1, DISABLE_SSID_FLAG,
Kalle Valo6fd1eac2011-07-21 10:22:50 +03001087 0, NULL);
1088 }
1089 }
1090
1091out:
Kalle Valocb938212011-10-27 18:47:46 +03001092 cfg80211_scan_done(vif->scan_req, aborted);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05301093 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +03001094}
1095
Thomas Pedersenc4f78632012-04-06 13:35:48 -07001096void ath6kl_cfg80211_ch_switch_notify(struct ath6kl_vif *vif, int freq,
1097 enum wmi_phy_mode mode)
1098{
1099 enum nl80211_channel_type type;
1100
1101 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1102 "channel switch notify nw_type %d freq %d mode %d\n",
1103 vif->nw_type, freq, mode);
1104
1105 type = (mode == WMI_11G_HT20) ? NL80211_CHAN_HT20 : NL80211_CHAN_NO_HT;
1106
1107 cfg80211_ch_switch_notify(vif->ndev, freq, type);
1108}
1109
Kalle Valobdcd8172011-07-18 00:22:30 +03001110static int ath6kl_cfg80211_add_key(struct wiphy *wiphy, struct net_device *ndev,
1111 u8 key_index, bool pairwise,
1112 const u8 *mac_addr,
1113 struct key_params *params)
1114{
Kalle Valod6d5c062011-11-25 13:17:37 +02001115 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301116 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001117 struct ath6kl_key *key = NULL;
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +05301118 int seq_len;
Kalle Valobdcd8172011-07-18 00:22:30 +03001119 u8 key_usage;
1120 u8 key_type;
Kalle Valobdcd8172011-07-18 00:22:30 +03001121
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301122 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001123 return -EIO;
1124
Jouni Malinen837cb972011-10-11 17:31:57 +03001125 if (params->cipher == CCKM_KRK_CIPHER_SUITE) {
1126 if (params->key_len != WMI_KRK_LEN)
1127 return -EINVAL;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301128 return ath6kl_wmi_add_krk_cmd(ar->wmi, vif->fw_vif_idx,
1129 params->key);
Jouni Malinen837cb972011-10-11 17:31:57 +03001130 }
1131
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301132 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001133 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1134 "%s: key index %d out of bounds\n", __func__,
1135 key_index);
1136 return -ENOENT;
1137 }
1138
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301139 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001140 memset(key, 0, sizeof(struct ath6kl_key));
1141
1142 if (pairwise)
1143 key_usage = PAIRWISE_USAGE;
1144 else
1145 key_usage = GROUP_USAGE;
1146
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +05301147 seq_len = params->seq_len;
1148 if (params->cipher == WLAN_CIPHER_SUITE_SMS4 &&
1149 seq_len > ATH6KL_KEY_SEQ_LEN) {
1150 /* Only first half of the WPI PN is configured */
1151 seq_len = ATH6KL_KEY_SEQ_LEN;
Kalle Valobdcd8172011-07-18 00:22:30 +03001152 }
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +05301153 if (params->key_len > WLAN_MAX_KEY_LEN ||
1154 seq_len > sizeof(key->seq))
1155 return -EINVAL;
1156
1157 key->key_len = params->key_len;
1158 memcpy(key->key, params->key, key->key_len);
1159 key->seq_len = seq_len;
1160 memcpy(key->seq, params->seq, key->seq_len);
1161 key->cipher = params->cipher;
Kalle Valobdcd8172011-07-18 00:22:30 +03001162
1163 switch (key->cipher) {
1164 case WLAN_CIPHER_SUITE_WEP40:
1165 case WLAN_CIPHER_SUITE_WEP104:
1166 key_type = WEP_CRYPT;
1167 break;
1168
1169 case WLAN_CIPHER_SUITE_TKIP:
1170 key_type = TKIP_CRYPT;
1171 break;
1172
1173 case WLAN_CIPHER_SUITE_CCMP:
1174 key_type = AES_CRYPT;
1175 break;
Dai Shuibing5e070212011-11-03 11:39:37 +02001176 case WLAN_CIPHER_SUITE_SMS4:
1177 key_type = WAPI_CRYPT;
1178 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001179
1180 default:
1181 return -ENOTSUPP;
1182 }
1183
Kalle Valoddc3d772012-03-07 20:03:58 +02001184 if (((vif->auth_mode == WPA_PSK_AUTH) ||
1185 (vif->auth_mode == WPA2_PSK_AUTH)) &&
1186 (key_usage & GROUP_USAGE))
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05301187 del_timer(&vif->disconnect_timer);
Kalle Valobdcd8172011-07-18 00:22:30 +03001188
1189 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1190 "%s: index %d, key_len %d, key_type 0x%x, key_usage 0x%x, seq_len %d\n",
1191 __func__, key_index, key->key_len, key_type,
1192 key_usage, key->seq_len);
1193
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301194 if (vif->nw_type == AP_NETWORK && !pairwise &&
Jouni Malinen47032902011-12-08 16:50:30 +02001195 (key_type == TKIP_CRYPT || key_type == AES_CRYPT ||
Vasanthakumar Thiagarajancc4d6232012-02-14 20:33:00 +05301196 key_type == WAPI_CRYPT)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001197 ar->ap_mode_bkey.valid = true;
1198 ar->ap_mode_bkey.key_index = key_index;
1199 ar->ap_mode_bkey.key_type = key_type;
1200 ar->ap_mode_bkey.key_len = key->key_len;
1201 memcpy(ar->ap_mode_bkey.key, key->key, key->key_len);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301202 if (!test_bit(CONNECTED, &vif->flags)) {
Kalle Valocdeb8602012-04-12 11:02:18 +03001203 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1204 "Delay initial group key configuration until AP mode has been started\n");
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001205 /*
1206 * The key will be set in ath6kl_connect_ap_mode() once
1207 * the connected event is received from the target.
1208 */
1209 return 0;
1210 }
1211 }
1212
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301213 if (vif->next_mode == AP_NETWORK && key_type == WEP_CRYPT &&
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301214 !test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen151411e2011-09-15 15:10:16 +03001215 /*
1216 * Store the key locally so that it can be re-configured after
1217 * the AP mode has properly started
1218 * (ath6kl_install_statioc_wep_keys).
1219 */
Kalle Valocdeb8602012-04-12 11:02:18 +03001220 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1221 "Delay WEP key configuration until AP mode has been started\n");
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301222 vif->wep_key_list[key_index].key_len = key->key_len;
1223 memcpy(vif->wep_key_list[key_index].key, key->key,
1224 key->key_len);
Jouni Malinen151411e2011-09-15 15:10:16 +03001225 return 0;
1226 }
1227
Vasanthakumar Thiagarajan7cefa442011-11-11 20:33:00 +05301228 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, key_index,
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001229 key_type, key_usage, key->key_len,
1230 key->seq, key->seq_len, key->key,
1231 KEY_OP_INIT_VAL,
1232 (u8 *) mac_addr, SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001233}
1234
1235static int ath6kl_cfg80211_del_key(struct wiphy *wiphy, struct net_device *ndev,
1236 u8 key_index, bool pairwise,
1237 const u8 *mac_addr)
1238{
Kalle Valod6d5c062011-11-25 13:17:37 +02001239 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301240 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001241
1242 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1243
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301244 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001245 return -EIO;
1246
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301247 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001248 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1249 "%s: key index %d out of bounds\n", __func__,
1250 key_index);
1251 return -ENOENT;
1252 }
1253
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301254 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001255 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1256 "%s: index %d is empty\n", __func__, key_index);
1257 return 0;
1258 }
1259
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301260 vif->keys[key_index].key_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001261
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301262 return ath6kl_wmi_deletekey_cmd(ar->wmi, vif->fw_vif_idx, key_index);
Kalle Valobdcd8172011-07-18 00:22:30 +03001263}
1264
1265static int ath6kl_cfg80211_get_key(struct wiphy *wiphy, struct net_device *ndev,
1266 u8 key_index, bool pairwise,
1267 const u8 *mac_addr, void *cookie,
1268 void (*callback) (void *cookie,
1269 struct key_params *))
1270{
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301271 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001272 struct ath6kl_key *key = NULL;
1273 struct key_params params;
1274
1275 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1276
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301277 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001278 return -EIO;
1279
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301280 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001281 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1282 "%s: key index %d out of bounds\n", __func__,
1283 key_index);
1284 return -ENOENT;
1285 }
1286
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301287 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001288 memset(&params, 0, sizeof(params));
1289 params.cipher = key->cipher;
1290 params.key_len = key->key_len;
1291 params.seq_len = key->seq_len;
1292 params.seq = key->seq;
1293 params.key = key->key;
1294
1295 callback(cookie, &params);
1296
1297 return key->key_len ? 0 : -ENOENT;
1298}
1299
1300static int ath6kl_cfg80211_set_default_key(struct wiphy *wiphy,
1301 struct net_device *ndev,
1302 u8 key_index, bool unicast,
1303 bool multicast)
1304{
Kalle Valod6d5c062011-11-25 13:17:37 +02001305 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301306 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001307 struct ath6kl_key *key = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +03001308 u8 key_usage;
Edward Lu229ed6b2011-08-30 21:58:07 +03001309 enum crypto_type key_type = NONE_CRYPT;
Kalle Valobdcd8172011-07-18 00:22:30 +03001310
1311 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1312
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301313 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001314 return -EIO;
1315
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301316 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001317 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1318 "%s: key index %d out of bounds\n",
1319 __func__, key_index);
1320 return -ENOENT;
1321 }
1322
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301323 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001324 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: invalid key index %d\n",
1325 __func__, key_index);
1326 return -EINVAL;
1327 }
1328
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301329 vif->def_txkey_index = key_index;
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301330 key = &vif->keys[vif->def_txkey_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001331 key_usage = GROUP_USAGE;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301332 if (vif->prwise_crypto == WEP_CRYPT)
Kalle Valobdcd8172011-07-18 00:22:30 +03001333 key_usage |= TX_USAGE;
Edward Lu229ed6b2011-08-30 21:58:07 +03001334 if (unicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301335 key_type = vif->prwise_crypto;
Edward Lu229ed6b2011-08-30 21:58:07 +03001336 if (multicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301337 key_type = vif->grp_crypto;
Kalle Valobdcd8172011-07-18 00:22:30 +03001338
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301339 if (vif->next_mode == AP_NETWORK && !test_bit(CONNECTED, &vif->flags))
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001340 return 0; /* Delay until AP mode has been started */
1341
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001342 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx,
1343 vif->def_txkey_index,
1344 key_type, key_usage,
1345 key->key_len, key->seq, key->seq_len,
1346 key->key,
1347 KEY_OP_INIT_VAL, NULL,
1348 SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001349}
1350
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301351void ath6kl_cfg80211_tkip_micerr_event(struct ath6kl_vif *vif, u8 keyid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001352 bool ismcast)
1353{
1354 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1355 "%s: keyid %d, ismcast %d\n", __func__, keyid, ismcast);
1356
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301357 cfg80211_michael_mic_failure(vif->ndev, vif->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001358 (ismcast ? NL80211_KEYTYPE_GROUP :
1359 NL80211_KEYTYPE_PAIRWISE), keyid, NULL,
1360 GFP_KERNEL);
1361}
1362
1363static int ath6kl_cfg80211_set_wiphy_params(struct wiphy *wiphy, u32 changed)
1364{
1365 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301366 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001367 int ret;
1368
1369 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: changed 0x%x\n", __func__,
1370 changed);
1371
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301372 vif = ath6kl_vif_first(ar);
1373 if (!vif)
1374 return -EIO;
1375
1376 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001377 return -EIO;
1378
1379 if (changed & WIPHY_PARAM_RTS_THRESHOLD) {
1380 ret = ath6kl_wmi_set_rts_cmd(ar->wmi, wiphy->rts_threshold);
1381 if (ret != 0) {
1382 ath6kl_err("ath6kl_wmi_set_rts_cmd failed\n");
1383 return -EIO;
1384 }
1385 }
1386
1387 return 0;
1388}
1389
1390/*
1391 * The type nl80211_tx_power_setting replaces the following
1392 * data type from 2.6.36 onwards
1393*/
1394static int ath6kl_cfg80211_set_txpower(struct wiphy *wiphy,
1395 enum nl80211_tx_power_setting type,
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001396 int mbm)
Kalle Valobdcd8172011-07-18 00:22:30 +03001397{
1398 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301399 struct ath6kl_vif *vif;
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001400 int dbm = MBM_TO_DBM(mbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001401
1402 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x, dbm %d\n", __func__,
1403 type, dbm);
1404
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301405 vif = ath6kl_vif_first(ar);
1406 if (!vif)
1407 return -EIO;
1408
1409 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001410 return -EIO;
1411
1412 switch (type) {
1413 case NL80211_TX_POWER_AUTOMATIC:
1414 return 0;
1415 case NL80211_TX_POWER_LIMITED:
Kalle Valod0d670a2012-03-07 20:03:58 +02001416 ar->tx_pwr = dbm;
Kalle Valobdcd8172011-07-18 00:22:30 +03001417 break;
1418 default:
1419 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x not supported\n",
1420 __func__, type);
1421 return -EOPNOTSUPP;
1422 }
1423
Kalle Valod0d670a2012-03-07 20:03:58 +02001424 ath6kl_wmi_set_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx, dbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001425
1426 return 0;
1427}
1428
1429static int ath6kl_cfg80211_get_txpower(struct wiphy *wiphy, int *dbm)
1430{
1431 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301432 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001433
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301434 vif = ath6kl_vif_first(ar);
1435 if (!vif)
1436 return -EIO;
1437
1438 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001439 return -EIO;
1440
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301441 if (test_bit(CONNECTED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001442 ar->tx_pwr = 0;
1443
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301444 if (ath6kl_wmi_get_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001445 ath6kl_err("ath6kl_wmi_get_tx_pwr_cmd failed\n");
1446 return -EIO;
1447 }
1448
1449 wait_event_interruptible_timeout(ar->event_wq, ar->tx_pwr != 0,
1450 5 * HZ);
1451
1452 if (signal_pending(current)) {
1453 ath6kl_err("target did not respond\n");
1454 return -EINTR;
1455 }
1456 }
1457
1458 *dbm = ar->tx_pwr;
1459 return 0;
1460}
1461
1462static int ath6kl_cfg80211_set_power_mgmt(struct wiphy *wiphy,
1463 struct net_device *dev,
1464 bool pmgmt, int timeout)
1465{
1466 struct ath6kl *ar = ath6kl_priv(dev);
1467 struct wmi_power_mode_cmd mode;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301468 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001469
1470 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: pmgmt %d, timeout %d\n",
1471 __func__, pmgmt, timeout);
1472
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301473 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001474 return -EIO;
1475
1476 if (pmgmt) {
1477 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: max perf\n", __func__);
1478 mode.pwr_mode = REC_POWER;
1479 } else {
1480 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: rec power\n", __func__);
1481 mode.pwr_mode = MAX_PERF_POWER;
1482 }
1483
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301484 if (ath6kl_wmi_powermode_cmd(ar->wmi, vif->fw_vif_idx,
Kalle Valo96f1fad2012-03-07 20:03:57 +02001485 mode.pwr_mode) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001486 ath6kl_err("wmi_powermode_cmd failed\n");
1487 return -EIO;
1488 }
1489
1490 return 0;
1491}
1492
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301493static struct net_device *ath6kl_cfg80211_add_iface(struct wiphy *wiphy,
1494 char *name,
1495 enum nl80211_iftype type,
1496 u32 *flags,
1497 struct vif_params *params)
1498{
1499 struct ath6kl *ar = wiphy_priv(wiphy);
1500 struct net_device *ndev;
1501 u8 if_idx, nw_type;
1502
Kalle Valo71f96ee2011-11-14 19:31:30 +02001503 if (ar->num_vif == ar->vif_max) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301504 ath6kl_err("Reached maximum number of supported vif\n");
1505 return ERR_PTR(-EINVAL);
1506 }
1507
1508 if (!ath6kl_is_valid_iftype(ar, type, &if_idx, &nw_type)) {
1509 ath6kl_err("Not a supported interface type\n");
1510 return ERR_PTR(-EINVAL);
1511 }
1512
1513 ndev = ath6kl_interface_add(ar, name, type, if_idx, nw_type);
1514 if (!ndev)
1515 return ERR_PTR(-ENOMEM);
1516
1517 ar->num_vif++;
1518
1519 return ndev;
1520}
1521
1522static int ath6kl_cfg80211_del_iface(struct wiphy *wiphy,
1523 struct net_device *ndev)
1524{
1525 struct ath6kl *ar = wiphy_priv(wiphy);
1526 struct ath6kl_vif *vif = netdev_priv(ndev);
1527
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301528 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301529 list_del(&vif->list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301530 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301531
1532 ath6kl_cleanup_vif(vif, test_bit(WMI_READY, &ar->flag));
1533
Kalle Valoc25889e2012-01-17 20:08:27 +02001534 ath6kl_cfg80211_vif_cleanup(vif);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301535
1536 return 0;
1537}
1538
Kalle Valobdcd8172011-07-18 00:22:30 +03001539static int ath6kl_cfg80211_change_iface(struct wiphy *wiphy,
1540 struct net_device *ndev,
1541 enum nl80211_iftype type, u32 *flags,
1542 struct vif_params *params)
1543{
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301544 struct ath6kl_vif *vif = netdev_priv(ndev);
Vasanthakumar Thiagarajan1e8d13b2012-04-06 20:24:30 +05301545 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +03001546
1547 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type %u\n", __func__, type);
1548
Vasanthakumar Thiagarajan1e8d13b2012-04-06 20:24:30 +05301549 /*
1550 * Don't bring up p2p on an interface which is not initialized
1551 * for p2p operation where fw does not have capability to switch
1552 * dynamically between non-p2p and p2p type interface.
1553 */
1554 if (!test_bit(ATH6KL_FW_CAPABILITY_STA_P2PDEV_DUPLEX,
1555 vif->ar->fw_capabilities) &&
1556 (type == NL80211_IFTYPE_P2P_CLIENT ||
1557 type == NL80211_IFTYPE_P2P_GO)) {
1558 if (vif->ar->vif_max == 1) {
1559 if (vif->fw_vif_idx != 0)
1560 return -EINVAL;
1561 else
1562 goto set_iface_type;
1563 }
1564
1565 for (i = vif->ar->max_norm_iface; i < vif->ar->vif_max; i++) {
1566 if (i == vif->fw_vif_idx)
1567 break;
1568 }
1569
1570 if (i == vif->ar->vif_max) {
1571 ath6kl_err("Invalid interface to bring up P2P\n");
1572 return -EINVAL;
1573 }
1574 }
1575
Thomas Pedersenc422d52d2012-05-15 00:09:23 -07001576 /* need to clean up enhanced bmiss detection fw state */
1577 ath6kl_cfg80211_sta_bmiss_enhance(vif, false);
1578
Vasanthakumar Thiagarajan1e8d13b2012-04-06 20:24:30 +05301579set_iface_type:
Kalle Valobdcd8172011-07-18 00:22:30 +03001580 switch (type) {
1581 case NL80211_IFTYPE_STATION:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301582 vif->next_mode = INFRA_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001583 break;
1584 case NL80211_IFTYPE_ADHOC:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301585 vif->next_mode = ADHOC_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001586 break;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001587 case NL80211_IFTYPE_AP:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301588 vif->next_mode = AP_NETWORK;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001589 break;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001590 case NL80211_IFTYPE_P2P_CLIENT:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301591 vif->next_mode = INFRA_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001592 break;
1593 case NL80211_IFTYPE_P2P_GO:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301594 vif->next_mode = AP_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001595 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001596 default:
1597 ath6kl_err("invalid interface type %u\n", type);
1598 return -EOPNOTSUPP;
1599 }
1600
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +05301601 vif->wdev.iftype = type;
Kalle Valobdcd8172011-07-18 00:22:30 +03001602
1603 return 0;
1604}
1605
1606static int ath6kl_cfg80211_join_ibss(struct wiphy *wiphy,
1607 struct net_device *dev,
1608 struct cfg80211_ibss_params *ibss_param)
1609{
1610 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301611 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001612 int status;
1613
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301614 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001615 return -EIO;
1616
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301617 vif->ssid_len = ibss_param->ssid_len;
1618 memcpy(vif->ssid, ibss_param->ssid, vif->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +03001619
1620 if (ibss_param->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301621 vif->ch_hint = ibss_param->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +03001622
1623 if (ibss_param->channel_fixed) {
1624 /*
1625 * TODO: channel_fixed: The channel should be fixed, do not
1626 * search for IBSSs to join on other channels. Target
1627 * firmware does not support this feature, needs to be
1628 * updated.
1629 */
1630 return -EOPNOTSUPP;
1631 }
1632
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301633 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001634 if (ibss_param->bssid && !is_broadcast_ether_addr(ibss_param->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301635 memcpy(vif->req_bssid, ibss_param->bssid,
1636 sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001637
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301638 ath6kl_set_wpa_version(vif, 0);
Kalle Valobdcd8172011-07-18 00:22:30 +03001639
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301640 status = ath6kl_set_auth_type(vif, NL80211_AUTHTYPE_OPEN_SYSTEM);
Kalle Valobdcd8172011-07-18 00:22:30 +03001641 if (status)
1642 return status;
1643
1644 if (ibss_param->privacy) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301645 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, true);
1646 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001647 } else {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301648 ath6kl_set_cipher(vif, 0, true);
1649 ath6kl_set_cipher(vif, 0, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001650 }
1651
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301652 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +03001653
1654 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1655 "%s: connect called with authmode %d dot11 auth %d"
1656 " PW crypto %d PW crypto len %d GRP crypto %d"
1657 " GRP crypto len %d channel hint %u\n",
1658 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301659 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
1660 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301661 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +03001662
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301663 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301664 vif->dot11_auth_mode, vif->auth_mode,
1665 vif->prwise_crypto,
1666 vif->prwise_crypto_len,
1667 vif->grp_crypto, vif->grp_crypto_len,
1668 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301669 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001670 ar->connect_ctrl_flags, SUBTYPE_NONE);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301671 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001672
1673 return 0;
1674}
1675
1676static int ath6kl_cfg80211_leave_ibss(struct wiphy *wiphy,
1677 struct net_device *dev)
1678{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301679 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001680
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301681 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001682 return -EIO;
1683
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301684 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301685 memset(vif->ssid, 0, sizeof(vif->ssid));
1686 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001687
1688 return 0;
1689}
1690
1691static const u32 cipher_suites[] = {
1692 WLAN_CIPHER_SUITE_WEP40,
1693 WLAN_CIPHER_SUITE_WEP104,
1694 WLAN_CIPHER_SUITE_TKIP,
1695 WLAN_CIPHER_SUITE_CCMP,
Jouni Malinen837cb972011-10-11 17:31:57 +03001696 CCKM_KRK_CIPHER_SUITE,
Dai Shuibing5e070212011-11-03 11:39:37 +02001697 WLAN_CIPHER_SUITE_SMS4,
Kalle Valobdcd8172011-07-18 00:22:30 +03001698};
1699
1700static bool is_rate_legacy(s32 rate)
1701{
1702 static const s32 legacy[] = { 1000, 2000, 5500, 11000,
1703 6000, 9000, 12000, 18000, 24000,
1704 36000, 48000, 54000
1705 };
1706 u8 i;
1707
1708 for (i = 0; i < ARRAY_SIZE(legacy); i++)
1709 if (rate == legacy[i])
1710 return true;
1711
1712 return false;
1713}
1714
1715static bool is_rate_ht20(s32 rate, u8 *mcs, bool *sgi)
1716{
1717 static const s32 ht20[] = { 6500, 13000, 19500, 26000, 39000,
1718 52000, 58500, 65000, 72200
1719 };
1720 u8 i;
1721
1722 for (i = 0; i < ARRAY_SIZE(ht20); i++) {
1723 if (rate == ht20[i]) {
1724 if (i == ARRAY_SIZE(ht20) - 1)
1725 /* last rate uses sgi */
1726 *sgi = true;
1727 else
1728 *sgi = false;
1729
1730 *mcs = i;
1731 return true;
1732 }
1733 }
1734 return false;
1735}
1736
1737static bool is_rate_ht40(s32 rate, u8 *mcs, bool *sgi)
1738{
1739 static const s32 ht40[] = { 13500, 27000, 40500, 54000,
1740 81000, 108000, 121500, 135000,
1741 150000
1742 };
1743 u8 i;
1744
1745 for (i = 0; i < ARRAY_SIZE(ht40); i++) {
1746 if (rate == ht40[i]) {
1747 if (i == ARRAY_SIZE(ht40) - 1)
1748 /* last rate uses sgi */
1749 *sgi = true;
1750 else
1751 *sgi = false;
1752
1753 *mcs = i;
1754 return true;
1755 }
1756 }
1757
1758 return false;
1759}
1760
1761static int ath6kl_get_station(struct wiphy *wiphy, struct net_device *dev,
1762 u8 *mac, struct station_info *sinfo)
1763{
1764 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301765 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001766 long left;
1767 bool sgi;
1768 s32 rate;
1769 int ret;
1770 u8 mcs;
1771
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301772 if (memcmp(mac, vif->bssid, ETH_ALEN) != 0)
Kalle Valobdcd8172011-07-18 00:22:30 +03001773 return -ENOENT;
1774
1775 if (down_interruptible(&ar->sem))
1776 return -EBUSY;
1777
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301778 set_bit(STATS_UPDATE_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001779
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301780 ret = ath6kl_wmi_get_stats_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +03001781
1782 if (ret != 0) {
1783 up(&ar->sem);
1784 return -EIO;
1785 }
1786
1787 left = wait_event_interruptible_timeout(ar->event_wq,
1788 !test_bit(STATS_UPDATE_PEND,
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301789 &vif->flags),
Kalle Valobdcd8172011-07-18 00:22:30 +03001790 WMI_TIMEOUT);
1791
1792 up(&ar->sem);
1793
1794 if (left == 0)
1795 return -ETIMEDOUT;
1796 else if (left < 0)
1797 return left;
1798
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301799 if (vif->target_stats.rx_byte) {
1800 sinfo->rx_bytes = vif->target_stats.rx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001801 sinfo->filled |= STATION_INFO_RX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301802 sinfo->rx_packets = vif->target_stats.rx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001803 sinfo->filled |= STATION_INFO_RX_PACKETS;
1804 }
1805
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301806 if (vif->target_stats.tx_byte) {
1807 sinfo->tx_bytes = vif->target_stats.tx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001808 sinfo->filled |= STATION_INFO_TX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301809 sinfo->tx_packets = vif->target_stats.tx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001810 sinfo->filled |= STATION_INFO_TX_PACKETS;
1811 }
1812
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301813 sinfo->signal = vif->target_stats.cs_rssi;
Kalle Valobdcd8172011-07-18 00:22:30 +03001814 sinfo->filled |= STATION_INFO_SIGNAL;
1815
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301816 rate = vif->target_stats.tx_ucast_rate;
Kalle Valobdcd8172011-07-18 00:22:30 +03001817
1818 if (is_rate_legacy(rate)) {
1819 sinfo->txrate.legacy = rate / 100;
1820 } else if (is_rate_ht20(rate, &mcs, &sgi)) {
1821 if (sgi) {
1822 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1823 sinfo->txrate.mcs = mcs - 1;
1824 } else {
1825 sinfo->txrate.mcs = mcs;
1826 }
1827
1828 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1829 } else if (is_rate_ht40(rate, &mcs, &sgi)) {
1830 if (sgi) {
1831 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1832 sinfo->txrate.mcs = mcs - 1;
1833 } else {
1834 sinfo->txrate.mcs = mcs;
1835 }
1836
1837 sinfo->txrate.flags |= RATE_INFO_FLAGS_40_MHZ_WIDTH;
1838 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1839 } else {
Kalle Valo9a730832011-09-27 23:33:28 +03001840 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1841 "invalid rate from stats: %d\n", rate);
1842 ath6kl_debug_war(ar, ATH6KL_WAR_INVALID_RATE);
Kalle Valobdcd8172011-07-18 00:22:30 +03001843 return 0;
1844 }
1845
1846 sinfo->filled |= STATION_INFO_TX_BITRATE;
1847
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301848 if (test_bit(CONNECTED, &vif->flags) &&
1849 test_bit(DTIM_PERIOD_AVAIL, &vif->flags) &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301850 vif->nw_type == INFRA_NETWORK) {
Jouni Malinen32c10872011-09-19 19:15:07 +03001851 sinfo->filled |= STATION_INFO_BSS_PARAM;
1852 sinfo->bss_param.flags = 0;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301853 sinfo->bss_param.dtim_period = vif->assoc_bss_dtim_period;
1854 sinfo->bss_param.beacon_interval = vif->assoc_bss_beacon_int;
Jouni Malinen32c10872011-09-19 19:15:07 +03001855 }
1856
Kalle Valobdcd8172011-07-18 00:22:30 +03001857 return 0;
1858}
1859
1860static int ath6kl_set_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1861 struct cfg80211_pmksa *pmksa)
1862{
1863 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301864 struct ath6kl_vif *vif = netdev_priv(netdev);
1865
1866 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001867 pmksa->pmkid, true);
1868}
1869
1870static int ath6kl_del_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1871 struct cfg80211_pmksa *pmksa)
1872{
1873 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301874 struct ath6kl_vif *vif = netdev_priv(netdev);
1875
1876 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001877 pmksa->pmkid, false);
1878}
1879
1880static int ath6kl_flush_pmksa(struct wiphy *wiphy, struct net_device *netdev)
1881{
1882 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301883 struct ath6kl_vif *vif = netdev_priv(netdev);
1884
1885 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301886 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx,
1887 vif->bssid, NULL, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001888 return 0;
1889}
1890
Raja Manid91e8ee2012-01-30 17:13:10 +05301891static int ath6kl_wow_usr(struct ath6kl *ar, struct ath6kl_vif *vif,
1892 struct cfg80211_wowlan *wow, u32 *filter)
Raja Mani6cb3c712011-11-07 22:52:45 +02001893{
Raja Manid91e8ee2012-01-30 17:13:10 +05301894 int ret, pos;
1895 u8 mask[WOW_MASK_SIZE];
Raja Mani6cb3c712011-11-07 22:52:45 +02001896 u16 i;
Raja Mani6cb3c712011-11-07 22:52:45 +02001897
Raja Manid91e8ee2012-01-30 17:13:10 +05301898 /* Configure the patterns that we received from the user. */
Raja Mani6cb3c712011-11-07 22:52:45 +02001899 for (i = 0; i < wow->n_patterns; i++) {
1900
1901 /*
1902 * Convert given nl80211 specific mask value to equivalent
1903 * driver specific mask value and send it to the chip along
1904 * with patterns. For example, If the mask value defined in
1905 * struct cfg80211_wowlan is 0xA (equivalent binary is 1010),
1906 * then equivalent driver specific mask value is
1907 * "0xFF 0x00 0xFF 0x00".
1908 */
1909 memset(&mask, 0, sizeof(mask));
1910 for (pos = 0; pos < wow->patterns[i].pattern_len; pos++) {
1911 if (wow->patterns[i].mask[pos / 8] & (0x1 << (pos % 8)))
1912 mask[pos] = 0xFF;
1913 }
1914 /*
1915 * Note: Pattern's offset is not passed as part of wowlan
1916 * parameter from CFG layer. So it's always passed as ZERO
1917 * to the firmware. It means, given WOW patterns are always
1918 * matched from the first byte of received pkt in the firmware.
1919 */
1920 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
Raja Manid91e8ee2012-01-30 17:13:10 +05301921 vif->fw_vif_idx, WOW_LIST_ID,
1922 wow->patterns[i].pattern_len,
1923 0 /* pattern offset */,
1924 wow->patterns[i].pattern, mask);
Raja Mani6cb3c712011-11-07 22:52:45 +02001925 if (ret)
1926 return ret;
1927 }
1928
Raja Manid91e8ee2012-01-30 17:13:10 +05301929 if (wow->disconnect)
1930 *filter |= WOW_FILTER_OPTION_NWK_DISASSOC;
1931
1932 if (wow->magic_pkt)
1933 *filter |= WOW_FILTER_OPTION_MAGIC_PACKET;
1934
1935 if (wow->gtk_rekey_failure)
1936 *filter |= WOW_FILTER_OPTION_GTK_ERROR;
1937
1938 if (wow->eap_identity_req)
1939 *filter |= WOW_FILTER_OPTION_EAP_REQ;
1940
1941 if (wow->four_way_handshake)
1942 *filter |= WOW_FILTER_OPTION_8021X_4WAYHS;
1943
1944 return 0;
1945}
1946
1947static int ath6kl_wow_ap(struct ath6kl *ar, struct ath6kl_vif *vif)
1948{
1949 static const u8 unicst_pattern[] = { 0x00, 0x00, 0x00,
1950 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1951 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1952 0x00, 0x08 };
1953 static const u8 unicst_mask[] = { 0x01, 0x00, 0x00,
1954 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1955 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1956 0x00, 0x7f };
1957 u8 unicst_offset = 0;
1958 static const u8 arp_pattern[] = { 0x08, 0x06 };
1959 static const u8 arp_mask[] = { 0xff, 0xff };
1960 u8 arp_offset = 20;
1961 static const u8 discvr_pattern[] = { 0xe0, 0x00, 0x00, 0xf8 };
1962 static const u8 discvr_mask[] = { 0xf0, 0x00, 0x00, 0xf8 };
1963 u8 discvr_offset = 38;
1964 static const u8 dhcp_pattern[] = { 0xff, 0xff, 0xff, 0xff,
1965 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1966 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
1967 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1968 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1969 0x00, 0x00, 0x00, 0x00, 0x00, 0x43 /* port 67 */ };
1970 static const u8 dhcp_mask[] = { 0xff, 0xff, 0xff, 0xff,
1971 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1972 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
1973 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1974 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1975 0x00, 0x00, 0x00, 0x00, 0xff, 0xff /* port 67 */ };
1976 u8 dhcp_offset = 0;
1977 int ret;
1978
1979 /* Setup unicast IP, EAPOL-like and ARP pkt pattern */
1980 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1981 vif->fw_vif_idx, WOW_LIST_ID,
1982 sizeof(unicst_pattern), unicst_offset,
1983 unicst_pattern, unicst_mask);
1984 if (ret) {
1985 ath6kl_err("failed to add WOW unicast IP pattern\n");
1986 return ret;
1987 }
1988
1989 /* Setup all ARP pkt pattern */
1990 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1991 vif->fw_vif_idx, WOW_LIST_ID,
1992 sizeof(arp_pattern), arp_offset,
1993 arp_pattern, arp_mask);
1994 if (ret) {
1995 ath6kl_err("failed to add WOW ARP pattern\n");
1996 return ret;
1997 }
1998
1999 /*
2000 * Setup multicast pattern for mDNS 224.0.0.251,
2001 * SSDP 239.255.255.250 and LLMNR 224.0.0.252
2002 */
2003 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
2004 vif->fw_vif_idx, WOW_LIST_ID,
2005 sizeof(discvr_pattern), discvr_offset,
2006 discvr_pattern, discvr_mask);
2007 if (ret) {
2008 ath6kl_err("failed to add WOW mDNS/SSDP/LLMNR pattern\n");
2009 return ret;
2010 }
2011
2012 /* Setup all DHCP broadcast pkt pattern */
2013 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
2014 vif->fw_vif_idx, WOW_LIST_ID,
2015 sizeof(dhcp_pattern), dhcp_offset,
2016 dhcp_pattern, dhcp_mask);
2017 if (ret) {
2018 ath6kl_err("failed to add WOW DHCP broadcast pattern\n");
2019 return ret;
2020 }
2021
2022 return 0;
2023}
2024
2025static int ath6kl_wow_sta(struct ath6kl *ar, struct ath6kl_vif *vif)
2026{
2027 struct net_device *ndev = vif->ndev;
2028 static const u8 discvr_pattern[] = { 0xe0, 0x00, 0x00, 0xf8 };
2029 static const u8 discvr_mask[] = { 0xf0, 0x00, 0x00, 0xf8 };
2030 u8 discvr_offset = 38;
2031 u8 mac_mask[ETH_ALEN];
2032 int ret;
2033
2034 /* Setup unicast pkt pattern */
2035 memset(mac_mask, 0xff, ETH_ALEN);
2036 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
2037 vif->fw_vif_idx, WOW_LIST_ID,
2038 ETH_ALEN, 0, ndev->dev_addr,
2039 mac_mask);
2040 if (ret) {
2041 ath6kl_err("failed to add WOW unicast pattern\n");
2042 return ret;
2043 }
2044
2045 /*
2046 * Setup multicast pattern for mDNS 224.0.0.251,
2047 * SSDP 239.255.255.250 and LLMNR 224.0.0.252
2048 */
2049 if ((ndev->flags & IFF_ALLMULTI) ||
2050 (ndev->flags & IFF_MULTICAST && netdev_mc_count(ndev) > 0)) {
2051 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
2052 vif->fw_vif_idx, WOW_LIST_ID,
2053 sizeof(discvr_pattern), discvr_offset,
2054 discvr_pattern, discvr_mask);
2055 if (ret) {
Kalle Valocdeb8602012-04-12 11:02:18 +03002056 ath6kl_err("failed to add WOW mDNS/SSDP/LLMNR pattern\n");
Raja Manid91e8ee2012-01-30 17:13:10 +05302057 return ret;
2058 }
2059 }
2060
2061 return 0;
2062}
2063
Raja Mani055bde42012-03-21 15:03:37 +05302064static int is_hsleep_mode_procsed(struct ath6kl_vif *vif)
2065{
2066 return test_bit(HOST_SLEEP_MODE_CMD_PROCESSED, &vif->flags);
2067}
2068
2069static bool is_ctrl_ep_empty(struct ath6kl *ar)
2070{
2071 return !ar->tx_pending[ar->ctrl_ep];
2072}
2073
2074static int ath6kl_cfg80211_host_sleep(struct ath6kl *ar, struct ath6kl_vif *vif)
2075{
2076 int ret, left;
2077
2078 clear_bit(HOST_SLEEP_MODE_CMD_PROCESSED, &vif->flags);
2079
2080 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2081 ATH6KL_HOST_MODE_ASLEEP);
2082 if (ret)
2083 return ret;
2084
2085 left = wait_event_interruptible_timeout(ar->event_wq,
2086 is_hsleep_mode_procsed(vif),
2087 WMI_TIMEOUT);
2088 if (left == 0) {
2089 ath6kl_warn("timeout, didn't get host sleep cmd processed event\n");
2090 ret = -ETIMEDOUT;
2091 } else if (left < 0) {
2092 ath6kl_warn("error while waiting for host sleep cmd processed event %d\n",
2093 left);
2094 ret = left;
2095 }
2096
2097 if (ar->tx_pending[ar->ctrl_ep]) {
2098 left = wait_event_interruptible_timeout(ar->event_wq,
2099 is_ctrl_ep_empty(ar),
2100 WMI_TIMEOUT);
2101 if (left == 0) {
2102 ath6kl_warn("clear wmi ctrl data timeout\n");
2103 ret = -ETIMEDOUT;
2104 } else if (left < 0) {
2105 ath6kl_warn("clear wmi ctrl data failed: %d\n", left);
2106 ret = left;
2107 }
2108 }
2109
2110 return ret;
2111}
2112
Raja Manid91e8ee2012-01-30 17:13:10 +05302113static int ath6kl_wow_suspend(struct ath6kl *ar, struct cfg80211_wowlan *wow)
2114{
2115 struct in_device *in_dev;
2116 struct in_ifaddr *ifa;
2117 struct ath6kl_vif *vif;
Raja Mani055bde42012-03-21 15:03:37 +05302118 int ret;
Raja Manid91e8ee2012-01-30 17:13:10 +05302119 u32 filter = 0;
Raja Manice0dc0c2012-02-20 19:08:08 +05302120 u16 i, bmiss_time;
Raja Manid91e8ee2012-01-30 17:13:10 +05302121 u8 index = 0;
2122 __be32 ips[MAX_IP_ADDRS];
2123
Thomas Pedersen77ed4e42012-04-19 16:29:19 -07002124 /* The FW currently can't support multi-vif WoW properly. */
2125 if (ar->num_vif > 1)
2126 return -EIO;
2127
Raja Manid91e8ee2012-01-30 17:13:10 +05302128 vif = ath6kl_vif_first(ar);
2129 if (!vif)
2130 return -EIO;
2131
2132 if (!ath6kl_cfg80211_ready(vif))
2133 return -EIO;
2134
2135 if (!test_bit(CONNECTED, &vif->flags))
Raja Mani3c411a42012-01-30 17:13:12 +05302136 return -ENOTCONN;
Raja Manid91e8ee2012-01-30 17:13:10 +05302137
2138 if (wow && (wow->n_patterns > WOW_MAX_FILTERS_PER_LIST))
2139 return -EINVAL;
2140
Naveen Gangadharan6821d4f2012-05-11 14:19:09 -07002141 if (!test_bit(NETDEV_MCAST_ALL_ON, &vif->flags) &&
2142 test_bit(ATH6KL_FW_CAPABILITY_WOW_MULTICAST_FILTER,
2143 ar->fw_capabilities)) {
Naveen Gangadharan6251d802012-04-20 12:46:56 -07002144 ret = ath6kl_wmi_mcast_filter_cmd(vif->ar->wmi,
2145 vif->fw_vif_idx, false);
2146 if (ret)
2147 return ret;
2148 }
2149
Raja Manid91e8ee2012-01-30 17:13:10 +05302150 /* Clear existing WOW patterns */
2151 for (i = 0; i < WOW_MAX_FILTERS_PER_LIST; i++)
2152 ath6kl_wmi_del_wow_pattern_cmd(ar->wmi, vif->fw_vif_idx,
2153 WOW_LIST_ID, i);
2154
2155 /*
2156 * Skip the default WOW pattern configuration
2157 * if the driver receives any WOW patterns from
2158 * the user.
2159 */
2160 if (wow)
2161 ret = ath6kl_wow_usr(ar, vif, wow, &filter);
2162 else if (vif->nw_type == AP_NETWORK)
2163 ret = ath6kl_wow_ap(ar, vif);
2164 else
2165 ret = ath6kl_wow_sta(ar, vif);
2166
2167 if (ret)
2168 return ret;
2169
Raja Mani390a8c82012-03-07 11:35:04 +05302170 netif_stop_queue(vif->ndev);
2171
Raja Manice0dc0c2012-02-20 19:08:08 +05302172 if (vif->nw_type != AP_NETWORK) {
2173 ret = ath6kl_wmi_listeninterval_cmd(ar->wmi, vif->fw_vif_idx,
2174 ATH6KL_MAX_WOW_LISTEN_INTL,
2175 0);
2176 if (ret)
2177 return ret;
2178
2179 /* Set listen interval x 15 times as bmiss time */
2180 bmiss_time = ATH6KL_MAX_WOW_LISTEN_INTL * 15;
2181 if (bmiss_time > ATH6KL_MAX_BMISS_TIME)
2182 bmiss_time = ATH6KL_MAX_BMISS_TIME;
2183
2184 ret = ath6kl_wmi_bmisstime_cmd(ar->wmi, vif->fw_vif_idx,
2185 bmiss_time, 0);
2186 if (ret)
2187 return ret;
2188
2189 ret = ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
2190 0xFFFF, 0, 0xFFFF, 0, 0, 0,
2191 0, 0, 0, 0);
2192 if (ret)
2193 return ret;
2194 }
2195
Raja Mani390a8c82012-03-07 11:35:04 +05302196 ar->state = ATH6KL_STATE_SUSPENDING;
2197
Raja Manic08631c2011-12-16 14:24:24 +05302198 /* Setup own IP addr for ARP agent. */
2199 in_dev = __in_dev_get_rtnl(vif->ndev);
2200 if (!in_dev)
2201 goto skip_arp;
2202
2203 ifa = in_dev->ifa_list;
2204 memset(&ips, 0, sizeof(ips));
2205
2206 /* Configure IP addr only if IP address count < MAX_IP_ADDRS */
2207 while (index < MAX_IP_ADDRS && ifa) {
2208 ips[index] = ifa->ifa_local;
2209 ifa = ifa->ifa_next;
2210 index++;
2211 }
2212
2213 if (ifa) {
2214 ath6kl_err("total IP addr count is exceeding fw limit\n");
2215 return -EINVAL;
2216 }
2217
2218 ret = ath6kl_wmi_set_ip_cmd(ar->wmi, vif->fw_vif_idx, ips[0], ips[1]);
2219 if (ret) {
2220 ath6kl_err("fail to setup ip for arp agent\n");
2221 return ret;
2222 }
2223
2224skip_arp:
Raja Mani6cb3c712011-11-07 22:52:45 +02002225 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
2226 ATH6KL_WOW_MODE_ENABLE,
2227 filter,
2228 WOW_HOST_REQ_DELAY);
2229 if (ret)
2230 return ret;
2231
Raja Mani055bde42012-03-21 15:03:37 +05302232 ret = ath6kl_cfg80211_host_sleep(ar, vif);
Raja Mani6cb3c712011-11-07 22:52:45 +02002233 if (ret)
2234 return ret;
2235
Raja Mani055bde42012-03-21 15:03:37 +05302236 return 0;
Raja Mani6cb3c712011-11-07 22:52:45 +02002237}
2238
2239static int ath6kl_wow_resume(struct ath6kl *ar)
2240{
2241 struct ath6kl_vif *vif;
2242 int ret;
2243
2244 vif = ath6kl_vif_first(ar);
2245 if (!vif)
2246 return -EIO;
2247
Raja Mani390a8c82012-03-07 11:35:04 +05302248 ar->state = ATH6KL_STATE_RESUMING;
2249
Raja Mani6cb3c712011-11-07 22:52:45 +02002250 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2251 ATH6KL_HOST_MODE_AWAKE);
Raja Mani390a8c82012-03-07 11:35:04 +05302252 if (ret) {
Kalle Valocdeb8602012-04-12 11:02:18 +03002253 ath6kl_warn("Failed to configure host sleep mode for wow resume: %d\n",
2254 ret);
Raja Mani390a8c82012-03-07 11:35:04 +05302255 ar->state = ATH6KL_STATE_WOW;
2256 return ret;
2257 }
2258
Raja Manice0dc0c2012-02-20 19:08:08 +05302259 if (vif->nw_type != AP_NETWORK) {
2260 ret = ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
2261 0, 0, 0, 0, 0, 0, 3, 0, 0, 0);
2262 if (ret)
2263 return ret;
2264
2265 ret = ath6kl_wmi_listeninterval_cmd(ar->wmi, vif->fw_vif_idx,
2266 vif->listen_intvl_t, 0);
2267 if (ret)
2268 return ret;
2269
2270 ret = ath6kl_wmi_bmisstime_cmd(ar->wmi, vif->fw_vif_idx,
2271 vif->bmiss_time_t, 0);
2272 if (ret)
2273 return ret;
2274 }
2275
Raja Mani390a8c82012-03-07 11:35:04 +05302276 ar->state = ATH6KL_STATE_ON;
2277
Naveen Gangadharan6821d4f2012-05-11 14:19:09 -07002278 if (!test_bit(NETDEV_MCAST_ALL_OFF, &vif->flags) &&
2279 test_bit(ATH6KL_FW_CAPABILITY_WOW_MULTICAST_FILTER,
2280 ar->fw_capabilities)) {
Naveen Gangadharan6251d802012-04-20 12:46:56 -07002281 ret = ath6kl_wmi_mcast_filter_cmd(vif->ar->wmi,
2282 vif->fw_vif_idx, true);
2283 if (ret)
2284 return ret;
2285 }
2286
Raja Mani390a8c82012-03-07 11:35:04 +05302287 netif_wake_queue(vif->ndev);
2288
2289 return 0;
Raja Mani6cb3c712011-11-07 22:52:45 +02002290}
2291
Raja Mani40abc2d2012-03-21 15:03:38 +05302292static int ath6kl_cfg80211_deepsleep_suspend(struct ath6kl *ar)
2293{
2294 struct ath6kl_vif *vif;
2295 int ret;
2296
2297 vif = ath6kl_vif_first(ar);
2298 if (!vif)
2299 return -EIO;
2300
Ming Jiang48f27582012-04-13 21:09:25 +08002301 if (!test_bit(WMI_READY, &ar->flag)) {
2302 ath6kl_err("deepsleep failed as wmi is not ready\n");
Raja Mani40abc2d2012-03-21 15:03:38 +05302303 return -EIO;
Ming Jiang48f27582012-04-13 21:09:25 +08002304 }
Raja Mani40abc2d2012-03-21 15:03:38 +05302305
2306 ath6kl_cfg80211_stop_all(ar);
2307
2308 /* Save the current power mode before enabling power save */
2309 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
2310
2311 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER);
2312 if (ret)
2313 return ret;
2314
2315 /* Disable WOW mode */
2316 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
2317 ATH6KL_WOW_MODE_DISABLE,
2318 0, 0);
2319 if (ret)
2320 return ret;
2321
2322 /* Flush all non control pkts in TX path */
2323 ath6kl_tx_data_cleanup(ar);
2324
2325 ret = ath6kl_cfg80211_host_sleep(ar, vif);
2326 if (ret)
2327 return ret;
2328
2329 return 0;
2330}
2331
2332static int ath6kl_cfg80211_deepsleep_resume(struct ath6kl *ar)
2333{
2334 struct ath6kl_vif *vif;
2335 int ret;
2336
2337 vif = ath6kl_vif_first(ar);
2338
2339 if (!vif)
2340 return -EIO;
2341
2342 if (ar->wmi->pwr_mode != ar->wmi->saved_pwr_mode) {
2343 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0,
2344 ar->wmi->saved_pwr_mode);
2345 if (ret)
2346 return ret;
2347 }
2348
2349 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2350 ATH6KL_HOST_MODE_AWAKE);
2351 if (ret)
2352 return ret;
2353
2354 ar->state = ATH6KL_STATE_ON;
2355
2356 /* Reset scan parameter to default values */
2357 ret = ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
2358 0, 0, 0, 0, 0, 0, 3, 0, 0, 0);
2359 if (ret)
2360 return ret;
2361
2362 return 0;
2363}
2364
Kalle Valo52d81a62011-11-01 08:44:21 +02002365int ath6kl_cfg80211_suspend(struct ath6kl *ar,
Raja Mani0f60e9f2011-11-07 22:52:45 +02002366 enum ath6kl_cfg_suspend_mode mode,
2367 struct cfg80211_wowlan *wow)
Kalle Valo52d81a62011-11-01 08:44:21 +02002368{
Vivek Natarajan3d794992012-03-28 19:21:26 +05302369 struct ath6kl_vif *vif;
Raja Mani390a8c82012-03-07 11:35:04 +05302370 enum ath6kl_state prev_state;
Kalle Valo52d81a62011-11-01 08:44:21 +02002371 int ret;
2372
Kalle Valo52d81a62011-11-01 08:44:21 +02002373 switch (mode) {
Raja Manid7c44e02011-11-07 22:52:46 +02002374 case ATH6KL_CFG_SUSPEND_WOW:
2375
2376 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode suspend\n");
2377
2378 /* Flush all non control pkts in TX path */
2379 ath6kl_tx_data_cleanup(ar);
2380
Raja Mani390a8c82012-03-07 11:35:04 +05302381 prev_state = ar->state;
2382
Raja Manid7c44e02011-11-07 22:52:46 +02002383 ret = ath6kl_wow_suspend(ar, wow);
Raja Mani390a8c82012-03-07 11:35:04 +05302384 if (ret) {
2385 ar->state = prev_state;
Raja Manid7c44e02011-11-07 22:52:46 +02002386 return ret;
Raja Mani390a8c82012-03-07 11:35:04 +05302387 }
Raja Mani1e9a9052012-03-06 15:03:59 +05302388
Raja Manid7c44e02011-11-07 22:52:46 +02002389 ar->state = ATH6KL_STATE_WOW;
2390 break;
2391
Kalle Valo52d81a62011-11-01 08:44:21 +02002392 case ATH6KL_CFG_SUSPEND_DEEPSLEEP:
Raja Mani524441e2011-11-07 22:52:46 +02002393
Raja Mani40abc2d2012-03-21 15:03:38 +05302394 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "deep sleep suspend\n");
Raja Mani524441e2011-11-07 22:52:46 +02002395
Raja Mani40abc2d2012-03-21 15:03:38 +05302396 ret = ath6kl_cfg80211_deepsleep_suspend(ar);
Kalle Valo52d81a62011-11-01 08:44:21 +02002397 if (ret) {
Raja Mani40abc2d2012-03-21 15:03:38 +05302398 ath6kl_err("deepsleep suspend failed: %d\n", ret);
2399 return ret;
Kalle Valo52d81a62011-11-01 08:44:21 +02002400 }
2401
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002402 ar->state = ATH6KL_STATE_DEEPSLEEP;
2403
Kalle Valo52d81a62011-11-01 08:44:21 +02002404 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002405
2406 case ATH6KL_CFG_SUSPEND_CUTPOWER:
Raja Mani524441e2011-11-07 22:52:46 +02002407
Kalle Valo7125f012011-12-13 14:51:37 +02002408 ath6kl_cfg80211_stop_all(ar);
Raja Mani524441e2011-11-07 22:52:46 +02002409
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002410 if (ar->state == ATH6KL_STATE_OFF) {
2411 ath6kl_dbg(ATH6KL_DBG_SUSPEND,
2412 "suspend hw off, no action for cutpower\n");
2413 break;
2414 }
2415
2416 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "suspend cutting power\n");
2417
2418 ret = ath6kl_init_hw_stop(ar);
2419 if (ret) {
2420 ath6kl_warn("failed to stop hw during suspend: %d\n",
2421 ret);
2422 }
2423
2424 ar->state = ATH6KL_STATE_CUTPOWER;
2425
2426 break;
2427
Kalle Valo10509f92011-12-13 14:52:07 +02002428 case ATH6KL_CFG_SUSPEND_SCHED_SCAN:
2429 /*
2430 * Nothing needed for schedule scan, firmware is already in
2431 * wow mode and sleeping most of the time.
2432 */
2433 break;
2434
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002435 default:
2436 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02002437 }
2438
Vivek Natarajan3d794992012-03-28 19:21:26 +05302439 list_for_each_entry(vif, &ar->vif_list, list)
2440 ath6kl_cfg80211_scan_complete_event(vif, true);
2441
Kalle Valo52d81a62011-11-01 08:44:21 +02002442 return 0;
2443}
Kalle Valod6a434d2012-01-17 20:09:36 +02002444EXPORT_SYMBOL(ath6kl_cfg80211_suspend);
Kalle Valo52d81a62011-11-01 08:44:21 +02002445
2446int ath6kl_cfg80211_resume(struct ath6kl *ar)
2447{
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002448 int ret;
2449
2450 switch (ar->state) {
Raja Manid7c44e02011-11-07 22:52:46 +02002451 case ATH6KL_STATE_WOW:
2452 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode resume\n");
2453
2454 ret = ath6kl_wow_resume(ar);
2455 if (ret) {
2456 ath6kl_warn("wow mode resume failed: %d\n", ret);
2457 return ret;
2458 }
2459
Raja Manid7c44e02011-11-07 22:52:46 +02002460 break;
2461
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002462 case ATH6KL_STATE_DEEPSLEEP:
Raja Mani40abc2d2012-03-21 15:03:38 +05302463 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "deep sleep resume\n");
2464
2465 ret = ath6kl_cfg80211_deepsleep_resume(ar);
2466 if (ret) {
2467 ath6kl_warn("deep sleep resume failed: %d\n", ret);
2468 return ret;
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002469 }
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002470 break;
2471
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002472 case ATH6KL_STATE_CUTPOWER:
2473 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "resume restoring power\n");
2474
2475 ret = ath6kl_init_hw_start(ar);
2476 if (ret) {
2477 ath6kl_warn("Failed to boot hw in resume: %d\n", ret);
2478 return ret;
2479 }
Raja Manid7c44e02011-11-07 22:52:46 +02002480 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002481
Kalle Valo10509f92011-12-13 14:52:07 +02002482 case ATH6KL_STATE_SCHED_SCAN:
2483 break;
2484
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002485 default:
2486 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02002487 }
2488
2489 return 0;
2490}
Kalle Valod6a434d2012-01-17 20:09:36 +02002491EXPORT_SYMBOL(ath6kl_cfg80211_resume);
Kalle Valo52d81a62011-11-01 08:44:21 +02002492
Kalle Valoabcb3442011-07-22 08:26:20 +03002493#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02002494
2495/* hif layer decides what suspend mode to use */
2496static int __ath6kl_cfg80211_suspend(struct wiphy *wiphy,
Kalle Valoabcb3442011-07-22 08:26:20 +03002497 struct cfg80211_wowlan *wow)
2498{
2499 struct ath6kl *ar = wiphy_priv(wiphy);
2500
Raja Mani0f60e9f2011-11-07 22:52:45 +02002501 return ath6kl_hif_suspend(ar, wow);
Kalle Valoabcb3442011-07-22 08:26:20 +03002502}
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002503
Kalle Valo52d81a62011-11-01 08:44:21 +02002504static int __ath6kl_cfg80211_resume(struct wiphy *wiphy)
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002505{
2506 struct ath6kl *ar = wiphy_priv(wiphy);
2507
2508 return ath6kl_hif_resume(ar);
2509}
Raja Mania918fb32011-11-07 22:52:46 +02002510
2511/*
2512 * FIXME: WOW suspend mode is selected if the host sdio controller supports
2513 * both sdio irq wake up and keep power. The target pulls sdio data line to
2514 * wake up the host when WOW pattern matches. This causes sdio irq handler
2515 * is being called in the host side which internally hits ath6kl's RX path.
2516 *
2517 * Since sdio interrupt is not disabled, RX path executes even before
2518 * the host executes the actual resume operation from PM module.
2519 *
2520 * In the current scenario, WOW resume should happen before start processing
2521 * any data from the target. So It's required to perform WOW resume in RX path.
2522 * Ideally we should perform WOW resume only in the actual platform
2523 * resume path. This area needs bit rework to avoid WOW resume in RX path.
2524 *
2525 * ath6kl_check_wow_status() is called from ath6kl_rx().
2526 */
2527void ath6kl_check_wow_status(struct ath6kl *ar)
2528{
Raja Mani390a8c82012-03-07 11:35:04 +05302529 if (ar->state == ATH6KL_STATE_SUSPENDING)
2530 return;
2531
Raja Mania918fb32011-11-07 22:52:46 +02002532 if (ar->state == ATH6KL_STATE_WOW)
2533 ath6kl_cfg80211_resume(ar);
2534}
2535
2536#else
2537
2538void ath6kl_check_wow_status(struct ath6kl *ar)
2539{
2540}
Kalle Valoabcb3442011-07-22 08:26:20 +03002541#endif
2542
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302543static int ath6kl_set_htcap(struct ath6kl_vif *vif, enum ieee80211_band band,
2544 bool ht_enable)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002545{
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302546 struct ath6kl_htcap *htcap = &vif->htcap;
Sujith Manoharane68f6752011-12-22 12:15:27 +05302547
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302548 if (htcap->ht_enable == ht_enable)
2549 return 0;
Sujith Manoharane68f6752011-12-22 12:15:27 +05302550
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302551 if (ht_enable) {
2552 /* Set default ht capabilities */
2553 htcap->ht_enable = true;
2554 htcap->cap_info = (band == IEEE80211_BAND_2GHZ) ?
2555 ath6kl_g_htcap : ath6kl_a_htcap;
2556 htcap->ampdu_factor = IEEE80211_HT_MAX_AMPDU_16K;
2557 } else /* Disable ht */
2558 memset(htcap, 0, sizeof(*htcap));
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002559
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302560 return ath6kl_wmi_set_htcap_cmd(vif->ar->wmi, vif->fw_vif_idx,
2561 band, htcap);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002562}
2563
Thomas Pedersen37a2f952012-04-19 15:31:57 -07002564static int ath6kl_restore_htcap(struct ath6kl_vif *vif)
2565{
2566 struct wiphy *wiphy = vif->ar->wiphy;
2567 int band, ret = 0;
2568
2569 for (band = 0; band < IEEE80211_NUM_BANDS; band++) {
2570 if (!wiphy->bands[band])
2571 continue;
2572
2573 ret = ath6kl_set_htcap(vif, band,
2574 wiphy->bands[band]->ht_cap.ht_supported);
2575 if (ret)
2576 return ret;
2577 }
2578
2579 return ret;
2580}
2581
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002582static bool ath6kl_is_p2p_ie(const u8 *pos)
2583{
2584 return pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
2585 pos[2] == 0x50 && pos[3] == 0x6f &&
2586 pos[4] == 0x9a && pos[5] == 0x09;
2587}
2588
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302589static int ath6kl_set_ap_probe_resp_ies(struct ath6kl_vif *vif,
2590 const u8 *ies, size_t ies_len)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002591{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302592 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002593 const u8 *pos;
2594 u8 *buf = NULL;
2595 size_t len = 0;
2596 int ret;
2597
2598 /*
2599 * Filter out P2P IE(s) since they will be included depending on
2600 * the Probe Request frame in ath6kl_send_go_probe_resp().
2601 */
2602
2603 if (ies && ies_len) {
2604 buf = kmalloc(ies_len, GFP_KERNEL);
2605 if (buf == NULL)
2606 return -ENOMEM;
2607 pos = ies;
2608 while (pos + 1 < ies + ies_len) {
2609 if (pos + 2 + pos[1] > ies + ies_len)
2610 break;
2611 if (!ath6kl_is_p2p_ie(pos)) {
2612 memcpy(buf + len, pos, 2 + pos[1]);
2613 len += 2 + pos[1];
2614 }
2615 pos += 2 + pos[1];
2616 }
2617 }
2618
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302619 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2620 WMI_FRAME_PROBE_RESP, buf, len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002621 kfree(buf);
2622 return ret;
2623}
2624
Johannes Berg88600202012-02-13 15:17:18 +01002625static int ath6kl_set_ies(struct ath6kl_vif *vif,
2626 struct cfg80211_beacon_data *info)
2627{
2628 struct ath6kl *ar = vif->ar;
2629 int res;
2630
Aarthi Thiruvengadam17a7b162012-03-08 12:25:02 -08002631 /* this also clears IE in fw if it's not set */
2632 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2633 WMI_FRAME_BEACON,
2634 info->beacon_ies,
2635 info->beacon_ies_len);
2636 if (res)
2637 return res;
Johannes Berg88600202012-02-13 15:17:18 +01002638
Aarthi Thiruvengadam17a7b162012-03-08 12:25:02 -08002639 /* this also clears IE in fw if it's not set */
2640 res = ath6kl_set_ap_probe_resp_ies(vif, info->proberesp_ies,
2641 info->proberesp_ies_len);
2642 if (res)
2643 return res;
Johannes Berg88600202012-02-13 15:17:18 +01002644
Aarthi Thiruvengadam17a7b162012-03-08 12:25:02 -08002645 /* this also clears IE in fw if it's not set */
2646 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2647 WMI_FRAME_ASSOC_RESP,
2648 info->assocresp_ies,
2649 info->assocresp_ies_len);
2650 if (res)
2651 return res;
Johannes Berg88600202012-02-13 15:17:18 +01002652
2653 return 0;
2654}
2655
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302656static int ath6kl_set_channel(struct wiphy *wiphy, struct net_device *dev,
2657 struct ieee80211_channel *chan,
2658 enum nl80211_channel_type channel_type)
2659{
2660 struct ath6kl_vif *vif;
2661
2662 /*
2663 * 'dev' could be NULL if a channel change is required for the hardware
2664 * device itself, instead of a particular VIF.
2665 *
2666 * FIXME: To be handled properly when monitor mode is supported.
2667 */
2668 if (!dev)
2669 return -EBUSY;
2670
2671 vif = netdev_priv(dev);
2672
2673 if (!ath6kl_cfg80211_ready(vif))
2674 return -EIO;
2675
2676 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: center_freq=%u hw_value=%u\n",
2677 __func__, chan->center_freq, chan->hw_value);
2678 vif->next_chan = chan->center_freq;
2679 vif->next_ch_type = channel_type;
2680 vif->next_ch_band = chan->band;
2681
2682 return 0;
2683}
2684
Thomas Pedersenc422d52d2012-05-15 00:09:23 -07002685void ath6kl_cfg80211_sta_bmiss_enhance(struct ath6kl_vif *vif, bool enable)
2686{
2687 int err;
2688
2689 if (WARN_ON(!test_bit(WMI_READY, &vif->ar->flag)))
2690 return;
2691
2692 if (vif->nw_type != INFRA_NETWORK)
2693 return;
2694
2695 if (!test_bit(ATH6KL_FW_CAPABILITY_BMISS_ENHANCE,
2696 vif->ar->fw_capabilities))
2697 return;
2698
2699 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s fw bmiss enhance\n",
2700 enable ? "enable" : "disable");
2701
2702 err = ath6kl_wmi_sta_bmiss_enhance_cmd(vif->ar->wmi,
2703 vif->fw_vif_idx, enable);
2704 if (err)
2705 ath6kl_err("failed to %s enhanced bmiss detection: %d\n",
2706 enable ? "enable" : "disable", err);
2707}
2708
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302709static int ath6kl_get_rsn_capab(struct cfg80211_beacon_data *beacon,
2710 u8 *rsn_capab)
2711{
2712 const u8 *rsn_ie;
2713 size_t rsn_ie_len;
2714 u16 cnt;
2715
2716 if (!beacon->tail)
2717 return -EINVAL;
2718
2719 rsn_ie = cfg80211_find_ie(WLAN_EID_RSN, beacon->tail, beacon->tail_len);
2720 if (!rsn_ie)
2721 return -EINVAL;
2722
2723 rsn_ie_len = *(rsn_ie + 1);
2724 /* skip element id and length */
2725 rsn_ie += 2;
2726
Vasanthakumar Thiagarajan9e8b16d2012-04-10 14:27:47 +05302727 /* skip version */
2728 if (rsn_ie_len < 2)
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302729 return -EINVAL;
Vasanthakumar Thiagarajan9e8b16d2012-04-10 14:27:47 +05302730 rsn_ie += 2;
2731 rsn_ie_len -= 2;
2732
2733 /* skip group cipher suite */
2734 if (rsn_ie_len < 4)
2735 return 0;
2736 rsn_ie += 4;
2737 rsn_ie_len -= 4;
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302738
2739 /* skip pairwise cipher suite */
2740 if (rsn_ie_len < 2)
Vasanthakumar Thiagarajan9e8b16d2012-04-10 14:27:47 +05302741 return 0;
Vasanthakumar Thiagarajan798985c2012-04-10 13:35:47 +05302742 cnt = get_unaligned_le16(rsn_ie);
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302743 rsn_ie += (2 + cnt * 4);
2744 rsn_ie_len -= (2 + cnt * 4);
2745
2746 /* skip akm suite */
2747 if (rsn_ie_len < 2)
Vasanthakumar Thiagarajan9e8b16d2012-04-10 14:27:47 +05302748 return 0;
Vasanthakumar Thiagarajan798985c2012-04-10 13:35:47 +05302749 cnt = get_unaligned_le16(rsn_ie);
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302750 rsn_ie += (2 + cnt * 4);
2751 rsn_ie_len -= (2 + cnt * 4);
2752
2753 if (rsn_ie_len < 2)
Vasanthakumar Thiagarajan9e8b16d2012-04-10 14:27:47 +05302754 return 0;
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302755
2756 memcpy(rsn_capab, rsn_ie, 2);
2757
2758 return 0;
2759}
2760
Johannes Berg88600202012-02-13 15:17:18 +01002761static int ath6kl_start_ap(struct wiphy *wiphy, struct net_device *dev,
2762 struct cfg80211_ap_settings *info)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002763{
2764 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302765 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002766 struct ieee80211_mgmt *mgmt;
Thomas Pedersen67cd22e2012-02-28 15:08:46 -08002767 bool hidden = false;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002768 u8 *ies;
2769 int ies_len;
2770 struct wmi_connect_cmd p;
2771 int res;
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302772 int i, ret;
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302773 u16 rsn_capab = 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002774
Johannes Berg88600202012-02-13 15:17:18 +01002775 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s:\n", __func__);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002776
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302777 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002778 return -EIO;
2779
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302780 if (vif->next_mode != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002781 return -EOPNOTSUPP;
2782
Johannes Berg88600202012-02-13 15:17:18 +01002783 res = ath6kl_set_ies(vif, &info->beacon);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002784
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002785 ar->ap_mode_bkey.valid = false;
2786
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002787 /* TODO:
2788 * info->interval
2789 * info->dtim_period
2790 */
2791
Johannes Berg88600202012-02-13 15:17:18 +01002792 if (info->beacon.head == NULL)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002793 return -EINVAL;
Johannes Berg88600202012-02-13 15:17:18 +01002794 mgmt = (struct ieee80211_mgmt *) info->beacon.head;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002795 ies = mgmt->u.beacon.variable;
Johannes Berg88600202012-02-13 15:17:18 +01002796 if (ies > info->beacon.head + info->beacon.head_len)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002797 return -EINVAL;
Johannes Berg88600202012-02-13 15:17:18 +01002798 ies_len = info->beacon.head + info->beacon.head_len - ies;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002799
2800 if (info->ssid == NULL)
2801 return -EINVAL;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302802 memcpy(vif->ssid, info->ssid, info->ssid_len);
2803 vif->ssid_len = info->ssid_len;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002804 if (info->hidden_ssid != NL80211_HIDDEN_SSID_NOT_IN_USE)
Thomas Pedersen67cd22e2012-02-28 15:08:46 -08002805 hidden = true;
2806
2807 res = ath6kl_wmi_ap_hidden_ssid(ar->wmi, vif->fw_vif_idx, hidden);
2808 if (res)
2809 return res;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002810
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302811 ret = ath6kl_set_auth_type(vif, info->auth_type);
2812 if (ret)
2813 return ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002814
2815 memset(&p, 0, sizeof(p));
2816
2817 for (i = 0; i < info->crypto.n_akm_suites; i++) {
2818 switch (info->crypto.akm_suites[i]) {
2819 case WLAN_AKM_SUITE_8021X:
2820 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2821 p.auth_mode |= WPA_AUTH;
2822 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2823 p.auth_mode |= WPA2_AUTH;
2824 break;
2825 case WLAN_AKM_SUITE_PSK:
2826 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2827 p.auth_mode |= WPA_PSK_AUTH;
2828 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2829 p.auth_mode |= WPA2_PSK_AUTH;
2830 break;
2831 }
2832 }
2833 if (p.auth_mode == 0)
2834 p.auth_mode = NONE_AUTH;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302835 vif->auth_mode = p.auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002836
2837 for (i = 0; i < info->crypto.n_ciphers_pairwise; i++) {
2838 switch (info->crypto.ciphers_pairwise[i]) {
2839 case WLAN_CIPHER_SUITE_WEP40:
2840 case WLAN_CIPHER_SUITE_WEP104:
2841 p.prwise_crypto_type |= WEP_CRYPT;
2842 break;
2843 case WLAN_CIPHER_SUITE_TKIP:
2844 p.prwise_crypto_type |= TKIP_CRYPT;
2845 break;
2846 case WLAN_CIPHER_SUITE_CCMP:
2847 p.prwise_crypto_type |= AES_CRYPT;
2848 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002849 case WLAN_CIPHER_SUITE_SMS4:
2850 p.prwise_crypto_type |= WAPI_CRYPT;
2851 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002852 }
2853 }
Edward Lu229ed6b2011-08-30 21:58:07 +03002854 if (p.prwise_crypto_type == 0) {
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002855 p.prwise_crypto_type = NONE_CRYPT;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302856 ath6kl_set_cipher(vif, 0, true);
Edward Lu229ed6b2011-08-30 21:58:07 +03002857 } else if (info->crypto.n_ciphers_pairwise == 1)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302858 ath6kl_set_cipher(vif, info->crypto.ciphers_pairwise[0], true);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002859
2860 switch (info->crypto.cipher_group) {
2861 case WLAN_CIPHER_SUITE_WEP40:
2862 case WLAN_CIPHER_SUITE_WEP104:
2863 p.grp_crypto_type = WEP_CRYPT;
2864 break;
2865 case WLAN_CIPHER_SUITE_TKIP:
2866 p.grp_crypto_type = TKIP_CRYPT;
2867 break;
2868 case WLAN_CIPHER_SUITE_CCMP:
2869 p.grp_crypto_type = AES_CRYPT;
2870 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002871 case WLAN_CIPHER_SUITE_SMS4:
2872 p.grp_crypto_type = WAPI_CRYPT;
2873 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002874 default:
2875 p.grp_crypto_type = NONE_CRYPT;
2876 break;
2877 }
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302878 ath6kl_set_cipher(vif, info->crypto.cipher_group, false);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002879
2880 p.nw_type = AP_NETWORK;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302881 vif->nw_type = vif->next_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002882
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302883 p.ssid_len = vif->ssid_len;
2884 memcpy(p.ssid, vif->ssid, vif->ssid_len);
2885 p.dot11_auth_mode = vif->dot11_auth_mode;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302886 p.ch = cpu_to_le16(vif->next_chan);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002887
Thirumalai Pachamuthuc1762a32012-01-12 18:21:39 +05302888 /* Enable uAPSD support by default */
2889 res = ath6kl_wmi_ap_set_apsd(ar->wmi, vif->fw_vif_idx, true);
2890 if (res < 0)
2891 return res;
2892
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002893 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_GO) {
2894 p.nw_subtype = SUBTYPE_P2PGO;
2895 } else {
2896 /*
2897 * Due to firmware limitation, it is not possible to
2898 * do P2P mgmt operations in AP mode
2899 */
2900 p.nw_subtype = SUBTYPE_NONE;
2901 }
2902
Vasanthakumar Thiagarajan03bdeb02012-03-21 20:58:39 +05302903 if (info->inactivity_timeout) {
2904 res = ath6kl_wmi_set_inact_period(ar->wmi, vif->fw_vif_idx,
2905 info->inactivity_timeout);
2906 if (res < 0)
2907 return res;
2908 }
2909
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302910 if (ath6kl_set_htcap(vif, vif->next_ch_band,
2911 vif->next_ch_type != NL80211_CHAN_NO_HT))
2912 return -EIO;
2913
Vasanthakumar Thiagarajand97c1212012-04-09 20:51:20 +05302914 /*
2915 * Get the PTKSA replay counter in the RSN IE. Supplicant
2916 * will use the RSN IE in M3 message and firmware has to
2917 * advertise the same in beacon/probe response. Send
2918 * the complete RSN IE capability field to firmware
2919 */
2920 if (!ath6kl_get_rsn_capab(&info->beacon, (u8 *) &rsn_capab) &&
2921 test_bit(ATH6KL_FW_CAPABILITY_RSN_CAP_OVERRIDE,
2922 ar->fw_capabilities)) {
2923 res = ath6kl_wmi_set_ie_cmd(ar->wmi, vif->fw_vif_idx,
2924 WLAN_EID_RSN, WMI_RSN_IE_CAPB,
2925 (const u8 *) &rsn_capab,
2926 sizeof(rsn_capab));
2927 if (res < 0)
2928 return res;
2929 }
2930
Thomas Pedersenc4f78632012-04-06 13:35:48 -07002931 memcpy(&vif->profile, &p, sizeof(p));
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302932 res = ath6kl_wmi_ap_profile_commit(ar->wmi, vif->fw_vif_idx, &p);
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002933 if (res < 0)
2934 return res;
2935
2936 return 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002937}
2938
Johannes Berg88600202012-02-13 15:17:18 +01002939static int ath6kl_change_beacon(struct wiphy *wiphy, struct net_device *dev,
2940 struct cfg80211_beacon_data *beacon)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002941{
Johannes Berg88600202012-02-13 15:17:18 +01002942 struct ath6kl_vif *vif = netdev_priv(dev);
2943
2944 if (!ath6kl_cfg80211_ready(vif))
2945 return -EIO;
2946
2947 if (vif->next_mode != AP_NETWORK)
2948 return -EOPNOTSUPP;
2949
2950 return ath6kl_set_ies(vif, beacon);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002951}
2952
Johannes Berg88600202012-02-13 15:17:18 +01002953static int ath6kl_stop_ap(struct wiphy *wiphy, struct net_device *dev)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002954{
2955 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302956 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002957
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302958 if (vif->nw_type != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002959 return -EOPNOTSUPP;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302960 if (!test_bit(CONNECTED, &vif->flags))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002961 return -ENOTCONN;
2962
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302963 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302964 clear_bit(CONNECTED, &vif->flags);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002965
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05302966 /* Restore ht setting in firmware */
Thomas Pedersen37a2f952012-04-19 15:31:57 -07002967 return ath6kl_restore_htcap(vif);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002968}
2969
Jouni Malinen33e53082011-12-27 11:02:56 +02002970static const u8 bcast_addr[ETH_ALEN] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
2971
2972static int ath6kl_del_station(struct wiphy *wiphy, struct net_device *dev,
2973 u8 *mac)
2974{
2975 struct ath6kl *ar = ath6kl_priv(dev);
2976 struct ath6kl_vif *vif = netdev_priv(dev);
2977 const u8 *addr = mac ? mac : bcast_addr;
2978
2979 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx, WMI_AP_DEAUTH,
2980 addr, WLAN_REASON_PREV_AUTH_NOT_VALID);
2981}
2982
Jouni Malinen23875132011-08-30 21:57:53 +03002983static int ath6kl_change_station(struct wiphy *wiphy, struct net_device *dev,
2984 u8 *mac, struct station_parameters *params)
2985{
2986 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302987 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen23875132011-08-30 21:57:53 +03002988
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302989 if (vif->nw_type != AP_NETWORK)
Jouni Malinen23875132011-08-30 21:57:53 +03002990 return -EOPNOTSUPP;
2991
2992 /* Use this only for authorizing/unauthorizing a station */
2993 if (!(params->sta_flags_mask & BIT(NL80211_STA_FLAG_AUTHORIZED)))
2994 return -EOPNOTSUPP;
2995
2996 if (params->sta_flags_set & BIT(NL80211_STA_FLAG_AUTHORIZED))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302997 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2998 WMI_AP_MLME_AUTHORIZE, mac, 0);
2999 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
3000 WMI_AP_MLME_UNAUTHORIZE, mac, 0);
Jouni Malinen23875132011-08-30 21:57:53 +03003001}
3002
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003003static int ath6kl_remain_on_channel(struct wiphy *wiphy,
3004 struct net_device *dev,
3005 struct ieee80211_channel *chan,
3006 enum nl80211_channel_type channel_type,
3007 unsigned int duration,
3008 u64 *cookie)
3009{
3010 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303011 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen10522612011-10-27 16:00:13 +03003012 u32 id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003013
3014 /* TODO: if already pending or ongoing remain-on-channel,
3015 * return -EBUSY */
Jouni Malinen10522612011-10-27 16:00:13 +03003016 id = ++vif->last_roc_id;
3017 if (id == 0) {
3018 /* Do not use 0 as the cookie value */
3019 id = ++vif->last_roc_id;
3020 }
3021 *cookie = id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003022
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303023 return ath6kl_wmi_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx,
3024 chan->center_freq, duration);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003025}
3026
3027static int ath6kl_cancel_remain_on_channel(struct wiphy *wiphy,
3028 struct net_device *dev,
3029 u64 cookie)
3030{
3031 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303032 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003033
Jouni Malinen10522612011-10-27 16:00:13 +03003034 if (cookie != vif->last_roc_id)
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003035 return -ENOENT;
Jouni Malinen10522612011-10-27 16:00:13 +03003036 vif->last_cancel_roc_id = cookie;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003037
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303038 return ath6kl_wmi_cancel_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003039}
3040
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303041static int ath6kl_send_go_probe_resp(struct ath6kl_vif *vif,
3042 const u8 *buf, size_t len,
3043 unsigned int freq)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003044{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303045 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003046 const u8 *pos;
3047 u8 *p2p;
3048 int p2p_len;
3049 int ret;
3050 const struct ieee80211_mgmt *mgmt;
3051
3052 mgmt = (const struct ieee80211_mgmt *) buf;
3053
3054 /* Include P2P IE(s) from the frame generated in user space. */
3055
3056 p2p = kmalloc(len, GFP_KERNEL);
3057 if (p2p == NULL)
3058 return -ENOMEM;
3059 p2p_len = 0;
3060
3061 pos = mgmt->u.probe_resp.variable;
3062 while (pos + 1 < buf + len) {
3063 if (pos + 2 + pos[1] > buf + len)
3064 break;
3065 if (ath6kl_is_p2p_ie(pos)) {
3066 memcpy(p2p + p2p_len, pos, 2 + pos[1]);
3067 p2p_len += 2 + pos[1];
3068 }
3069 pos += 2 + pos[1];
3070 }
3071
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303072 ret = ath6kl_wmi_send_probe_response_cmd(ar->wmi, vif->fw_vif_idx, freq,
3073 mgmt->da, p2p, p2p_len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003074 kfree(p2p);
3075 return ret;
3076}
3077
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08003078static bool ath6kl_mgmt_powersave_ap(struct ath6kl_vif *vif,
3079 u32 id,
3080 u32 freq,
3081 u32 wait,
3082 const u8 *buf,
3083 size_t len,
3084 bool *more_data,
3085 bool no_cck)
3086{
3087 struct ieee80211_mgmt *mgmt;
3088 struct ath6kl_sta *conn;
3089 bool is_psq_empty = false;
3090 struct ath6kl_mgmt_buff *mgmt_buf;
3091 size_t mgmt_buf_size;
3092 struct ath6kl *ar = vif->ar;
3093
3094 mgmt = (struct ieee80211_mgmt *) buf;
3095 if (is_multicast_ether_addr(mgmt->da))
3096 return false;
3097
3098 conn = ath6kl_find_sta(vif, mgmt->da);
3099 if (!conn)
3100 return false;
3101
3102 if (conn->sta_flags & STA_PS_SLEEP) {
3103 if (!(conn->sta_flags & STA_PS_POLLED)) {
3104 /* Queue the frames if the STA is sleeping */
3105 mgmt_buf_size = len + sizeof(struct ath6kl_mgmt_buff);
3106 mgmt_buf = kmalloc(mgmt_buf_size, GFP_KERNEL);
3107 if (!mgmt_buf)
3108 return false;
3109
3110 INIT_LIST_HEAD(&mgmt_buf->list);
3111 mgmt_buf->id = id;
3112 mgmt_buf->freq = freq;
3113 mgmt_buf->wait = wait;
3114 mgmt_buf->len = len;
3115 mgmt_buf->no_cck = no_cck;
3116 memcpy(mgmt_buf->buf, buf, len);
3117 spin_lock_bh(&conn->psq_lock);
3118 is_psq_empty = skb_queue_empty(&conn->psq) &&
3119 (conn->mgmt_psq_len == 0);
3120 list_add_tail(&mgmt_buf->list, &conn->mgmt_psq);
3121 conn->mgmt_psq_len++;
3122 spin_unlock_bh(&conn->psq_lock);
3123
3124 /*
3125 * If this is the first pkt getting queued
3126 * for this STA, update the PVB for this
3127 * STA.
3128 */
3129 if (is_psq_empty)
3130 ath6kl_wmi_set_pvb_cmd(ar->wmi, vif->fw_vif_idx,
3131 conn->aid, 1);
3132 return true;
3133 }
3134
3135 /*
3136 * This tx is because of a PsPoll.
3137 * Determine if MoreData bit has to be set.
3138 */
3139 spin_lock_bh(&conn->psq_lock);
3140 if (!skb_queue_empty(&conn->psq) || (conn->mgmt_psq_len != 0))
3141 *more_data = true;
3142 spin_unlock_bh(&conn->psq_lock);
3143 }
3144
3145 return false;
3146}
3147
Aarthi Thiruvengadamc86e4f42012-03-15 14:34:56 -07003148/* Check if SSID length is greater than DIRECT- */
3149static bool ath6kl_is_p2p_go_ssid(const u8 *buf, size_t len)
3150{
3151 const struct ieee80211_mgmt *mgmt;
3152 mgmt = (const struct ieee80211_mgmt *) buf;
3153
3154 /* variable[1] contains the SSID tag length */
3155 if (buf + len >= &mgmt->u.probe_resp.variable[1] &&
3156 (mgmt->u.probe_resp.variable[1] > P2P_WILDCARD_SSID_LEN)) {
3157 return true;
3158 }
3159
3160 return false;
3161}
3162
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003163static int ath6kl_mgmt_tx(struct wiphy *wiphy, struct net_device *dev,
3164 struct ieee80211_channel *chan, bool offchan,
3165 enum nl80211_channel_type channel_type,
3166 bool channel_type_valid, unsigned int wait,
Johannes Berge247bd902011-11-04 11:18:21 +01003167 const u8 *buf, size_t len, bool no_cck,
3168 bool dont_wait_for_ack, u64 *cookie)
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003169{
3170 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05303171 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003172 u32 id;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003173 const struct ieee80211_mgmt *mgmt;
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08003174 bool more_data, queued;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003175
3176 mgmt = (const struct ieee80211_mgmt *) buf;
Aarthi Thiruvengadamc86e4f42012-03-15 14:34:56 -07003177 if (vif->nw_type == AP_NETWORK && test_bit(CONNECTED, &vif->flags) &&
3178 ieee80211_is_probe_resp(mgmt->frame_control) &&
3179 ath6kl_is_p2p_go_ssid(buf, len)) {
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003180 /*
Aarthi Thiruvengadamc86e4f42012-03-15 14:34:56 -07003181 * Send Probe Response frame in GO mode using a separate WMI
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003182 * command to allow the target to fill in the generic IEs.
3183 */
3184 *cookie = 0; /* TX status not supported */
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303185 return ath6kl_send_go_probe_resp(vif, buf, len,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03003186 chan->center_freq);
3187 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003188
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05303189 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003190 if (id == 0) {
3191 /*
3192 * 0 is a reserved value in the WMI command and shall not be
3193 * used for the command.
3194 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05303195 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003196 }
3197
3198 *cookie = id;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08003199
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08003200 /* AP mode Power saving processing */
3201 if (vif->nw_type == AP_NETWORK) {
3202 queued = ath6kl_mgmt_powersave_ap(vif,
3203 id, chan->center_freq,
3204 wait, buf,
3205 len, &more_data, no_cck);
3206 if (queued)
3207 return 0;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08003208 }
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08003209
3210 return ath6kl_wmi_send_mgmt_cmd(ar->wmi, vif->fw_vif_idx, id,
3211 chan->center_freq, wait,
3212 buf, len, no_cck);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003213}
3214
Jouni Malinenae32c302011-08-30 21:58:01 +03003215static void ath6kl_mgmt_frame_register(struct wiphy *wiphy,
3216 struct net_device *dev,
3217 u16 frame_type, bool reg)
3218{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05303219 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinenae32c302011-08-30 21:58:01 +03003220
3221 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: frame_type=0x%x reg=%d\n",
3222 __func__, frame_type, reg);
3223 if (frame_type == IEEE80211_STYPE_PROBE_REQ) {
3224 /*
3225 * Note: This notification callback is not allowed to sleep, so
3226 * we cannot send WMI_PROBE_REQ_REPORT_CMD here. Instead, we
3227 * hardcode target to report Probe Request frames all the time.
3228 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05303229 vif->probe_req_report = reg;
Jouni Malinenae32c302011-08-30 21:58:01 +03003230 }
3231}
3232
Kalle Valo10509f92011-12-13 14:52:07 +02003233static int ath6kl_cfg80211_sscan_start(struct wiphy *wiphy,
3234 struct net_device *dev,
3235 struct cfg80211_sched_scan_request *request)
3236{
3237 struct ath6kl *ar = ath6kl_priv(dev);
3238 struct ath6kl_vif *vif = netdev_priv(dev);
3239 u16 interval;
3240 int ret;
Kalle Valo10509f92011-12-13 14:52:07 +02003241
3242 if (ar->state != ATH6KL_STATE_ON)
3243 return -EIO;
3244
3245 if (vif->sme_state != SME_DISCONNECTED)
3246 return -EBUSY;
3247
Thomas Pedersen77ed4e42012-04-19 16:29:19 -07003248 /* The FW currently can't support multi-vif WoW properly. */
3249 if (ar->num_vif > 1)
3250 return -EIO;
3251
Kalle Valob4d13d32012-03-21 10:01:09 +02003252 ath6kl_cfg80211_scan_complete_event(vif, true);
3253
Jouni Malinen3b8ffc62012-04-16 19:28:03 +03003254 ret = ath6kl_set_probed_ssids(ar, vif, request->ssids,
Naveen Singhdd45b752012-05-16 13:29:00 +03003255 request->n_ssids,
3256 request->match_sets,
3257 request->n_match_sets);
Jouni Malinen3b8ffc62012-04-16 19:28:03 +03003258 if (ret < 0)
3259 return ret;
Kalle Valo10509f92011-12-13 14:52:07 +02003260
Naveen Singhdd45b752012-05-16 13:29:00 +03003261 if (!request->n_match_sets) {
3262 ret = ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
3263 ALL_BSS_FILTER, 0);
3264 if (ret < 0)
3265 return ret;
3266 } else {
3267 ret = ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
3268 MATCHED_SSID_FILTER, 0);
3269 if (ret < 0)
3270 return ret;
3271 }
3272
Kalle Valo10509f92011-12-13 14:52:07 +02003273 /* fw uses seconds, also make sure that it's >0 */
3274 interval = max_t(u16, 1, request->interval / 1000);
3275
3276 ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
3277 interval, interval,
Subramania Sharma Thandaveswarand472b5e2012-04-16 16:09:57 +05303278 vif->bg_scan_period, 0, 0, 0, 3, 0, 0, 0);
Kalle Valo10509f92011-12-13 14:52:07 +02003279
Kalle Valo10509f92011-12-13 14:52:07 +02003280 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
3281 ATH6KL_WOW_MODE_ENABLE,
3282 WOW_FILTER_SSID,
3283 WOW_HOST_REQ_DELAY);
3284 if (ret) {
3285 ath6kl_warn("Failed to enable wow with ssid filter: %d\n", ret);
3286 return ret;
3287 }
3288
3289 /* this also clears IE in fw if it's not set */
3290 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
3291 WMI_FRAME_PROBE_REQ,
3292 request->ie, request->ie_len);
3293 if (ret) {
3294 ath6kl_warn("Failed to set probe request IE for scheduled scan: %d",
3295 ret);
3296 return ret;
3297 }
3298
3299 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
3300 ATH6KL_HOST_MODE_ASLEEP);
3301 if (ret) {
3302 ath6kl_warn("Failed to enable host sleep mode for sched scan: %d\n",
3303 ret);
3304 return ret;
3305 }
3306
3307 ar->state = ATH6KL_STATE_SCHED_SCAN;
3308
3309 return ret;
3310}
3311
3312static int ath6kl_cfg80211_sscan_stop(struct wiphy *wiphy,
3313 struct net_device *dev)
3314{
3315 struct ath6kl_vif *vif = netdev_priv(dev);
3316 bool stopped;
3317
3318 stopped = __ath6kl_cfg80211_sscan_stop(vif);
3319
3320 if (!stopped)
3321 return -EIO;
3322
3323 return 0;
3324}
3325
Jouni Malinenf80574a2011-08-30 21:58:04 +03003326static const struct ieee80211_txrx_stypes
3327ath6kl_mgmt_stypes[NUM_NL80211_IFTYPES] = {
3328 [NL80211_IFTYPE_STATION] = {
3329 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3330 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
3331 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3332 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
3333 },
Jouni Malinenba1f6fe2011-12-27 11:03:53 +02003334 [NL80211_IFTYPE_AP] = {
3335 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3336 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
3337 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3338 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
3339 },
Jouni Malinenf80574a2011-08-30 21:58:04 +03003340 [NL80211_IFTYPE_P2P_CLIENT] = {
3341 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3342 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
3343 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3344 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
3345 },
3346 [NL80211_IFTYPE_P2P_GO] = {
3347 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3348 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
3349 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
3350 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
3351 },
3352};
3353
Kalle Valobdcd8172011-07-18 00:22:30 +03003354static struct cfg80211_ops ath6kl_cfg80211_ops = {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303355 .add_virtual_intf = ath6kl_cfg80211_add_iface,
3356 .del_virtual_intf = ath6kl_cfg80211_del_iface,
Kalle Valobdcd8172011-07-18 00:22:30 +03003357 .change_virtual_intf = ath6kl_cfg80211_change_iface,
3358 .scan = ath6kl_cfg80211_scan,
3359 .connect = ath6kl_cfg80211_connect,
3360 .disconnect = ath6kl_cfg80211_disconnect,
3361 .add_key = ath6kl_cfg80211_add_key,
3362 .get_key = ath6kl_cfg80211_get_key,
3363 .del_key = ath6kl_cfg80211_del_key,
3364 .set_default_key = ath6kl_cfg80211_set_default_key,
3365 .set_wiphy_params = ath6kl_cfg80211_set_wiphy_params,
3366 .set_tx_power = ath6kl_cfg80211_set_txpower,
3367 .get_tx_power = ath6kl_cfg80211_get_txpower,
3368 .set_power_mgmt = ath6kl_cfg80211_set_power_mgmt,
3369 .join_ibss = ath6kl_cfg80211_join_ibss,
3370 .leave_ibss = ath6kl_cfg80211_leave_ibss,
3371 .get_station = ath6kl_get_station,
3372 .set_pmksa = ath6kl_set_pmksa,
3373 .del_pmksa = ath6kl_del_pmksa,
3374 .flush_pmksa = ath6kl_flush_pmksa,
Kalle Valo003353b0d2011-09-01 10:14:21 +03003375 CFG80211_TESTMODE_CMD(ath6kl_tm_cmd)
Kalle Valoabcb3442011-07-22 08:26:20 +03003376#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02003377 .suspend = __ath6kl_cfg80211_suspend,
3378 .resume = __ath6kl_cfg80211_resume,
Kalle Valoabcb3442011-07-22 08:26:20 +03003379#endif
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03003380 .set_channel = ath6kl_set_channel,
Johannes Berg88600202012-02-13 15:17:18 +01003381 .start_ap = ath6kl_start_ap,
3382 .change_beacon = ath6kl_change_beacon,
3383 .stop_ap = ath6kl_stop_ap,
Jouni Malinen33e53082011-12-27 11:02:56 +02003384 .del_station = ath6kl_del_station,
Jouni Malinen23875132011-08-30 21:57:53 +03003385 .change_station = ath6kl_change_station,
Jouni Malinen63fa1e02011-08-30 21:57:55 +03003386 .remain_on_channel = ath6kl_remain_on_channel,
3387 .cancel_remain_on_channel = ath6kl_cancel_remain_on_channel,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03003388 .mgmt_tx = ath6kl_mgmt_tx,
Jouni Malinenae32c302011-08-30 21:58:01 +03003389 .mgmt_frame_register = ath6kl_mgmt_frame_register,
Kalle Valo10509f92011-12-13 14:52:07 +02003390 .sched_scan_start = ath6kl_cfg80211_sscan_start,
3391 .sched_scan_stop = ath6kl_cfg80211_sscan_stop,
Kalle Valobdcd8172011-07-18 00:22:30 +03003392};
3393
Kalle Valo7125f012011-12-13 14:51:37 +02003394void ath6kl_cfg80211_stop(struct ath6kl_vif *vif)
Kalle Valoec4b7f62011-11-01 08:44:04 +02003395{
Kalle Valo10509f92011-12-13 14:52:07 +02003396 ath6kl_cfg80211_sscan_disable(vif);
3397
Kalle Valoec4b7f62011-11-01 08:44:04 +02003398 switch (vif->sme_state) {
Kalle Valoc97a31b2011-12-13 14:51:10 +02003399 case SME_DISCONNECTED:
3400 break;
Kalle Valoec4b7f62011-11-01 08:44:04 +02003401 case SME_CONNECTING:
3402 cfg80211_connect_result(vif->ndev, vif->bssid, NULL, 0,
3403 NULL, 0,
3404 WLAN_STATUS_UNSPECIFIED_FAILURE,
3405 GFP_KERNEL);
3406 break;
3407 case SME_CONNECTED:
Kalle Valoec4b7f62011-11-01 08:44:04 +02003408 cfg80211_disconnected(vif->ndev, 0, NULL, 0, GFP_KERNEL);
3409 break;
3410 }
3411
3412 if (test_bit(CONNECTED, &vif->flags) ||
3413 test_bit(CONNECT_PEND, &vif->flags))
Kalle Valo7125f012011-12-13 14:51:37 +02003414 ath6kl_wmi_disconnect_cmd(vif->ar->wmi, vif->fw_vif_idx);
Kalle Valoec4b7f62011-11-01 08:44:04 +02003415
3416 vif->sme_state = SME_DISCONNECTED;
Kalle Valo1f40525512011-11-01 08:44:13 +02003417 clear_bit(CONNECTED, &vif->flags);
3418 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valoec4b7f62011-11-01 08:44:04 +02003419
3420 /* disable scanning */
Kalle Valo7125f012011-12-13 14:51:37 +02003421 if (ath6kl_wmi_scanparams_cmd(vif->ar->wmi, vif->fw_vif_idx, 0xFFFF,
3422 0, 0, 0, 0, 0, 0, 0, 0, 0) != 0)
3423 ath6kl_warn("failed to disable scan during stop\n");
Kalle Valoec4b7f62011-11-01 08:44:04 +02003424
3425 ath6kl_cfg80211_scan_complete_event(vif, true);
3426}
3427
Kalle Valo7125f012011-12-13 14:51:37 +02003428void ath6kl_cfg80211_stop_all(struct ath6kl *ar)
3429{
3430 struct ath6kl_vif *vif;
3431
3432 vif = ath6kl_vif_first(ar);
3433 if (!vif) {
3434 /* save the current power mode before enabling power save */
3435 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
3436
3437 if (ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER) != 0)
Kalle Valocdeb8602012-04-12 11:02:18 +03003438 ath6kl_warn("ath6kl_deep_sleep_enable: wmi_powermode_cmd failed\n");
Kalle Valo7125f012011-12-13 14:51:37 +02003439 return;
3440 }
3441
3442 /*
3443 * FIXME: we should take ar->list_lock to protect changes in the
3444 * vif_list, but that's not trivial to do as ath6kl_cfg80211_stop()
3445 * sleeps.
3446 */
3447 list_for_each_entry(vif, &ar->vif_list, list)
3448 ath6kl_cfg80211_stop(vif);
3449}
3450
Kalle Valoc25889e2012-01-17 20:08:27 +02003451static int ath6kl_cfg80211_vif_init(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +03003452{
Vasanthakumar Thiagarajan7baef812012-01-21 15:22:50 +05303453 vif->aggr_cntxt = aggr_init(vif);
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05303454 if (!vif->aggr_cntxt) {
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303455 ath6kl_err("failed to initialize aggr\n");
3456 return -ENOMEM;
3457 }
Kalle Valobdcd8172011-07-18 00:22:30 +03003458
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05303459 setup_timer(&vif->disconnect_timer, disconnect_timer_handler,
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05303460 (unsigned long) vif->ndev);
Kalle Valo10509f92011-12-13 14:52:07 +02003461 setup_timer(&vif->sched_scan_timer, ath6kl_wmi_sscan_timer,
3462 (unsigned long) vif);
3463
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05303464 set_bit(WMM_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan478ac022011-10-25 19:34:19 +05303465 spin_lock_init(&vif->if_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303466
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05303467 INIT_LIST_HEAD(&vif->mc_filter);
3468
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303469 return 0;
3470}
3471
Kalle Valoc25889e2012-01-17 20:08:27 +02003472void ath6kl_cfg80211_vif_cleanup(struct ath6kl_vif *vif)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303473{
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303474 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05303475 struct ath6kl_mc_filter *mc_filter, *tmp;
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303476
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05303477 aggr_module_destroy(vif->aggr_cntxt);
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05303478
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303479 ar->avail_idx_map |= BIT(vif->fw_vif_idx);
3480
3481 if (vif->nw_type == ADHOC_NETWORK)
3482 ar->ibss_if_active = false;
3483
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05303484 list_for_each_entry_safe(mc_filter, tmp, &vif->mc_filter, list) {
3485 list_del(&mc_filter->list);
3486 kfree(mc_filter);
3487 }
3488
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05303489 unregister_netdevice(vif->ndev);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303490
3491 ar->num_vif--;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303492}
3493
3494struct net_device *ath6kl_interface_add(struct ath6kl *ar, char *name,
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303495 enum nl80211_iftype type, u8 fw_vif_idx,
3496 u8 nw_type)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303497{
3498 struct net_device *ndev;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05303499 struct ath6kl_vif *vif;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303500
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303501 ndev = alloc_netdev(sizeof(*vif), name, ether_setup);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303502 if (!ndev)
3503 return NULL;
3504
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05303505 vif = netdev_priv(ndev);
3506 ndev->ieee80211_ptr = &vif->wdev;
3507 vif->wdev.wiphy = ar->wiphy;
3508 vif->ar = ar;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05303509 vif->ndev = ndev;
3510 SET_NETDEV_DEV(ndev, wiphy_dev(vif->wdev.wiphy));
3511 vif->wdev.netdev = ndev;
3512 vif->wdev.iftype = type;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05303513 vif->fw_vif_idx = fw_vif_idx;
Kalle Valod0d670a2012-03-07 20:03:58 +02003514 vif->nw_type = nw_type;
3515 vif->next_mode = nw_type;
Raja Mani8f46fcc2012-02-20 19:08:07 +05303516 vif->listen_intvl_t = ATH6KL_DEFAULT_LISTEN_INTVAL;
Raja Manice0dc0c2012-02-20 19:08:08 +05303517 vif->bmiss_time_t = ATH6KL_DEFAULT_BMISS_TIME;
Raja Manieb389872012-04-16 16:09:56 +05303518 vif->bg_scan_period = 0;
Vasanthakumar Thiagarajandf90b362012-04-09 19:03:58 +05303519 vif->htcap.ht_enable = true;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303520
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303521 memcpy(ndev->dev_addr, ar->mac_addr, ETH_ALEN);
3522 if (fw_vif_idx != 0)
3523 ndev->dev_addr[0] = (ndev->dev_addr[0] ^ (1 << fw_vif_idx)) |
3524 0x2;
3525
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303526 init_netdev(ndev);
3527
Vasanthakumar Thiagarajane29f25f2011-10-25 19:34:15 +05303528 ath6kl_init_control_info(vif);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303529
Kalle Valoc25889e2012-01-17 20:08:27 +02003530 if (ath6kl_cfg80211_vif_init(vif))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303531 goto err;
3532
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05303533 if (register_netdevice(ndev))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303534 goto err;
3535
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303536 ar->avail_idx_map &= ~BIT(fw_vif_idx);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05303537 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05303538 set_bit(WLAN_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303539 ar->wlan_pwr_state = WLAN_POWER_STATE_ON;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05303540 set_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303541
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303542 if (type == NL80211_IFTYPE_ADHOC)
3543 ar->ibss_if_active = true;
3544
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05303545 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05303546 list_add_tail(&vif->list, &ar->vif_list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05303547 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05303548
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303549 return ndev;
3550
3551err:
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05303552 aggr_module_destroy(vif->aggr_cntxt);
3553 free_netdev(ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303554 return NULL;
3555}
3556
Kalle Valo46d33a22012-01-17 20:08:40 +02003557int ath6kl_cfg80211_init(struct ath6kl *ar)
3558{
3559 struct wiphy *wiphy = ar->wiphy;
Thomas Pedersend92917e2012-04-19 15:31:56 -07003560 bool band_2gig = false, band_5gig = false, ht = false;
Kalle Valo46d33a22012-01-17 20:08:40 +02003561 int ret;
3562
3563 wiphy->mgmt_stypes = ath6kl_mgmt_stypes;
3564
3565 wiphy->max_remain_on_channel_duration = 5000;
3566
3567 /* set device pointer for wiphy */
3568 set_wiphy_dev(wiphy, ar->dev);
3569
3570 wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
3571 BIT(NL80211_IFTYPE_ADHOC) |
3572 BIT(NL80211_IFTYPE_AP);
3573 if (ar->p2p) {
3574 wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_GO) |
3575 BIT(NL80211_IFTYPE_P2P_CLIENT);
3576 }
3577
3578 /* max num of ssids that can be probed during scanning */
Jouni Malinen8ab54152012-05-09 22:14:51 +03003579 wiphy->max_scan_ssids = MAX_PROBED_SSIDS;
Naveen Singhdd45b752012-05-16 13:29:00 +03003580
3581 /* max num of ssids that can be matched after scan */
3582 if (test_bit(ATH6KL_FW_CAPABILITY_SCHED_SCAN_MATCH_LIST,
3583 ar->fw_capabilities))
3584 wiphy->max_match_sets = MAX_PROBED_SSIDS;
3585
Kalle Valo46d33a22012-01-17 20:08:40 +02003586 wiphy->max_scan_ie_len = 1000; /* FIX: what is correct limit? */
Thomas Pedersend92917e2012-04-19 15:31:56 -07003587 switch (ar->hw.cap) {
3588 case WMI_11AN_CAP:
3589 ht = true;
3590 case WMI_11A_CAP:
3591 band_5gig = true;
3592 break;
3593 case WMI_11GN_CAP:
3594 ht = true;
3595 case WMI_11G_CAP:
3596 band_2gig = true;
3597 break;
3598 case WMI_11AGN_CAP:
3599 ht = true;
3600 case WMI_11AG_CAP:
3601 band_2gig = true;
3602 band_5gig = true;
3603 break;
3604 default:
3605 ath6kl_err("invalid phy capability!\n");
3606 return -EINVAL;
3607 }
3608
Vasanthakumar Thiagarajan7fd1ce72012-04-25 12:38:18 +05303609 /*
3610 * Even if the fw has HT support, advertise HT cap only when
3611 * the firmware has support to override RSN capability, otherwise
3612 * 4-way handshake would fail.
3613 */
3614 if (!(ht &&
3615 test_bit(ATH6KL_FW_CAPABILITY_RSN_CAP_OVERRIDE,
3616 ar->fw_capabilities))) {
Thomas Pedersend92917e2012-04-19 15:31:56 -07003617 ath6kl_band_2ghz.ht_cap.cap = 0;
3618 ath6kl_band_2ghz.ht_cap.ht_supported = false;
3619 ath6kl_band_5ghz.ht_cap.cap = 0;
3620 ath6kl_band_5ghz.ht_cap.ht_supported = false;
3621 }
3622 if (band_2gig)
3623 wiphy->bands[IEEE80211_BAND_2GHZ] = &ath6kl_band_2ghz;
3624 if (band_5gig)
3625 wiphy->bands[IEEE80211_BAND_5GHZ] = &ath6kl_band_5ghz;
3626
Kalle Valo46d33a22012-01-17 20:08:40 +02003627 wiphy->signal_type = CFG80211_SIGNAL_TYPE_MBM;
3628
3629 wiphy->cipher_suites = cipher_suites;
3630 wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
3631
3632 wiphy->wowlan.flags = WIPHY_WOWLAN_MAGIC_PKT |
3633 WIPHY_WOWLAN_DISCONNECT |
3634 WIPHY_WOWLAN_GTK_REKEY_FAILURE |
3635 WIPHY_WOWLAN_SUPPORTS_GTK_REKEY |
3636 WIPHY_WOWLAN_EAP_IDENTITY_REQ |
3637 WIPHY_WOWLAN_4WAY_HANDSHAKE;
3638 wiphy->wowlan.n_patterns = WOW_MAX_FILTERS_PER_LIST;
3639 wiphy->wowlan.pattern_min_len = 1;
3640 wiphy->wowlan.pattern_max_len = WOW_PATTERN_SIZE;
3641
Jouni Malinen8ab54152012-05-09 22:14:51 +03003642 wiphy->max_sched_scan_ssids = MAX_PROBED_SSIDS;
Kalle Valo46d33a22012-01-17 20:08:40 +02003643
Vasanthakumar Thiagarajanf2afdac2012-02-28 20:20:19 +05303644 ar->wiphy->flags |= WIPHY_FLAG_SUPPORTS_FW_ROAM |
3645 WIPHY_FLAG_HAVE_AP_SME |
3646 WIPHY_FLAG_HAS_REMAIN_ON_CHANNEL |
3647 WIPHY_FLAG_AP_PROBE_RESP_OFFLOAD;
3648
3649 if (test_bit(ATH6KL_FW_CAPABILITY_SCHED_SCAN, ar->fw_capabilities))
3650 ar->wiphy->flags |= WIPHY_FLAG_SUPPORTS_SCHED_SCAN;
3651
Vasanthakumar Thiagarajan03bdeb02012-03-21 20:58:39 +05303652 if (test_bit(ATH6KL_FW_CAPABILITY_INACTIVITY_TIMEOUT,
3653 ar->fw_capabilities))
3654 ar->wiphy->features = NL80211_FEATURE_INACTIVITY_TIMER;
3655
Vasanthakumar Thiagarajanf2afdac2012-02-28 20:20:19 +05303656 ar->wiphy->probe_resp_offload =
3657 NL80211_PROBE_RESP_OFFLOAD_SUPPORT_WPS |
3658 NL80211_PROBE_RESP_OFFLOAD_SUPPORT_WPS2 |
Jouni Malinena432e7c2012-04-16 19:25:35 +03003659 NL80211_PROBE_RESP_OFFLOAD_SUPPORT_P2P;
Vasanthakumar Thiagarajanf2afdac2012-02-28 20:20:19 +05303660
Kalle Valo46d33a22012-01-17 20:08:40 +02003661 ret = wiphy_register(wiphy);
3662 if (ret < 0) {
3663 ath6kl_err("couldn't register wiphy device\n");
3664 return ret;
3665 }
3666
Vasanthakumar Thiagarajane5348a12012-02-25 14:43:17 +05303667 ar->wiphy_registered = true;
3668
Kalle Valo46d33a22012-01-17 20:08:40 +02003669 return 0;
3670}
3671
3672void ath6kl_cfg80211_cleanup(struct ath6kl *ar)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303673{
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05303674 wiphy_unregister(ar->wiphy);
Vasanthakumar Thiagarajane5348a12012-02-25 14:43:17 +05303675
3676 ar->wiphy_registered = false;
Kalle Valo45eaa782012-01-17 20:09:05 +02003677}
Kalle Valo46d33a22012-01-17 20:08:40 +02003678
Kalle Valo45eaa782012-01-17 20:09:05 +02003679struct ath6kl *ath6kl_cfg80211_create(void)
3680{
3681 struct ath6kl *ar;
3682 struct wiphy *wiphy;
3683
3684 /* create a new wiphy for use with cfg80211 */
3685 wiphy = wiphy_new(&ath6kl_cfg80211_ops, sizeof(struct ath6kl));
3686
3687 if (!wiphy) {
3688 ath6kl_err("couldn't allocate wiphy device\n");
3689 return NULL;
3690 }
3691
3692 ar = wiphy_priv(wiphy);
3693 ar->wiphy = wiphy;
3694
3695 return ar;
3696}
3697
3698/* Note: ar variable must not be accessed after calling this! */
3699void ath6kl_cfg80211_destroy(struct ath6kl *ar)
3700{
Vasanthakumar Thiagarajan1d2a4452012-01-21 15:22:53 +05303701 int i;
3702
3703 for (i = 0; i < AP_MAX_NUM_STA; i++)
3704 kfree(ar->sta_list[i].aggr_conn);
3705
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05303706 wiphy_free(ar->wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03003707}
Kalle Valo45eaa782012-01-17 20:09:05 +02003708