blob: d8d48379853006deda05b6bf4aebe26a121320c0 [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
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040018#include <linux/moduleparam.h>
Raja Manic08631c2011-12-16 14:24:24 +053019#include <linux/inetdevice.h>
Kalle Valod6a434d2012-01-17 20:09:36 +020020#include <linux/export.h>
Paul Gortmaker6eb07ca2011-09-15 19:46:05 -040021
Kalle Valobdcd8172011-07-18 00:22:30 +030022#include "core.h"
23#include "cfg80211.h"
24#include "debug.h"
Kalle Valoabcb3442011-07-22 08:26:20 +030025#include "hif-ops.h"
Kalle Valo003353b0d2011-09-01 10:14:21 +030026#include "testmode.h"
Kalle Valobdcd8172011-07-18 00:22:30 +030027
28#define RATETAB_ENT(_rate, _rateid, _flags) { \
29 .bitrate = (_rate), \
30 .flags = (_flags), \
31 .hw_value = (_rateid), \
32}
33
34#define CHAN2G(_channel, _freq, _flags) { \
35 .band = IEEE80211_BAND_2GHZ, \
36 .hw_value = (_channel), \
37 .center_freq = (_freq), \
38 .flags = (_flags), \
39 .max_antenna_gain = 0, \
40 .max_power = 30, \
41}
42
43#define CHAN5G(_channel, _flags) { \
44 .band = IEEE80211_BAND_5GHZ, \
45 .hw_value = (_channel), \
46 .center_freq = 5000 + (5 * (_channel)), \
47 .flags = (_flags), \
48 .max_antenna_gain = 0, \
49 .max_power = 30, \
50}
51
52static struct ieee80211_rate ath6kl_rates[] = {
53 RATETAB_ENT(10, 0x1, 0),
54 RATETAB_ENT(20, 0x2, 0),
55 RATETAB_ENT(55, 0x4, 0),
56 RATETAB_ENT(110, 0x8, 0),
57 RATETAB_ENT(60, 0x10, 0),
58 RATETAB_ENT(90, 0x20, 0),
59 RATETAB_ENT(120, 0x40, 0),
60 RATETAB_ENT(180, 0x80, 0),
61 RATETAB_ENT(240, 0x100, 0),
62 RATETAB_ENT(360, 0x200, 0),
63 RATETAB_ENT(480, 0x400, 0),
64 RATETAB_ENT(540, 0x800, 0),
65};
66
67#define ath6kl_a_rates (ath6kl_rates + 4)
68#define ath6kl_a_rates_size 8
69#define ath6kl_g_rates (ath6kl_rates + 0)
70#define ath6kl_g_rates_size 12
71
72static struct ieee80211_channel ath6kl_2ghz_channels[] = {
73 CHAN2G(1, 2412, 0),
74 CHAN2G(2, 2417, 0),
75 CHAN2G(3, 2422, 0),
76 CHAN2G(4, 2427, 0),
77 CHAN2G(5, 2432, 0),
78 CHAN2G(6, 2437, 0),
79 CHAN2G(7, 2442, 0),
80 CHAN2G(8, 2447, 0),
81 CHAN2G(9, 2452, 0),
82 CHAN2G(10, 2457, 0),
83 CHAN2G(11, 2462, 0),
84 CHAN2G(12, 2467, 0),
85 CHAN2G(13, 2472, 0),
86 CHAN2G(14, 2484, 0),
87};
88
89static struct ieee80211_channel ath6kl_5ghz_a_channels[] = {
90 CHAN5G(34, 0), CHAN5G(36, 0),
91 CHAN5G(38, 0), CHAN5G(40, 0),
92 CHAN5G(42, 0), CHAN5G(44, 0),
93 CHAN5G(46, 0), CHAN5G(48, 0),
94 CHAN5G(52, 0), CHAN5G(56, 0),
95 CHAN5G(60, 0), CHAN5G(64, 0),
96 CHAN5G(100, 0), CHAN5G(104, 0),
97 CHAN5G(108, 0), CHAN5G(112, 0),
98 CHAN5G(116, 0), CHAN5G(120, 0),
99 CHAN5G(124, 0), CHAN5G(128, 0),
100 CHAN5G(132, 0), CHAN5G(136, 0),
101 CHAN5G(140, 0), CHAN5G(149, 0),
102 CHAN5G(153, 0), CHAN5G(157, 0),
103 CHAN5G(161, 0), CHAN5G(165, 0),
104 CHAN5G(184, 0), CHAN5G(188, 0),
105 CHAN5G(192, 0), CHAN5G(196, 0),
106 CHAN5G(200, 0), CHAN5G(204, 0),
107 CHAN5G(208, 0), CHAN5G(212, 0),
108 CHAN5G(216, 0),
109};
110
111static struct ieee80211_supported_band ath6kl_band_2ghz = {
112 .n_channels = ARRAY_SIZE(ath6kl_2ghz_channels),
113 .channels = ath6kl_2ghz_channels,
114 .n_bitrates = ath6kl_g_rates_size,
115 .bitrates = ath6kl_g_rates,
116};
117
118static struct ieee80211_supported_band ath6kl_band_5ghz = {
119 .n_channels = ARRAY_SIZE(ath6kl_5ghz_a_channels),
120 .channels = ath6kl_5ghz_a_channels,
121 .n_bitrates = ath6kl_a_rates_size,
122 .bitrates = ath6kl_a_rates,
123};
124
Jouni Malinen837cb972011-10-11 17:31:57 +0300125#define CCKM_KRK_CIPHER_SUITE 0x004096ff /* use for KRK */
126
Kalle Valo10509f92011-12-13 14:52:07 +0200127/* returns true if scheduled scan was stopped */
128static bool __ath6kl_cfg80211_sscan_stop(struct ath6kl_vif *vif)
129{
130 struct ath6kl *ar = vif->ar;
131
132 if (ar->state != ATH6KL_STATE_SCHED_SCAN)
133 return false;
134
135 del_timer_sync(&vif->sched_scan_timer);
136
137 ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
138 ATH6KL_HOST_MODE_AWAKE);
139
140 ar->state = ATH6KL_STATE_ON;
141
142 return true;
143}
144
145static void ath6kl_cfg80211_sscan_disable(struct ath6kl_vif *vif)
146{
147 struct ath6kl *ar = vif->ar;
148 bool stopped;
149
150 stopped = __ath6kl_cfg80211_sscan_stop(vif);
151
152 if (!stopped)
153 return;
154
155 cfg80211_sched_scan_stopped(ar->wiphy);
156}
157
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530158static int ath6kl_set_wpa_version(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300159 enum nl80211_wpa_versions wpa_version)
160{
161 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: %u\n", __func__, wpa_version);
162
163 if (!wpa_version) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530164 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300165 } else if (wpa_version & NL80211_WPA_VERSION_2) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530166 vif->auth_mode = WPA2_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300167 } else if (wpa_version & NL80211_WPA_VERSION_1) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530168 vif->auth_mode = WPA_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300169 } else {
170 ath6kl_err("%s: %u not supported\n", __func__, wpa_version);
171 return -ENOTSUPP;
172 }
173
174 return 0;
175}
176
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530177static int ath6kl_set_auth_type(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300178 enum nl80211_auth_type auth_type)
179{
Kalle Valobdcd8172011-07-18 00:22:30 +0300180 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, auth_type);
181
182 switch (auth_type) {
183 case NL80211_AUTHTYPE_OPEN_SYSTEM:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530184 vif->dot11_auth_mode = OPEN_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300185 break;
186 case NL80211_AUTHTYPE_SHARED_KEY:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530187 vif->dot11_auth_mode = SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300188 break;
189 case NL80211_AUTHTYPE_NETWORK_EAP:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530190 vif->dot11_auth_mode = LEAP_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300191 break;
192
193 case NL80211_AUTHTYPE_AUTOMATIC:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530194 vif->dot11_auth_mode = OPEN_AUTH | SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300195 break;
196
197 default:
Masanari Iida3c325fb2012-01-31 23:32:55 +0900198 ath6kl_err("%s: 0x%x not supported\n", __func__, auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300199 return -ENOTSUPP;
200 }
201
202 return 0;
203}
204
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530205static int ath6kl_set_cipher(struct ath6kl_vif *vif, u32 cipher, bool ucast)
Kalle Valobdcd8172011-07-18 00:22:30 +0300206{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530207 u8 *ar_cipher = ucast ? &vif->prwise_crypto : &vif->grp_crypto;
208 u8 *ar_cipher_len = ucast ? &vif->prwise_crypto_len :
209 &vif->grp_crypto_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300210
211 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: cipher 0x%x, ucast %u\n",
212 __func__, cipher, ucast);
213
214 switch (cipher) {
215 case 0:
216 /* our own hack to use value 0 as no crypto used */
217 *ar_cipher = NONE_CRYPT;
218 *ar_cipher_len = 0;
219 break;
220 case WLAN_CIPHER_SUITE_WEP40:
221 *ar_cipher = WEP_CRYPT;
222 *ar_cipher_len = 5;
223 break;
224 case WLAN_CIPHER_SUITE_WEP104:
225 *ar_cipher = WEP_CRYPT;
226 *ar_cipher_len = 13;
227 break;
228 case WLAN_CIPHER_SUITE_TKIP:
229 *ar_cipher = TKIP_CRYPT;
230 *ar_cipher_len = 0;
231 break;
232 case WLAN_CIPHER_SUITE_CCMP:
233 *ar_cipher = AES_CRYPT;
234 *ar_cipher_len = 0;
235 break;
Dai Shuibing5e070212011-11-03 11:39:37 +0200236 case WLAN_CIPHER_SUITE_SMS4:
237 *ar_cipher = WAPI_CRYPT;
238 *ar_cipher_len = 0;
239 break;
Kalle Valobdcd8172011-07-18 00:22:30 +0300240 default:
241 ath6kl_err("cipher 0x%x not supported\n", cipher);
242 return -ENOTSUPP;
243 }
244
245 return 0;
246}
247
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530248static void ath6kl_set_key_mgmt(struct ath6kl_vif *vif, u32 key_mgmt)
Kalle Valobdcd8172011-07-18 00:22:30 +0300249{
250 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, key_mgmt);
251
252 if (key_mgmt == WLAN_AKM_SUITE_PSK) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530253 if (vif->auth_mode == WPA_AUTH)
254 vif->auth_mode = WPA_PSK_AUTH;
255 else if (vif->auth_mode == WPA2_AUTH)
256 vif->auth_mode = WPA2_PSK_AUTH;
Jouni Malinen837cb972011-10-11 17:31:57 +0300257 } else if (key_mgmt == 0x00409600) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530258 if (vif->auth_mode == WPA_AUTH)
259 vif->auth_mode = WPA_AUTH_CCKM;
260 else if (vif->auth_mode == WPA2_AUTH)
261 vif->auth_mode = WPA2_AUTH_CCKM;
Kalle Valobdcd8172011-07-18 00:22:30 +0300262 } else if (key_mgmt != WLAN_AKM_SUITE_8021X) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530263 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300264 }
265}
266
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530267static bool ath6kl_cfg80211_ready(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +0300268{
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530269 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530270
Kalle Valobdcd8172011-07-18 00:22:30 +0300271 if (!test_bit(WMI_READY, &ar->flag)) {
272 ath6kl_err("wmi is not ready\n");
273 return false;
274 }
275
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530276 if (!test_bit(WLAN_ENABLED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300277 ath6kl_err("wlan disabled\n");
278 return false;
279 }
280
281 return true;
282}
283
Kevin Fang6981ffd2011-10-07 08:51:19 +0800284static bool ath6kl_is_wpa_ie(const u8 *pos)
285{
286 return pos[0] == WLAN_EID_WPA && pos[1] >= 4 &&
287 pos[2] == 0x00 && pos[3] == 0x50 &&
288 pos[4] == 0xf2 && pos[5] == 0x01;
289}
290
291static bool ath6kl_is_rsn_ie(const u8 *pos)
292{
293 return pos[0] == WLAN_EID_RSN;
294}
295
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700296static bool ath6kl_is_wps_ie(const u8 *pos)
297{
298 return (pos[0] == WLAN_EID_VENDOR_SPECIFIC &&
299 pos[1] >= 4 &&
300 pos[2] == 0x00 && pos[3] == 0x50 && pos[4] == 0xf2 &&
301 pos[5] == 0x04);
302}
303
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530304static int ath6kl_set_assoc_req_ies(struct ath6kl_vif *vif, const u8 *ies,
305 size_t ies_len)
Kevin Fang6981ffd2011-10-07 08:51:19 +0800306{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530307 struct ath6kl *ar = vif->ar;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800308 const u8 *pos;
309 u8 *buf = NULL;
310 size_t len = 0;
311 int ret;
312
313 /*
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700314 * Clear previously set flag
315 */
316
317 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
318
319 /*
Kevin Fang6981ffd2011-10-07 08:51:19 +0800320 * Filter out RSN/WPA IE(s)
321 */
322
323 if (ies && ies_len) {
324 buf = kmalloc(ies_len, GFP_KERNEL);
325 if (buf == NULL)
326 return -ENOMEM;
327 pos = ies;
328
329 while (pos + 1 < ies + ies_len) {
330 if (pos + 2 + pos[1] > ies + ies_len)
331 break;
332 if (!(ath6kl_is_wpa_ie(pos) || ath6kl_is_rsn_ie(pos))) {
333 memcpy(buf + len, pos, 2 + pos[1]);
334 len += 2 + pos[1];
335 }
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700336
337 if (ath6kl_is_wps_ie(pos))
338 ar->connect_ctrl_flags |= CONNECT_WPS_FLAG;
339
Kevin Fang6981ffd2011-10-07 08:51:19 +0800340 pos += 2 + pos[1];
341 }
342 }
343
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530344 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
345 WMI_FRAME_ASSOC_REQ, buf, len);
Kevin Fang6981ffd2011-10-07 08:51:19 +0800346 kfree(buf);
347 return ret;
348}
349
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530350static int ath6kl_nliftype_to_drv_iftype(enum nl80211_iftype type, u8 *nw_type)
351{
352 switch (type) {
353 case NL80211_IFTYPE_STATION:
354 *nw_type = INFRA_NETWORK;
355 break;
356 case NL80211_IFTYPE_ADHOC:
357 *nw_type = ADHOC_NETWORK;
358 break;
359 case NL80211_IFTYPE_AP:
360 *nw_type = AP_NETWORK;
361 break;
362 case NL80211_IFTYPE_P2P_CLIENT:
363 *nw_type = INFRA_NETWORK;
364 break;
365 case NL80211_IFTYPE_P2P_GO:
366 *nw_type = AP_NETWORK;
367 break;
368 default:
369 ath6kl_err("invalid interface type %u\n", type);
370 return -ENOTSUPP;
371 }
372
373 return 0;
374}
375
376static bool ath6kl_is_valid_iftype(struct ath6kl *ar, enum nl80211_iftype type,
377 u8 *if_idx, u8 *nw_type)
378{
379 int i;
380
381 if (ath6kl_nliftype_to_drv_iftype(type, nw_type))
382 return false;
383
384 if (ar->ibss_if_active || ((type == NL80211_IFTYPE_ADHOC) &&
385 ar->num_vif))
386 return false;
387
388 if (type == NL80211_IFTYPE_STATION ||
389 type == NL80211_IFTYPE_AP || type == NL80211_IFTYPE_ADHOC) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200390 for (i = 0; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530391 if ((ar->avail_idx_map >> i) & BIT(0)) {
392 *if_idx = i;
393 return true;
394 }
395 }
396 }
397
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530398 if (type == NL80211_IFTYPE_P2P_CLIENT ||
399 type == NL80211_IFTYPE_P2P_GO) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200400 for (i = ar->max_norm_iface; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530401 if ((ar->avail_idx_map >> i) & BIT(0)) {
402 *if_idx = i;
403 return true;
404 }
405 }
406 }
407
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530408 return false;
409}
410
Kalle Valobdcd8172011-07-18 00:22:30 +0300411static int ath6kl_cfg80211_connect(struct wiphy *wiphy, struct net_device *dev,
412 struct cfg80211_connect_params *sme)
413{
414 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530415 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300416 int status;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800417 u8 nw_subtype = (ar->p2p) ? SUBTYPE_P2PDEV : SUBTYPE_NONE;
Kalle Valobdcd8172011-07-18 00:22:30 +0300418
Kalle Valo10509f92011-12-13 14:52:07 +0200419 ath6kl_cfg80211_sscan_disable(vif);
420
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530421 vif->sme_state = SME_CONNECTING;
Kalle Valobdcd8172011-07-18 00:22:30 +0300422
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530423 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300424 return -EIO;
425
426 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
427 ath6kl_err("destroy in progress\n");
428 return -EBUSY;
429 }
430
431 if (test_bit(SKIP_SCAN, &ar->flag) &&
432 ((sme->channel && sme->channel->center_freq == 0) ||
433 (sme->bssid && is_zero_ether_addr(sme->bssid)))) {
434 ath6kl_err("SkipScan: channel or bssid invalid\n");
435 return -EINVAL;
436 }
437
438 if (down_interruptible(&ar->sem)) {
439 ath6kl_err("busy, couldn't get access\n");
440 return -ERESTARTSYS;
441 }
442
443 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
444 ath6kl_err("busy, destroy in progress\n");
445 up(&ar->sem);
446 return -EBUSY;
447 }
448
449 if (ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)]) {
450 /*
451 * sleep until the command queue drains
452 */
453 wait_event_interruptible_timeout(ar->event_wq,
454 ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)] == 0,
455 WMI_TIMEOUT);
456 if (signal_pending(current)) {
457 ath6kl_err("cmd queue drain timeout\n");
458 up(&ar->sem);
459 return -EINTR;
460 }
461 }
462
Jouni Malinen6e786cb2011-12-15 14:16:00 +0200463 status = ath6kl_set_assoc_req_ies(vif, sme->ie, sme->ie_len);
464 if (status) {
465 up(&ar->sem);
466 return status;
467 }
468
469 if (sme->ie == NULL || sme->ie_len == 0)
Raja Mani542c5192011-11-15 14:14:56 +0530470 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800471
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530472 if (test_bit(CONNECTED, &vif->flags) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530473 vif->ssid_len == sme->ssid_len &&
474 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530475 vif->reconnect_flag = true;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530476 status = ath6kl_wmi_reconnect_cmd(ar->wmi, vif->fw_vif_idx,
477 vif->req_bssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530478 vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300479
480 up(&ar->sem);
481 if (status) {
482 ath6kl_err("wmi_reconnect_cmd failed\n");
483 return -EIO;
484 }
485 return 0;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530486 } else if (vif->ssid_len == sme->ssid_len &&
487 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530488 ath6kl_disconnect(vif);
Kalle Valobdcd8172011-07-18 00:22:30 +0300489 }
490
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530491 memset(vif->ssid, 0, sizeof(vif->ssid));
492 vif->ssid_len = sme->ssid_len;
493 memcpy(vif->ssid, sme->ssid, sme->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +0300494
495 if (sme->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530496 vif->ch_hint = sme->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +0300497
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530498 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300499 if (sme->bssid && !is_broadcast_ether_addr(sme->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530500 memcpy(vif->req_bssid, sme->bssid, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300501
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530502 ath6kl_set_wpa_version(vif, sme->crypto.wpa_versions);
Kalle Valobdcd8172011-07-18 00:22:30 +0300503
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530504 status = ath6kl_set_auth_type(vif, sme->auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300505 if (status) {
506 up(&ar->sem);
507 return status;
508 }
509
510 if (sme->crypto.n_ciphers_pairwise)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530511 ath6kl_set_cipher(vif, sme->crypto.ciphers_pairwise[0], true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300512 else
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530513 ath6kl_set_cipher(vif, 0, true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300514
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530515 ath6kl_set_cipher(vif, sme->crypto.cipher_group, false);
Kalle Valobdcd8172011-07-18 00:22:30 +0300516
517 if (sme->crypto.n_akm_suites)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530518 ath6kl_set_key_mgmt(vif, sme->crypto.akm_suites[0]);
Kalle Valobdcd8172011-07-18 00:22:30 +0300519
520 if ((sme->key_len) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530521 (vif->auth_mode == NONE_AUTH) &&
522 (vif->prwise_crypto == WEP_CRYPT)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300523 struct ath6kl_key *key = NULL;
524
Vivek Natarajan792ecb32011-12-29 16:18:39 +0530525 if (sme->key_idx > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300526 ath6kl_err("key index %d out of bounds\n",
527 sme->key_idx);
528 up(&ar->sem);
529 return -ENOENT;
530 }
531
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530532 key = &vif->keys[sme->key_idx];
Kalle Valobdcd8172011-07-18 00:22:30 +0300533 key->key_len = sme->key_len;
534 memcpy(key->key, sme->key, key->key_len);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530535 key->cipher = vif->prwise_crypto;
536 vif->def_txkey_index = sme->key_idx;
Kalle Valobdcd8172011-07-18 00:22:30 +0300537
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530538 ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, sme->key_idx,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530539 vif->prwise_crypto,
Kalle Valobdcd8172011-07-18 00:22:30 +0300540 GROUP_USAGE | TX_USAGE,
541 key->key_len,
Jouni Malinenf4bb9a62011-11-02 23:45:55 +0200542 NULL, 0,
Kalle Valobdcd8172011-07-18 00:22:30 +0300543 key->key, KEY_OP_INIT_VAL, NULL,
544 NO_SYNC_WMIFLAG);
545 }
546
547 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530548 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530549 if (ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
550 ALL_BSS_FILTER, 0) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300551 ath6kl_err("couldn't set bss filtering\n");
552 up(&ar->sem);
553 return -EIO;
554 }
555 }
556
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530557 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +0300558
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800559 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_CLIENT)
560 nw_subtype = SUBTYPE_P2PCLIENT;
561
Kalle Valobdcd8172011-07-18 00:22:30 +0300562 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
563 "%s: connect called with authmode %d dot11 auth %d"
564 " PW crypto %d PW crypto len %d GRP crypto %d"
565 " GRP crypto len %d channel hint %u\n",
566 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530567 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
568 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530569 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300570
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530571 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530572 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530573 vif->dot11_auth_mode, vif->auth_mode,
574 vif->prwise_crypto,
575 vif->prwise_crypto_len,
576 vif->grp_crypto, vif->grp_crypto_len,
577 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530578 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800579 ar->connect_ctrl_flags, nw_subtype);
Kalle Valobdcd8172011-07-18 00:22:30 +0300580
581 up(&ar->sem);
582
583 if (status == -EINVAL) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530584 memset(vif->ssid, 0, sizeof(vif->ssid));
585 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300586 ath6kl_err("invalid request\n");
587 return -ENOENT;
588 } else if (status) {
589 ath6kl_err("ath6kl_wmi_connect_cmd failed\n");
590 return -EIO;
591 }
592
593 if ((!(ar->connect_ctrl_flags & CONNECT_DO_WPA_OFFLOAD)) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530594 ((vif->auth_mode == WPA_PSK_AUTH)
595 || (vif->auth_mode == WPA2_PSK_AUTH))) {
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +0530596 mod_timer(&vif->disconnect_timer,
Kalle Valobdcd8172011-07-18 00:22:30 +0300597 jiffies + msecs_to_jiffies(DISCON_TIMER_INTVAL));
598 }
599
600 ar->connect_ctrl_flags &= ~CONNECT_DO_WPA_OFFLOAD;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530601 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300602
603 return 0;
604}
605
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530606static struct cfg80211_bss *
607ath6kl_add_bss_if_needed(struct ath6kl_vif *vif,
608 enum network_type nw_type,
609 const u8 *bssid,
610 struct ieee80211_channel *chan,
611 const u8 *beacon_ie,
612 size_t beacon_ie_len)
Jouni Malinen01cac472011-09-19 19:14:59 +0300613{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530614 struct ath6kl *ar = vif->ar;
Jouni Malinen01cac472011-09-19 19:14:59 +0300615 struct cfg80211_bss *bss;
Raja Mani4eab6f42011-11-09 17:02:23 +0530616 u16 cap_mask, cap_val;
Jouni Malinen01cac472011-09-19 19:14:59 +0300617 u8 *ie;
618
Raja Mani4eab6f42011-11-09 17:02:23 +0530619 if (nw_type & ADHOC_NETWORK) {
620 cap_mask = WLAN_CAPABILITY_IBSS;
621 cap_val = WLAN_CAPABILITY_IBSS;
622 } else {
623 cap_mask = WLAN_CAPABILITY_ESS;
624 cap_val = WLAN_CAPABILITY_ESS;
625 }
626
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530627 bss = cfg80211_get_bss(ar->wiphy, chan, bssid,
Raja Mani4eab6f42011-11-09 17:02:23 +0530628 vif->ssid, vif->ssid_len,
629 cap_mask, cap_val);
Jouni Malinen01cac472011-09-19 19:14:59 +0300630 if (bss == NULL) {
631 /*
632 * Since cfg80211 may not yet know about the BSS,
633 * generate a partial entry until the first BSS info
634 * event becomes available.
635 *
636 * Prepend SSID element since it is not included in the Beacon
637 * IEs from the target.
638 */
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530639 ie = kmalloc(2 + vif->ssid_len + beacon_ie_len, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300640 if (ie == NULL)
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530641 return NULL;
Jouni Malinen01cac472011-09-19 19:14:59 +0300642 ie[0] = WLAN_EID_SSID;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530643 ie[1] = vif->ssid_len;
644 memcpy(ie + 2, vif->ssid, vif->ssid_len);
645 memcpy(ie + 2 + vif->ssid_len, beacon_ie, beacon_ie_len);
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530646 bss = cfg80211_inform_bss(ar->wiphy, chan,
Raja Mani4eab6f42011-11-09 17:02:23 +0530647 bssid, 0, cap_val, 100,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530648 ie, 2 + vif->ssid_len + beacon_ie_len,
Jouni Malinen01cac472011-09-19 19:14:59 +0300649 0, GFP_KERNEL);
650 if (bss)
Raja Mani4eab6f42011-11-09 17:02:23 +0530651 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "added bss %pM to "
652 "cfg80211\n", bssid);
Jouni Malinen01cac472011-09-19 19:14:59 +0300653 kfree(ie);
654 } else
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530655 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "cfg80211 already has a bss\n");
Jouni Malinen01cac472011-09-19 19:14:59 +0300656
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530657 return bss;
Jouni Malinen01cac472011-09-19 19:14:59 +0300658}
659
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530660void ath6kl_cfg80211_connect_event(struct ath6kl_vif *vif, u16 channel,
Kalle Valobdcd8172011-07-18 00:22:30 +0300661 u8 *bssid, u16 listen_intvl,
662 u16 beacon_intvl,
663 enum network_type nw_type,
664 u8 beacon_ie_len, u8 assoc_req_len,
665 u8 assoc_resp_len, u8 *assoc_info)
666{
Jouni Malinen01cac472011-09-19 19:14:59 +0300667 struct ieee80211_channel *chan;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530668 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530669 struct cfg80211_bss *bss;
Kalle Valobdcd8172011-07-18 00:22:30 +0300670
671 /* capinfo + listen interval */
672 u8 assoc_req_ie_offset = sizeof(u16) + sizeof(u16);
673
674 /* capinfo + status code + associd */
675 u8 assoc_resp_ie_offset = sizeof(u16) + sizeof(u16) + sizeof(u16);
676
677 u8 *assoc_req_ie = assoc_info + beacon_ie_len + assoc_req_ie_offset;
678 u8 *assoc_resp_ie = assoc_info + beacon_ie_len + assoc_req_len +
679 assoc_resp_ie_offset;
680
681 assoc_req_len -= assoc_req_ie_offset;
682 assoc_resp_len -= assoc_resp_ie_offset;
683
Jouni Malinen32c10872011-09-19 19:15:07 +0300684 /*
685 * Store Beacon interval here; DTIM period will be available only once
686 * a Beacon frame from the AP is seen.
687 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530688 vif->assoc_bss_beacon_int = beacon_intvl;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530689 clear_bit(DTIM_PERIOD_AVAIL, &vif->flags);
Jouni Malinen32c10872011-09-19 19:15:07 +0300690
Kalle Valobdcd8172011-07-18 00:22:30 +0300691 if (nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530692 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300693 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
694 "%s: ath6k not in ibss mode\n", __func__);
695 return;
696 }
697 }
698
699 if (nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530700 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
701 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300702 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
703 "%s: ath6k not in station mode\n", __func__);
704 return;
705 }
706 }
707
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530708 chan = ieee80211_get_channel(ar->wiphy, (int) channel);
Kalle Valobdcd8172011-07-18 00:22:30 +0300709
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530710 bss = ath6kl_add_bss_if_needed(vif, nw_type, bssid, chan,
711 assoc_info, beacon_ie_len);
712 if (!bss) {
Raja Mani4eab6f42011-11-09 17:02:23 +0530713 ath6kl_err("could not add cfg80211 bss entry\n");
Kalle Valobdcd8172011-07-18 00:22:30 +0300714 return;
715 }
716
Raja Mani4eab6f42011-11-09 17:02:23 +0530717 if (nw_type & ADHOC_NETWORK) {
718 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "ad-hoc %s selected\n",
719 nw_type & ADHOC_CREATOR ? "creator" : "joiner");
720 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530721 cfg80211_put_bss(bss);
Jouni Malinen01cac472011-09-19 19:14:59 +0300722 return;
723 }
724
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530725 if (vif->sme_state == SME_CONNECTING) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300726 /* inform connect result to cfg80211 */
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530727 vif->sme_state = SME_CONNECTED;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530728 cfg80211_connect_result(vif->ndev, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300729 assoc_req_ie, assoc_req_len,
730 assoc_resp_ie, assoc_resp_len,
731 WLAN_STATUS_SUCCESS, GFP_KERNEL);
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530732 cfg80211_put_bss(bss);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530733 } else if (vif->sme_state == SME_CONNECTED) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300734 /* inform roam event to cfg80211 */
Vasanthakumar Thiagarajan5e13fd32011-12-13 17:19:57 +0530735 cfg80211_roamed_bss(vif->ndev, bss, assoc_req_ie, assoc_req_len,
736 assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300737 }
738}
739
740static int ath6kl_cfg80211_disconnect(struct wiphy *wiphy,
741 struct net_device *dev, u16 reason_code)
742{
Kalle Valod6d5c062011-11-25 13:17:37 +0200743 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530744 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300745
746 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: reason=%u\n", __func__,
747 reason_code);
748
Kalle Valo10509f92011-12-13 14:52:07 +0200749 ath6kl_cfg80211_sscan_disable(vif);
750
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530751 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300752 return -EIO;
753
754 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
755 ath6kl_err("busy, destroy in progress\n");
756 return -EBUSY;
757 }
758
759 if (down_interruptible(&ar->sem)) {
760 ath6kl_err("busy, couldn't get access\n");
761 return -ERESTARTSYS;
762 }
763
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530764 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530765 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530766 memset(vif->ssid, 0, sizeof(vif->ssid));
767 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300768
769 if (!test_bit(SKIP_SCAN, &ar->flag))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530770 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300771
772 up(&ar->sem);
773
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530774 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan170826d2011-09-10 15:26:35 +0530775
Kalle Valobdcd8172011-07-18 00:22:30 +0300776 return 0;
777}
778
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530779void ath6kl_cfg80211_disconnect_event(struct ath6kl_vif *vif, u8 reason,
Kalle Valobdcd8172011-07-18 00:22:30 +0300780 u8 *bssid, u8 assoc_resp_len,
781 u8 *assoc_info, u16 proto_reason)
782{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530783 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530784
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530785 if (vif->scan_req) {
786 cfg80211_scan_done(vif->scan_req, true);
787 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300788 }
789
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530790 if (vif->nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530791 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300792 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
793 "%s: ath6k not in ibss mode\n", __func__);
794 return;
795 }
796 memset(bssid, 0, ETH_ALEN);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530797 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300798 return;
799 }
800
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530801 if (vif->nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530802 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
803 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300804 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
805 "%s: ath6k not in station mode\n", __func__);
806 return;
807 }
808 }
809
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530810 /*
811 * Send a disconnect command to target when a disconnect event is
812 * received with reason code other than 3 (DISCONNECT_CMD - disconnect
813 * request from host) to make the firmware stop trying to connect even
814 * after giving disconnect event. There will be one more disconnect
815 * event for this disconnect command with reason code DISCONNECT_CMD
816 * which will be notified to cfg80211.
817 */
Kalle Valobdcd8172011-07-18 00:22:30 +0300818
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530819 if (reason != DISCONNECT_CMD) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530820 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +0300821 return;
822 }
823
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530824 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300825
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530826 if (vif->sme_state == SME_CONNECTING) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530827 cfg80211_connect_result(vif->ndev,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530828 bssid, NULL, 0,
829 NULL, 0,
830 WLAN_STATUS_UNSPECIFIED_FAILURE,
831 GFP_KERNEL);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530832 } else if (vif->sme_state == SME_CONNECTED) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530833 cfg80211_disconnected(vif->ndev, reason,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530834 NULL, 0, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300835 }
836
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530837 vif->sme_state = SME_DISCONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300838}
839
Kalle Valobdcd8172011-07-18 00:22:30 +0300840static int ath6kl_cfg80211_scan(struct wiphy *wiphy, struct net_device *ndev,
841 struct cfg80211_scan_request *request)
842{
Kalle Valod6d5c062011-11-25 13:17:37 +0200843 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530844 struct ath6kl_vif *vif = netdev_priv(ndev);
Edward Lu1276c9e2011-08-30 21:58:00 +0300845 s8 n_channels = 0;
846 u16 *channels = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300847 int ret = 0;
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530848 u32 force_fg_scan = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300849
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530850 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300851 return -EIO;
852
Kalle Valo10509f92011-12-13 14:52:07 +0200853 ath6kl_cfg80211_sscan_disable(vif);
854
Kalle Valobdcd8172011-07-18 00:22:30 +0300855 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530856 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300857 ret = ath6kl_wmi_bssfilter_cmd(
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530858 ar->wmi, vif->fw_vif_idx,
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530859 (test_bit(CONNECTED, &vif->flags) ?
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300860 ALL_BUT_BSS_FILTER : ALL_BSS_FILTER), 0);
861 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300862 ath6kl_err("couldn't set bss filtering\n");
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300863 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +0300864 }
865 }
866
867 if (request->n_ssids && request->ssids[0].ssid_len) {
868 u8 i;
869
870 if (request->n_ssids > (MAX_PROBED_SSID_INDEX - 1))
871 request->n_ssids = MAX_PROBED_SSID_INDEX - 1;
872
873 for (i = 0; i < request->n_ssids; i++)
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530874 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
875 i + 1, SPECIFIC_SSID_FLAG,
Kalle Valobdcd8172011-07-18 00:22:30 +0300876 request->ssids[i].ssid_len,
877 request->ssids[i].ssid);
878 }
879
Kalle Valo10509f92011-12-13 14:52:07 +0200880 /*
881 * FIXME: we should clear the IE in fw if it's not set so just
882 * remove the check altogether
883 */
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300884 if (request->ie) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530885 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
886 WMI_FRAME_PROBE_REQ,
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300887 request->ie, request->ie_len);
888 if (ret) {
889 ath6kl_err("failed to set Probe Request appie for "
890 "scan");
891 return ret;
892 }
893 }
894
Jouni Malinen11869be2011-09-02 20:07:06 +0300895 /*
896 * Scan only the requested channels if the request specifies a set of
897 * channels. If the list is longer than the target supports, do not
898 * configure the list and instead, scan all available channels.
899 */
900 if (request->n_channels > 0 &&
901 request->n_channels <= WMI_MAX_CHANNELS) {
Edward Lu1276c9e2011-08-30 21:58:00 +0300902 u8 i;
903
Jouni Malinen11869be2011-09-02 20:07:06 +0300904 n_channels = request->n_channels;
Edward Lu1276c9e2011-08-30 21:58:00 +0300905
906 channels = kzalloc(n_channels * sizeof(u16), GFP_KERNEL);
907 if (channels == NULL) {
908 ath6kl_warn("failed to set scan channels, "
909 "scan all channels");
910 n_channels = 0;
911 }
912
913 for (i = 0; i < n_channels; i++)
914 channels[i] = request->channels[i]->center_freq;
915 }
916
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530917 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530918 force_fg_scan = 1;
919
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800920 if (test_bit(ATH6KL_FW_CAPABILITY_STA_P2PDEV_DUPLEX,
921 ar->fw_capabilities)) {
922 /*
923 * If capable of doing P2P mgmt operations using
924 * station interface, send additional information like
925 * supported rates to advertise and xmit rates for
926 * probe requests
927 */
928 ret = ath6kl_wmi_beginscan_cmd(ar->wmi, vif->fw_vif_idx,
929 WMI_LONG_SCAN, force_fg_scan,
Vasanthakumar Thiagarajan13423c32012-02-21 15:30:42 +0530930 false, 0,
931 ATH6KL_FG_SCAN_INTERVAL,
932 n_channels, channels,
933 request->no_cck,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800934 request->rates);
935 } else {
936 ret = ath6kl_wmi_startscan_cmd(ar->wmi, vif->fw_vif_idx,
937 WMI_LONG_SCAN, force_fg_scan,
Vasanthakumar Thiagarajan13423c32012-02-21 15:30:42 +0530938 false, 0,
939 ATH6KL_FG_SCAN_INTERVAL,
940 n_channels, channels);
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -0800941 }
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300942 if (ret)
Kalle Valobdcd8172011-07-18 00:22:30 +0300943 ath6kl_err("wmi_startscan_cmd failed\n");
Jouni Malinen11869be2011-09-02 20:07:06 +0300944 else
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530945 vif->scan_req = request;
Kalle Valobdcd8172011-07-18 00:22:30 +0300946
Edward Lu1276c9e2011-08-30 21:58:00 +0300947 kfree(channels);
948
Kalle Valobdcd8172011-07-18 00:22:30 +0300949 return ret;
950}
951
Kalle Valo1c17d312011-11-01 08:43:56 +0200952void ath6kl_cfg80211_scan_complete_event(struct ath6kl_vif *vif, bool aborted)
Kalle Valobdcd8172011-07-18 00:22:30 +0300953{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530954 struct ath6kl *ar = vif->ar;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300955 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +0300956
Kalle Valo1c17d312011-11-01 08:43:56 +0200957 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: status%s\n", __func__,
958 aborted ? " aborted" : "");
Kalle Valobdcd8172011-07-18 00:22:30 +0300959
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530960 if (!vif->scan_req)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300961 return;
Kalle Valobdcd8172011-07-18 00:22:30 +0300962
Kalle Valo1c17d312011-11-01 08:43:56 +0200963 if (aborted)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300964 goto out;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300965
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530966 if (vif->scan_req->n_ssids && vif->scan_req->ssids[0].ssid_len) {
967 for (i = 0; i < vif->scan_req->n_ssids; i++) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530968 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
969 i + 1, DISABLE_SSID_FLAG,
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300970 0, NULL);
971 }
972 }
973
974out:
Kalle Valocb938212011-10-27 18:47:46 +0300975 cfg80211_scan_done(vif->scan_req, aborted);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530976 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300977}
978
979static int ath6kl_cfg80211_add_key(struct wiphy *wiphy, struct net_device *ndev,
980 u8 key_index, bool pairwise,
981 const u8 *mac_addr,
982 struct key_params *params)
983{
Kalle Valod6d5c062011-11-25 13:17:37 +0200984 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530985 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300986 struct ath6kl_key *key = NULL;
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +0530987 int seq_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300988 u8 key_usage;
989 u8 key_type;
Kalle Valobdcd8172011-07-18 00:22:30 +0300990
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530991 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300992 return -EIO;
993
Jouni Malinen837cb972011-10-11 17:31:57 +0300994 if (params->cipher == CCKM_KRK_CIPHER_SUITE) {
995 if (params->key_len != WMI_KRK_LEN)
996 return -EINVAL;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530997 return ath6kl_wmi_add_krk_cmd(ar->wmi, vif->fw_vif_idx,
998 params->key);
Jouni Malinen837cb972011-10-11 17:31:57 +0300999 }
1000
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301001 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001002 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1003 "%s: key index %d out of bounds\n", __func__,
1004 key_index);
1005 return -ENOENT;
1006 }
1007
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301008 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001009 memset(key, 0, sizeof(struct ath6kl_key));
1010
1011 if (pairwise)
1012 key_usage = PAIRWISE_USAGE;
1013 else
1014 key_usage = GROUP_USAGE;
1015
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +05301016 seq_len = params->seq_len;
1017 if (params->cipher == WLAN_CIPHER_SUITE_SMS4 &&
1018 seq_len > ATH6KL_KEY_SEQ_LEN) {
1019 /* Only first half of the WPI PN is configured */
1020 seq_len = ATH6KL_KEY_SEQ_LEN;
Kalle Valobdcd8172011-07-18 00:22:30 +03001021 }
Sujith Manoharan4a8ce2f2012-01-10 09:53:38 +05301022 if (params->key_len > WLAN_MAX_KEY_LEN ||
1023 seq_len > sizeof(key->seq))
1024 return -EINVAL;
1025
1026 key->key_len = params->key_len;
1027 memcpy(key->key, params->key, key->key_len);
1028 key->seq_len = seq_len;
1029 memcpy(key->seq, params->seq, key->seq_len);
1030 key->cipher = params->cipher;
Kalle Valobdcd8172011-07-18 00:22:30 +03001031
1032 switch (key->cipher) {
1033 case WLAN_CIPHER_SUITE_WEP40:
1034 case WLAN_CIPHER_SUITE_WEP104:
1035 key_type = WEP_CRYPT;
1036 break;
1037
1038 case WLAN_CIPHER_SUITE_TKIP:
1039 key_type = TKIP_CRYPT;
1040 break;
1041
1042 case WLAN_CIPHER_SUITE_CCMP:
1043 key_type = AES_CRYPT;
1044 break;
Dai Shuibing5e070212011-11-03 11:39:37 +02001045 case WLAN_CIPHER_SUITE_SMS4:
1046 key_type = WAPI_CRYPT;
1047 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001048
1049 default:
1050 return -ENOTSUPP;
1051 }
1052
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301053 if (((vif->auth_mode == WPA_PSK_AUTH)
1054 || (vif->auth_mode == WPA2_PSK_AUTH))
Kalle Valobdcd8172011-07-18 00:22:30 +03001055 && (key_usage & GROUP_USAGE))
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05301056 del_timer(&vif->disconnect_timer);
Kalle Valobdcd8172011-07-18 00:22:30 +03001057
1058 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1059 "%s: index %d, key_len %d, key_type 0x%x, key_usage 0x%x, seq_len %d\n",
1060 __func__, key_index, key->key_len, key_type,
1061 key_usage, key->seq_len);
1062
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301063 if (vif->nw_type == AP_NETWORK && !pairwise &&
Jouni Malinen47032902011-12-08 16:50:30 +02001064 (key_type == TKIP_CRYPT || key_type == AES_CRYPT ||
Vasanthakumar Thiagarajancc4d6232012-02-14 20:33:00 +05301065 key_type == WAPI_CRYPT)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001066 ar->ap_mode_bkey.valid = true;
1067 ar->ap_mode_bkey.key_index = key_index;
1068 ar->ap_mode_bkey.key_type = key_type;
1069 ar->ap_mode_bkey.key_len = key->key_len;
1070 memcpy(ar->ap_mode_bkey.key, key->key, key->key_len);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301071 if (!test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001072 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay initial group "
1073 "key configuration until AP mode has been "
1074 "started\n");
1075 /*
1076 * The key will be set in ath6kl_connect_ap_mode() once
1077 * the connected event is received from the target.
1078 */
1079 return 0;
1080 }
1081 }
1082
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301083 if (vif->next_mode == AP_NETWORK && key_type == WEP_CRYPT &&
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301084 !test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen151411e2011-09-15 15:10:16 +03001085 /*
1086 * Store the key locally so that it can be re-configured after
1087 * the AP mode has properly started
1088 * (ath6kl_install_statioc_wep_keys).
1089 */
1090 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay WEP key configuration "
1091 "until AP mode has been started\n");
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301092 vif->wep_key_list[key_index].key_len = key->key_len;
1093 memcpy(vif->wep_key_list[key_index].key, key->key,
1094 key->key_len);
Jouni Malinen151411e2011-09-15 15:10:16 +03001095 return 0;
1096 }
1097
Vasanthakumar Thiagarajan7cefa442011-11-11 20:33:00 +05301098 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, key_index,
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001099 key_type, key_usage, key->key_len,
1100 key->seq, key->seq_len, key->key,
1101 KEY_OP_INIT_VAL,
1102 (u8 *) mac_addr, SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001103}
1104
1105static int ath6kl_cfg80211_del_key(struct wiphy *wiphy, struct net_device *ndev,
1106 u8 key_index, bool pairwise,
1107 const u8 *mac_addr)
1108{
Kalle Valod6d5c062011-11-25 13:17:37 +02001109 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301110 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001111
1112 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1113
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301114 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001115 return -EIO;
1116
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301117 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001118 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1119 "%s: key index %d out of bounds\n", __func__,
1120 key_index);
1121 return -ENOENT;
1122 }
1123
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301124 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001125 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1126 "%s: index %d is empty\n", __func__, key_index);
1127 return 0;
1128 }
1129
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301130 vif->keys[key_index].key_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001131
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301132 return ath6kl_wmi_deletekey_cmd(ar->wmi, vif->fw_vif_idx, key_index);
Kalle Valobdcd8172011-07-18 00:22:30 +03001133}
1134
1135static int ath6kl_cfg80211_get_key(struct wiphy *wiphy, struct net_device *ndev,
1136 u8 key_index, bool pairwise,
1137 const u8 *mac_addr, void *cookie,
1138 void (*callback) (void *cookie,
1139 struct key_params *))
1140{
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301141 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001142 struct ath6kl_key *key = NULL;
1143 struct key_params params;
1144
1145 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1146
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301147 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001148 return -EIO;
1149
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301150 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001151 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1152 "%s: key index %d out of bounds\n", __func__,
1153 key_index);
1154 return -ENOENT;
1155 }
1156
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301157 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001158 memset(&params, 0, sizeof(params));
1159 params.cipher = key->cipher;
1160 params.key_len = key->key_len;
1161 params.seq_len = key->seq_len;
1162 params.seq = key->seq;
1163 params.key = key->key;
1164
1165 callback(cookie, &params);
1166
1167 return key->key_len ? 0 : -ENOENT;
1168}
1169
1170static int ath6kl_cfg80211_set_default_key(struct wiphy *wiphy,
1171 struct net_device *ndev,
1172 u8 key_index, bool unicast,
1173 bool multicast)
1174{
Kalle Valod6d5c062011-11-25 13:17:37 +02001175 struct ath6kl *ar = ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301176 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001177 struct ath6kl_key *key = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +03001178 u8 key_usage;
Edward Lu229ed6b2011-08-30 21:58:07 +03001179 enum crypto_type key_type = NONE_CRYPT;
Kalle Valobdcd8172011-07-18 00:22:30 +03001180
1181 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1182
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301183 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001184 return -EIO;
1185
Vivek Natarajan792ecb32011-12-29 16:18:39 +05301186 if (key_index > WMI_MAX_KEY_INDEX) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001187 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1188 "%s: key index %d out of bounds\n",
1189 __func__, key_index);
1190 return -ENOENT;
1191 }
1192
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301193 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001194 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: invalid key index %d\n",
1195 __func__, key_index);
1196 return -EINVAL;
1197 }
1198
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301199 vif->def_txkey_index = key_index;
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301200 key = &vif->keys[vif->def_txkey_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001201 key_usage = GROUP_USAGE;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301202 if (vif->prwise_crypto == WEP_CRYPT)
Kalle Valobdcd8172011-07-18 00:22:30 +03001203 key_usage |= TX_USAGE;
Edward Lu229ed6b2011-08-30 21:58:07 +03001204 if (unicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301205 key_type = vif->prwise_crypto;
Edward Lu229ed6b2011-08-30 21:58:07 +03001206 if (multicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301207 key_type = vif->grp_crypto;
Kalle Valobdcd8172011-07-18 00:22:30 +03001208
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301209 if (vif->next_mode == AP_NETWORK && !test_bit(CONNECTED, &vif->flags))
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001210 return 0; /* Delay until AP mode has been started */
1211
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001212 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx,
1213 vif->def_txkey_index,
1214 key_type, key_usage,
1215 key->key_len, key->seq, key->seq_len,
1216 key->key,
1217 KEY_OP_INIT_VAL, NULL,
1218 SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001219}
1220
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301221void ath6kl_cfg80211_tkip_micerr_event(struct ath6kl_vif *vif, u8 keyid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001222 bool ismcast)
1223{
1224 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1225 "%s: keyid %d, ismcast %d\n", __func__, keyid, ismcast);
1226
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301227 cfg80211_michael_mic_failure(vif->ndev, vif->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001228 (ismcast ? NL80211_KEYTYPE_GROUP :
1229 NL80211_KEYTYPE_PAIRWISE), keyid, NULL,
1230 GFP_KERNEL);
1231}
1232
1233static int ath6kl_cfg80211_set_wiphy_params(struct wiphy *wiphy, u32 changed)
1234{
1235 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301236 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001237 int ret;
1238
1239 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: changed 0x%x\n", __func__,
1240 changed);
1241
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301242 vif = ath6kl_vif_first(ar);
1243 if (!vif)
1244 return -EIO;
1245
1246 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001247 return -EIO;
1248
1249 if (changed & WIPHY_PARAM_RTS_THRESHOLD) {
1250 ret = ath6kl_wmi_set_rts_cmd(ar->wmi, wiphy->rts_threshold);
1251 if (ret != 0) {
1252 ath6kl_err("ath6kl_wmi_set_rts_cmd failed\n");
1253 return -EIO;
1254 }
1255 }
1256
1257 return 0;
1258}
1259
1260/*
1261 * The type nl80211_tx_power_setting replaces the following
1262 * data type from 2.6.36 onwards
1263*/
1264static int ath6kl_cfg80211_set_txpower(struct wiphy *wiphy,
1265 enum nl80211_tx_power_setting type,
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001266 int mbm)
Kalle Valobdcd8172011-07-18 00:22:30 +03001267{
1268 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301269 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001270 u8 ath6kl_dbm;
Luis R. Rodriguezb992a282011-11-23 11:08:14 -05001271 int dbm = MBM_TO_DBM(mbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001272
1273 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x, dbm %d\n", __func__,
1274 type, dbm);
1275
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301276 vif = ath6kl_vif_first(ar);
1277 if (!vif)
1278 return -EIO;
1279
1280 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001281 return -EIO;
1282
1283 switch (type) {
1284 case NL80211_TX_POWER_AUTOMATIC:
1285 return 0;
1286 case NL80211_TX_POWER_LIMITED:
1287 ar->tx_pwr = ath6kl_dbm = dbm;
1288 break;
1289 default:
1290 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x not supported\n",
1291 __func__, type);
1292 return -EOPNOTSUPP;
1293 }
1294
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301295 ath6kl_wmi_set_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx, ath6kl_dbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001296
1297 return 0;
1298}
1299
1300static int ath6kl_cfg80211_get_txpower(struct wiphy *wiphy, int *dbm)
1301{
1302 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301303 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001304
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301305 vif = ath6kl_vif_first(ar);
1306 if (!vif)
1307 return -EIO;
1308
1309 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001310 return -EIO;
1311
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301312 if (test_bit(CONNECTED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001313 ar->tx_pwr = 0;
1314
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301315 if (ath6kl_wmi_get_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001316 ath6kl_err("ath6kl_wmi_get_tx_pwr_cmd failed\n");
1317 return -EIO;
1318 }
1319
1320 wait_event_interruptible_timeout(ar->event_wq, ar->tx_pwr != 0,
1321 5 * HZ);
1322
1323 if (signal_pending(current)) {
1324 ath6kl_err("target did not respond\n");
1325 return -EINTR;
1326 }
1327 }
1328
1329 *dbm = ar->tx_pwr;
1330 return 0;
1331}
1332
1333static int ath6kl_cfg80211_set_power_mgmt(struct wiphy *wiphy,
1334 struct net_device *dev,
1335 bool pmgmt, int timeout)
1336{
1337 struct ath6kl *ar = ath6kl_priv(dev);
1338 struct wmi_power_mode_cmd mode;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301339 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001340
1341 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: pmgmt %d, timeout %d\n",
1342 __func__, pmgmt, timeout);
1343
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301344 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001345 return -EIO;
1346
1347 if (pmgmt) {
1348 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: max perf\n", __func__);
1349 mode.pwr_mode = REC_POWER;
1350 } else {
1351 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: rec power\n", __func__);
1352 mode.pwr_mode = MAX_PERF_POWER;
1353 }
1354
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301355 if (ath6kl_wmi_powermode_cmd(ar->wmi, vif->fw_vif_idx,
1356 mode.pwr_mode) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001357 ath6kl_err("wmi_powermode_cmd failed\n");
1358 return -EIO;
1359 }
1360
1361 return 0;
1362}
1363
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301364static struct net_device *ath6kl_cfg80211_add_iface(struct wiphy *wiphy,
1365 char *name,
1366 enum nl80211_iftype type,
1367 u32 *flags,
1368 struct vif_params *params)
1369{
1370 struct ath6kl *ar = wiphy_priv(wiphy);
1371 struct net_device *ndev;
1372 u8 if_idx, nw_type;
1373
Kalle Valo71f96ee2011-11-14 19:31:30 +02001374 if (ar->num_vif == ar->vif_max) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301375 ath6kl_err("Reached maximum number of supported vif\n");
1376 return ERR_PTR(-EINVAL);
1377 }
1378
1379 if (!ath6kl_is_valid_iftype(ar, type, &if_idx, &nw_type)) {
1380 ath6kl_err("Not a supported interface type\n");
1381 return ERR_PTR(-EINVAL);
1382 }
1383
1384 ndev = ath6kl_interface_add(ar, name, type, if_idx, nw_type);
1385 if (!ndev)
1386 return ERR_PTR(-ENOMEM);
1387
1388 ar->num_vif++;
1389
1390 return ndev;
1391}
1392
1393static int ath6kl_cfg80211_del_iface(struct wiphy *wiphy,
1394 struct net_device *ndev)
1395{
1396 struct ath6kl *ar = wiphy_priv(wiphy);
1397 struct ath6kl_vif *vif = netdev_priv(ndev);
1398
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301399 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301400 list_del(&vif->list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301401 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301402
1403 ath6kl_cleanup_vif(vif, test_bit(WMI_READY, &ar->flag));
1404
Kalle Valoc25889e2012-01-17 20:08:27 +02001405 ath6kl_cfg80211_vif_cleanup(vif);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301406
1407 return 0;
1408}
1409
Kalle Valobdcd8172011-07-18 00:22:30 +03001410static int ath6kl_cfg80211_change_iface(struct wiphy *wiphy,
1411 struct net_device *ndev,
1412 enum nl80211_iftype type, u32 *flags,
1413 struct vif_params *params)
1414{
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301415 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001416
1417 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type %u\n", __func__, type);
1418
Kalle Valobdcd8172011-07-18 00:22:30 +03001419 switch (type) {
1420 case NL80211_IFTYPE_STATION:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301421 vif->next_mode = INFRA_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001422 break;
1423 case NL80211_IFTYPE_ADHOC:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301424 vif->next_mode = ADHOC_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001425 break;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001426 case NL80211_IFTYPE_AP:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301427 vif->next_mode = AP_NETWORK;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001428 break;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001429 case NL80211_IFTYPE_P2P_CLIENT:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301430 vif->next_mode = INFRA_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001431 break;
1432 case NL80211_IFTYPE_P2P_GO:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301433 vif->next_mode = AP_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001434 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001435 default:
1436 ath6kl_err("invalid interface type %u\n", type);
1437 return -EOPNOTSUPP;
1438 }
1439
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +05301440 vif->wdev.iftype = type;
Kalle Valobdcd8172011-07-18 00:22:30 +03001441
1442 return 0;
1443}
1444
1445static int ath6kl_cfg80211_join_ibss(struct wiphy *wiphy,
1446 struct net_device *dev,
1447 struct cfg80211_ibss_params *ibss_param)
1448{
1449 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301450 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001451 int status;
1452
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301453 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001454 return -EIO;
1455
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301456 vif->ssid_len = ibss_param->ssid_len;
1457 memcpy(vif->ssid, ibss_param->ssid, vif->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +03001458
1459 if (ibss_param->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301460 vif->ch_hint = ibss_param->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +03001461
1462 if (ibss_param->channel_fixed) {
1463 /*
1464 * TODO: channel_fixed: The channel should be fixed, do not
1465 * search for IBSSs to join on other channels. Target
1466 * firmware does not support this feature, needs to be
1467 * updated.
1468 */
1469 return -EOPNOTSUPP;
1470 }
1471
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301472 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001473 if (ibss_param->bssid && !is_broadcast_ether_addr(ibss_param->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301474 memcpy(vif->req_bssid, ibss_param->bssid,
1475 sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001476
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301477 ath6kl_set_wpa_version(vif, 0);
Kalle Valobdcd8172011-07-18 00:22:30 +03001478
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301479 status = ath6kl_set_auth_type(vif, NL80211_AUTHTYPE_OPEN_SYSTEM);
Kalle Valobdcd8172011-07-18 00:22:30 +03001480 if (status)
1481 return status;
1482
1483 if (ibss_param->privacy) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301484 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, true);
1485 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001486 } else {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301487 ath6kl_set_cipher(vif, 0, true);
1488 ath6kl_set_cipher(vif, 0, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001489 }
1490
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301491 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +03001492
1493 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1494 "%s: connect called with authmode %d dot11 auth %d"
1495 " PW crypto %d PW crypto len %d GRP crypto %d"
1496 " GRP crypto len %d channel hint %u\n",
1497 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301498 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
1499 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301500 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +03001501
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301502 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301503 vif->dot11_auth_mode, vif->auth_mode,
1504 vif->prwise_crypto,
1505 vif->prwise_crypto_len,
1506 vif->grp_crypto, vif->grp_crypto_len,
1507 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301508 vif->req_bssid, vif->ch_hint,
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08001509 ar->connect_ctrl_flags, SUBTYPE_NONE);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301510 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001511
1512 return 0;
1513}
1514
1515static int ath6kl_cfg80211_leave_ibss(struct wiphy *wiphy,
1516 struct net_device *dev)
1517{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301518 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001519
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301520 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001521 return -EIO;
1522
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301523 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301524 memset(vif->ssid, 0, sizeof(vif->ssid));
1525 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001526
1527 return 0;
1528}
1529
1530static const u32 cipher_suites[] = {
1531 WLAN_CIPHER_SUITE_WEP40,
1532 WLAN_CIPHER_SUITE_WEP104,
1533 WLAN_CIPHER_SUITE_TKIP,
1534 WLAN_CIPHER_SUITE_CCMP,
Jouni Malinen837cb972011-10-11 17:31:57 +03001535 CCKM_KRK_CIPHER_SUITE,
Dai Shuibing5e070212011-11-03 11:39:37 +02001536 WLAN_CIPHER_SUITE_SMS4,
Kalle Valobdcd8172011-07-18 00:22:30 +03001537};
1538
1539static bool is_rate_legacy(s32 rate)
1540{
1541 static const s32 legacy[] = { 1000, 2000, 5500, 11000,
1542 6000, 9000, 12000, 18000, 24000,
1543 36000, 48000, 54000
1544 };
1545 u8 i;
1546
1547 for (i = 0; i < ARRAY_SIZE(legacy); i++)
1548 if (rate == legacy[i])
1549 return true;
1550
1551 return false;
1552}
1553
1554static bool is_rate_ht20(s32 rate, u8 *mcs, bool *sgi)
1555{
1556 static const s32 ht20[] = { 6500, 13000, 19500, 26000, 39000,
1557 52000, 58500, 65000, 72200
1558 };
1559 u8 i;
1560
1561 for (i = 0; i < ARRAY_SIZE(ht20); i++) {
1562 if (rate == ht20[i]) {
1563 if (i == ARRAY_SIZE(ht20) - 1)
1564 /* last rate uses sgi */
1565 *sgi = true;
1566 else
1567 *sgi = false;
1568
1569 *mcs = i;
1570 return true;
1571 }
1572 }
1573 return false;
1574}
1575
1576static bool is_rate_ht40(s32 rate, u8 *mcs, bool *sgi)
1577{
1578 static const s32 ht40[] = { 13500, 27000, 40500, 54000,
1579 81000, 108000, 121500, 135000,
1580 150000
1581 };
1582 u8 i;
1583
1584 for (i = 0; i < ARRAY_SIZE(ht40); i++) {
1585 if (rate == ht40[i]) {
1586 if (i == ARRAY_SIZE(ht40) - 1)
1587 /* last rate uses sgi */
1588 *sgi = true;
1589 else
1590 *sgi = false;
1591
1592 *mcs = i;
1593 return true;
1594 }
1595 }
1596
1597 return false;
1598}
1599
1600static int ath6kl_get_station(struct wiphy *wiphy, struct net_device *dev,
1601 u8 *mac, struct station_info *sinfo)
1602{
1603 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301604 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001605 long left;
1606 bool sgi;
1607 s32 rate;
1608 int ret;
1609 u8 mcs;
1610
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301611 if (memcmp(mac, vif->bssid, ETH_ALEN) != 0)
Kalle Valobdcd8172011-07-18 00:22:30 +03001612 return -ENOENT;
1613
1614 if (down_interruptible(&ar->sem))
1615 return -EBUSY;
1616
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301617 set_bit(STATS_UPDATE_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001618
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301619 ret = ath6kl_wmi_get_stats_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +03001620
1621 if (ret != 0) {
1622 up(&ar->sem);
1623 return -EIO;
1624 }
1625
1626 left = wait_event_interruptible_timeout(ar->event_wq,
1627 !test_bit(STATS_UPDATE_PEND,
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301628 &vif->flags),
Kalle Valobdcd8172011-07-18 00:22:30 +03001629 WMI_TIMEOUT);
1630
1631 up(&ar->sem);
1632
1633 if (left == 0)
1634 return -ETIMEDOUT;
1635 else if (left < 0)
1636 return left;
1637
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301638 if (vif->target_stats.rx_byte) {
1639 sinfo->rx_bytes = vif->target_stats.rx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001640 sinfo->filled |= STATION_INFO_RX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301641 sinfo->rx_packets = vif->target_stats.rx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001642 sinfo->filled |= STATION_INFO_RX_PACKETS;
1643 }
1644
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301645 if (vif->target_stats.tx_byte) {
1646 sinfo->tx_bytes = vif->target_stats.tx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001647 sinfo->filled |= STATION_INFO_TX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301648 sinfo->tx_packets = vif->target_stats.tx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001649 sinfo->filled |= STATION_INFO_TX_PACKETS;
1650 }
1651
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301652 sinfo->signal = vif->target_stats.cs_rssi;
Kalle Valobdcd8172011-07-18 00:22:30 +03001653 sinfo->filled |= STATION_INFO_SIGNAL;
1654
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301655 rate = vif->target_stats.tx_ucast_rate;
Kalle Valobdcd8172011-07-18 00:22:30 +03001656
1657 if (is_rate_legacy(rate)) {
1658 sinfo->txrate.legacy = rate / 100;
1659 } else if (is_rate_ht20(rate, &mcs, &sgi)) {
1660 if (sgi) {
1661 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1662 sinfo->txrate.mcs = mcs - 1;
1663 } else {
1664 sinfo->txrate.mcs = mcs;
1665 }
1666
1667 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1668 } else if (is_rate_ht40(rate, &mcs, &sgi)) {
1669 if (sgi) {
1670 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1671 sinfo->txrate.mcs = mcs - 1;
1672 } else {
1673 sinfo->txrate.mcs = mcs;
1674 }
1675
1676 sinfo->txrate.flags |= RATE_INFO_FLAGS_40_MHZ_WIDTH;
1677 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1678 } else {
Kalle Valo9a730832011-09-27 23:33:28 +03001679 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1680 "invalid rate from stats: %d\n", rate);
1681 ath6kl_debug_war(ar, ATH6KL_WAR_INVALID_RATE);
Kalle Valobdcd8172011-07-18 00:22:30 +03001682 return 0;
1683 }
1684
1685 sinfo->filled |= STATION_INFO_TX_BITRATE;
1686
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301687 if (test_bit(CONNECTED, &vif->flags) &&
1688 test_bit(DTIM_PERIOD_AVAIL, &vif->flags) &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301689 vif->nw_type == INFRA_NETWORK) {
Jouni Malinen32c10872011-09-19 19:15:07 +03001690 sinfo->filled |= STATION_INFO_BSS_PARAM;
1691 sinfo->bss_param.flags = 0;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301692 sinfo->bss_param.dtim_period = vif->assoc_bss_dtim_period;
1693 sinfo->bss_param.beacon_interval = vif->assoc_bss_beacon_int;
Jouni Malinen32c10872011-09-19 19:15:07 +03001694 }
1695
Kalle Valobdcd8172011-07-18 00:22:30 +03001696 return 0;
1697}
1698
1699static int ath6kl_set_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1700 struct cfg80211_pmksa *pmksa)
1701{
1702 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301703 struct ath6kl_vif *vif = netdev_priv(netdev);
1704
1705 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001706 pmksa->pmkid, true);
1707}
1708
1709static int ath6kl_del_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1710 struct cfg80211_pmksa *pmksa)
1711{
1712 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301713 struct ath6kl_vif *vif = netdev_priv(netdev);
1714
1715 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001716 pmksa->pmkid, false);
1717}
1718
1719static int ath6kl_flush_pmksa(struct wiphy *wiphy, struct net_device *netdev)
1720{
1721 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301722 struct ath6kl_vif *vif = netdev_priv(netdev);
1723
1724 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301725 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx,
1726 vif->bssid, NULL, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001727 return 0;
1728}
1729
Raja Manid91e8ee2012-01-30 17:13:10 +05301730static int ath6kl_wow_usr(struct ath6kl *ar, struct ath6kl_vif *vif,
1731 struct cfg80211_wowlan *wow, u32 *filter)
Raja Mani6cb3c712011-11-07 22:52:45 +02001732{
Raja Manid91e8ee2012-01-30 17:13:10 +05301733 int ret, pos;
1734 u8 mask[WOW_MASK_SIZE];
Raja Mani6cb3c712011-11-07 22:52:45 +02001735 u16 i;
Raja Mani6cb3c712011-11-07 22:52:45 +02001736
Raja Manid91e8ee2012-01-30 17:13:10 +05301737 /* Configure the patterns that we received from the user. */
Raja Mani6cb3c712011-11-07 22:52:45 +02001738 for (i = 0; i < wow->n_patterns; i++) {
1739
1740 /*
1741 * Convert given nl80211 specific mask value to equivalent
1742 * driver specific mask value and send it to the chip along
1743 * with patterns. For example, If the mask value defined in
1744 * struct cfg80211_wowlan is 0xA (equivalent binary is 1010),
1745 * then equivalent driver specific mask value is
1746 * "0xFF 0x00 0xFF 0x00".
1747 */
1748 memset(&mask, 0, sizeof(mask));
1749 for (pos = 0; pos < wow->patterns[i].pattern_len; pos++) {
1750 if (wow->patterns[i].mask[pos / 8] & (0x1 << (pos % 8)))
1751 mask[pos] = 0xFF;
1752 }
1753 /*
1754 * Note: Pattern's offset is not passed as part of wowlan
1755 * parameter from CFG layer. So it's always passed as ZERO
1756 * to the firmware. It means, given WOW patterns are always
1757 * matched from the first byte of received pkt in the firmware.
1758 */
1759 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
Raja Manid91e8ee2012-01-30 17:13:10 +05301760 vif->fw_vif_idx, WOW_LIST_ID,
1761 wow->patterns[i].pattern_len,
1762 0 /* pattern offset */,
1763 wow->patterns[i].pattern, mask);
Raja Mani6cb3c712011-11-07 22:52:45 +02001764 if (ret)
1765 return ret;
1766 }
1767
Raja Manid91e8ee2012-01-30 17:13:10 +05301768 if (wow->disconnect)
1769 *filter |= WOW_FILTER_OPTION_NWK_DISASSOC;
1770
1771 if (wow->magic_pkt)
1772 *filter |= WOW_FILTER_OPTION_MAGIC_PACKET;
1773
1774 if (wow->gtk_rekey_failure)
1775 *filter |= WOW_FILTER_OPTION_GTK_ERROR;
1776
1777 if (wow->eap_identity_req)
1778 *filter |= WOW_FILTER_OPTION_EAP_REQ;
1779
1780 if (wow->four_way_handshake)
1781 *filter |= WOW_FILTER_OPTION_8021X_4WAYHS;
1782
1783 return 0;
1784}
1785
1786static int ath6kl_wow_ap(struct ath6kl *ar, struct ath6kl_vif *vif)
1787{
1788 static const u8 unicst_pattern[] = { 0x00, 0x00, 0x00,
1789 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1790 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1791 0x00, 0x08 };
1792 static const u8 unicst_mask[] = { 0x01, 0x00, 0x00,
1793 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1794 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1795 0x00, 0x7f };
1796 u8 unicst_offset = 0;
1797 static const u8 arp_pattern[] = { 0x08, 0x06 };
1798 static const u8 arp_mask[] = { 0xff, 0xff };
1799 u8 arp_offset = 20;
1800 static const u8 discvr_pattern[] = { 0xe0, 0x00, 0x00, 0xf8 };
1801 static const u8 discvr_mask[] = { 0xf0, 0x00, 0x00, 0xf8 };
1802 u8 discvr_offset = 38;
1803 static const u8 dhcp_pattern[] = { 0xff, 0xff, 0xff, 0xff,
1804 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1805 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x08, 0x00,
1806 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1807 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1808 0x00, 0x00, 0x00, 0x00, 0x00, 0x43 /* port 67 */ };
1809 static const u8 dhcp_mask[] = { 0xff, 0xff, 0xff, 0xff,
1810 0xff, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1811 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff,
1812 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1813 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1814 0x00, 0x00, 0x00, 0x00, 0xff, 0xff /* port 67 */ };
1815 u8 dhcp_offset = 0;
1816 int ret;
1817
1818 /* Setup unicast IP, EAPOL-like and ARP pkt pattern */
1819 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1820 vif->fw_vif_idx, WOW_LIST_ID,
1821 sizeof(unicst_pattern), unicst_offset,
1822 unicst_pattern, unicst_mask);
1823 if (ret) {
1824 ath6kl_err("failed to add WOW unicast IP pattern\n");
1825 return ret;
1826 }
1827
1828 /* Setup all ARP pkt pattern */
1829 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1830 vif->fw_vif_idx, WOW_LIST_ID,
1831 sizeof(arp_pattern), arp_offset,
1832 arp_pattern, arp_mask);
1833 if (ret) {
1834 ath6kl_err("failed to add WOW ARP pattern\n");
1835 return ret;
1836 }
1837
1838 /*
1839 * Setup multicast pattern for mDNS 224.0.0.251,
1840 * SSDP 239.255.255.250 and LLMNR 224.0.0.252
1841 */
1842 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1843 vif->fw_vif_idx, WOW_LIST_ID,
1844 sizeof(discvr_pattern), discvr_offset,
1845 discvr_pattern, discvr_mask);
1846 if (ret) {
1847 ath6kl_err("failed to add WOW mDNS/SSDP/LLMNR pattern\n");
1848 return ret;
1849 }
1850
1851 /* Setup all DHCP broadcast pkt pattern */
1852 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1853 vif->fw_vif_idx, WOW_LIST_ID,
1854 sizeof(dhcp_pattern), dhcp_offset,
1855 dhcp_pattern, dhcp_mask);
1856 if (ret) {
1857 ath6kl_err("failed to add WOW DHCP broadcast pattern\n");
1858 return ret;
1859 }
1860
1861 return 0;
1862}
1863
1864static int ath6kl_wow_sta(struct ath6kl *ar, struct ath6kl_vif *vif)
1865{
1866 struct net_device *ndev = vif->ndev;
1867 static const u8 discvr_pattern[] = { 0xe0, 0x00, 0x00, 0xf8 };
1868 static const u8 discvr_mask[] = { 0xf0, 0x00, 0x00, 0xf8 };
1869 u8 discvr_offset = 38;
1870 u8 mac_mask[ETH_ALEN];
1871 int ret;
1872
1873 /* Setup unicast pkt pattern */
1874 memset(mac_mask, 0xff, ETH_ALEN);
1875 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1876 vif->fw_vif_idx, WOW_LIST_ID,
1877 ETH_ALEN, 0, ndev->dev_addr,
1878 mac_mask);
1879 if (ret) {
1880 ath6kl_err("failed to add WOW unicast pattern\n");
1881 return ret;
1882 }
1883
1884 /*
1885 * Setup multicast pattern for mDNS 224.0.0.251,
1886 * SSDP 239.255.255.250 and LLMNR 224.0.0.252
1887 */
1888 if ((ndev->flags & IFF_ALLMULTI) ||
1889 (ndev->flags & IFF_MULTICAST && netdev_mc_count(ndev) > 0)) {
1890 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1891 vif->fw_vif_idx, WOW_LIST_ID,
1892 sizeof(discvr_pattern), discvr_offset,
1893 discvr_pattern, discvr_mask);
1894 if (ret) {
1895 ath6kl_err("failed to add WOW mDNS/SSDP/LLMNR "
1896 "pattern\n");
1897 return ret;
1898 }
1899 }
1900
1901 return 0;
1902}
1903
1904static int ath6kl_wow_suspend(struct ath6kl *ar, struct cfg80211_wowlan *wow)
1905{
1906 struct in_device *in_dev;
1907 struct in_ifaddr *ifa;
1908 struct ath6kl_vif *vif;
1909 int ret, left;
1910 u32 filter = 0;
1911 u16 i;
1912 u8 index = 0;
1913 __be32 ips[MAX_IP_ADDRS];
1914
1915 vif = ath6kl_vif_first(ar);
1916 if (!vif)
1917 return -EIO;
1918
1919 if (!ath6kl_cfg80211_ready(vif))
1920 return -EIO;
1921
1922 if (!test_bit(CONNECTED, &vif->flags))
Raja Mani3c411a42012-01-30 17:13:12 +05301923 return -ENOTCONN;
Raja Manid91e8ee2012-01-30 17:13:10 +05301924
1925 if (wow && (wow->n_patterns > WOW_MAX_FILTERS_PER_LIST))
1926 return -EINVAL;
1927
1928 /* Clear existing WOW patterns */
1929 for (i = 0; i < WOW_MAX_FILTERS_PER_LIST; i++)
1930 ath6kl_wmi_del_wow_pattern_cmd(ar->wmi, vif->fw_vif_idx,
1931 WOW_LIST_ID, i);
1932
1933 /*
1934 * Skip the default WOW pattern configuration
1935 * if the driver receives any WOW patterns from
1936 * the user.
1937 */
1938 if (wow)
1939 ret = ath6kl_wow_usr(ar, vif, wow, &filter);
1940 else if (vif->nw_type == AP_NETWORK)
1941 ret = ath6kl_wow_ap(ar, vif);
1942 else
1943 ret = ath6kl_wow_sta(ar, vif);
1944
1945 if (ret)
1946 return ret;
1947
Raja Manic08631c2011-12-16 14:24:24 +05301948 /* Setup own IP addr for ARP agent. */
1949 in_dev = __in_dev_get_rtnl(vif->ndev);
1950 if (!in_dev)
1951 goto skip_arp;
1952
1953 ifa = in_dev->ifa_list;
1954 memset(&ips, 0, sizeof(ips));
1955
1956 /* Configure IP addr only if IP address count < MAX_IP_ADDRS */
1957 while (index < MAX_IP_ADDRS && ifa) {
1958 ips[index] = ifa->ifa_local;
1959 ifa = ifa->ifa_next;
1960 index++;
1961 }
1962
1963 if (ifa) {
1964 ath6kl_err("total IP addr count is exceeding fw limit\n");
1965 return -EINVAL;
1966 }
1967
1968 ret = ath6kl_wmi_set_ip_cmd(ar->wmi, vif->fw_vif_idx, ips[0], ips[1]);
1969 if (ret) {
1970 ath6kl_err("fail to setup ip for arp agent\n");
1971 return ret;
1972 }
1973
1974skip_arp:
Raja Mani6cb3c712011-11-07 22:52:45 +02001975 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
1976 ATH6KL_WOW_MODE_ENABLE,
1977 filter,
1978 WOW_HOST_REQ_DELAY);
1979 if (ret)
1980 return ret;
1981
Raja Mani081c7a82012-01-30 17:13:11 +05301982 clear_bit(HOST_SLEEP_MODE_CMD_PROCESSED, &vif->flags);
1983
Raja Mani6cb3c712011-11-07 22:52:45 +02001984 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
1985 ATH6KL_HOST_MODE_ASLEEP);
1986 if (ret)
1987 return ret;
1988
Raja Mani081c7a82012-01-30 17:13:11 +05301989 left = wait_event_interruptible_timeout(ar->event_wq,
1990 test_bit(HOST_SLEEP_MODE_CMD_PROCESSED, &vif->flags),
1991 WMI_TIMEOUT);
1992 if (left == 0) {
1993 ath6kl_warn("timeout, didn't get host sleep cmd "
1994 "processed event\n");
1995 ret = -ETIMEDOUT;
1996 } else if (left < 0) {
1997 ath6kl_warn("error while waiting for host sleep cmd "
1998 "processed event %d\n", left);
1999 ret = left;
2000 }
2001
Raja Mani6cb3c712011-11-07 22:52:45 +02002002 if (ar->tx_pending[ar->ctrl_ep]) {
2003 left = wait_event_interruptible_timeout(ar->event_wq,
2004 ar->tx_pending[ar->ctrl_ep] == 0, WMI_TIMEOUT);
2005 if (left == 0) {
2006 ath6kl_warn("clear wmi ctrl data timeout\n");
2007 ret = -ETIMEDOUT;
2008 } else if (left < 0) {
2009 ath6kl_warn("clear wmi ctrl data failed: %d\n", left);
2010 ret = left;
2011 }
2012 }
2013
2014 return ret;
2015}
2016
2017static int ath6kl_wow_resume(struct ath6kl *ar)
2018{
2019 struct ath6kl_vif *vif;
2020 int ret;
2021
2022 vif = ath6kl_vif_first(ar);
2023 if (!vif)
2024 return -EIO;
2025
2026 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2027 ATH6KL_HOST_MODE_AWAKE);
2028 return ret;
2029}
2030
Kalle Valo52d81a62011-11-01 08:44:21 +02002031int ath6kl_cfg80211_suspend(struct ath6kl *ar,
Raja Mani0f60e9f2011-11-07 22:52:45 +02002032 enum ath6kl_cfg_suspend_mode mode,
2033 struct cfg80211_wowlan *wow)
Kalle Valo52d81a62011-11-01 08:44:21 +02002034{
2035 int ret;
2036
Kalle Valo52d81a62011-11-01 08:44:21 +02002037 switch (mode) {
Raja Manid7c44e02011-11-07 22:52:46 +02002038 case ATH6KL_CFG_SUSPEND_WOW:
2039
2040 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode suspend\n");
2041
2042 /* Flush all non control pkts in TX path */
2043 ath6kl_tx_data_cleanup(ar);
2044
2045 ret = ath6kl_wow_suspend(ar, wow);
2046 if (ret) {
2047 ath6kl_err("wow suspend failed: %d\n", ret);
2048 return ret;
2049 }
2050 ar->state = ATH6KL_STATE_WOW;
2051 break;
2052
Kalle Valo52d81a62011-11-01 08:44:21 +02002053 case ATH6KL_CFG_SUSPEND_DEEPSLEEP:
Raja Mani524441e2011-11-07 22:52:46 +02002054
Kalle Valo7125f012011-12-13 14:51:37 +02002055 ath6kl_cfg80211_stop_all(ar);
Raja Mani524441e2011-11-07 22:52:46 +02002056
Kalle Valo52d81a62011-11-01 08:44:21 +02002057 /* save the current power mode before enabling power save */
2058 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
2059
2060 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER);
2061 if (ret) {
2062 ath6kl_warn("wmi powermode command failed during suspend: %d\n",
2063 ret);
2064 }
2065
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002066 ar->state = ATH6KL_STATE_DEEPSLEEP;
2067
Kalle Valo52d81a62011-11-01 08:44:21 +02002068 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002069
2070 case ATH6KL_CFG_SUSPEND_CUTPOWER:
Raja Mani524441e2011-11-07 22:52:46 +02002071
Kalle Valo7125f012011-12-13 14:51:37 +02002072 ath6kl_cfg80211_stop_all(ar);
Raja Mani524441e2011-11-07 22:52:46 +02002073
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002074 if (ar->state == ATH6KL_STATE_OFF) {
2075 ath6kl_dbg(ATH6KL_DBG_SUSPEND,
2076 "suspend hw off, no action for cutpower\n");
2077 break;
2078 }
2079
2080 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "suspend cutting power\n");
2081
2082 ret = ath6kl_init_hw_stop(ar);
2083 if (ret) {
2084 ath6kl_warn("failed to stop hw during suspend: %d\n",
2085 ret);
2086 }
2087
2088 ar->state = ATH6KL_STATE_CUTPOWER;
2089
2090 break;
2091
Kalle Valo10509f92011-12-13 14:52:07 +02002092 case ATH6KL_CFG_SUSPEND_SCHED_SCAN:
2093 /*
2094 * Nothing needed for schedule scan, firmware is already in
2095 * wow mode and sleeping most of the time.
2096 */
2097 break;
2098
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002099 default:
2100 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02002101 }
2102
2103 return 0;
2104}
Kalle Valod6a434d2012-01-17 20:09:36 +02002105EXPORT_SYMBOL(ath6kl_cfg80211_suspend);
Kalle Valo52d81a62011-11-01 08:44:21 +02002106
2107int ath6kl_cfg80211_resume(struct ath6kl *ar)
2108{
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002109 int ret;
2110
2111 switch (ar->state) {
Raja Manid7c44e02011-11-07 22:52:46 +02002112 case ATH6KL_STATE_WOW:
2113 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode resume\n");
2114
2115 ret = ath6kl_wow_resume(ar);
2116 if (ret) {
2117 ath6kl_warn("wow mode resume failed: %d\n", ret);
2118 return ret;
2119 }
2120
2121 ar->state = ATH6KL_STATE_ON;
2122 break;
2123
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002124 case ATH6KL_STATE_DEEPSLEEP:
2125 if (ar->wmi->pwr_mode != ar->wmi->saved_pwr_mode) {
2126 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0,
2127 ar->wmi->saved_pwr_mode);
2128 if (ret) {
2129 ath6kl_warn("wmi powermode command failed during resume: %d\n",
2130 ret);
2131 }
2132 }
2133
2134 ar->state = ATH6KL_STATE_ON;
2135
2136 break;
2137
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002138 case ATH6KL_STATE_CUTPOWER:
2139 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "resume restoring power\n");
2140
2141 ret = ath6kl_init_hw_start(ar);
2142 if (ret) {
2143 ath6kl_warn("Failed to boot hw in resume: %d\n", ret);
2144 return ret;
2145 }
Raja Manid7c44e02011-11-07 22:52:46 +02002146 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02002147
Kalle Valo10509f92011-12-13 14:52:07 +02002148 case ATH6KL_STATE_SCHED_SCAN:
2149 break;
2150
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002151 default:
2152 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02002153 }
2154
2155 return 0;
2156}
Kalle Valod6a434d2012-01-17 20:09:36 +02002157EXPORT_SYMBOL(ath6kl_cfg80211_resume);
Kalle Valo52d81a62011-11-01 08:44:21 +02002158
Kalle Valoabcb3442011-07-22 08:26:20 +03002159#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02002160
2161/* hif layer decides what suspend mode to use */
2162static int __ath6kl_cfg80211_suspend(struct wiphy *wiphy,
Kalle Valoabcb3442011-07-22 08:26:20 +03002163 struct cfg80211_wowlan *wow)
2164{
2165 struct ath6kl *ar = wiphy_priv(wiphy);
2166
Raja Mani0f60e9f2011-11-07 22:52:45 +02002167 return ath6kl_hif_suspend(ar, wow);
Kalle Valoabcb3442011-07-22 08:26:20 +03002168}
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002169
Kalle Valo52d81a62011-11-01 08:44:21 +02002170static int __ath6kl_cfg80211_resume(struct wiphy *wiphy)
Chilam Ngaa6cffc2011-10-05 10:12:52 +03002171{
2172 struct ath6kl *ar = wiphy_priv(wiphy);
2173
2174 return ath6kl_hif_resume(ar);
2175}
Raja Mania918fb32011-11-07 22:52:46 +02002176
2177/*
2178 * FIXME: WOW suspend mode is selected if the host sdio controller supports
2179 * both sdio irq wake up and keep power. The target pulls sdio data line to
2180 * wake up the host when WOW pattern matches. This causes sdio irq handler
2181 * is being called in the host side which internally hits ath6kl's RX path.
2182 *
2183 * Since sdio interrupt is not disabled, RX path executes even before
2184 * the host executes the actual resume operation from PM module.
2185 *
2186 * In the current scenario, WOW resume should happen before start processing
2187 * any data from the target. So It's required to perform WOW resume in RX path.
2188 * Ideally we should perform WOW resume only in the actual platform
2189 * resume path. This area needs bit rework to avoid WOW resume in RX path.
2190 *
2191 * ath6kl_check_wow_status() is called from ath6kl_rx().
2192 */
2193void ath6kl_check_wow_status(struct ath6kl *ar)
2194{
2195 if (ar->state == ATH6KL_STATE_WOW)
2196 ath6kl_cfg80211_resume(ar);
2197}
2198
2199#else
2200
2201void ath6kl_check_wow_status(struct ath6kl *ar)
2202{
2203}
Kalle Valoabcb3442011-07-22 08:26:20 +03002204#endif
2205
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002206static int ath6kl_set_channel(struct wiphy *wiphy, struct net_device *dev,
2207 struct ieee80211_channel *chan,
2208 enum nl80211_channel_type channel_type)
2209{
Sujith Manoharane68f6752011-12-22 12:15:27 +05302210 struct ath6kl_vif *vif;
2211
2212 /*
2213 * 'dev' could be NULL if a channel change is required for the hardware
2214 * device itself, instead of a particular VIF.
2215 *
2216 * FIXME: To be handled properly when monitor mode is supported.
2217 */
2218 if (!dev)
2219 return -EBUSY;
2220
2221 vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002222
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302223 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002224 return -EIO;
2225
2226 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: center_freq=%u hw_value=%u\n",
2227 __func__, chan->center_freq, chan->hw_value);
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302228 vif->next_chan = chan->center_freq;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002229
2230 return 0;
2231}
2232
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002233static bool ath6kl_is_p2p_ie(const u8 *pos)
2234{
2235 return pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
2236 pos[2] == 0x50 && pos[3] == 0x6f &&
2237 pos[4] == 0x9a && pos[5] == 0x09;
2238}
2239
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302240static int ath6kl_set_ap_probe_resp_ies(struct ath6kl_vif *vif,
2241 const u8 *ies, size_t ies_len)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002242{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302243 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002244 const u8 *pos;
2245 u8 *buf = NULL;
2246 size_t len = 0;
2247 int ret;
2248
2249 /*
2250 * Filter out P2P IE(s) since they will be included depending on
2251 * the Probe Request frame in ath6kl_send_go_probe_resp().
2252 */
2253
2254 if (ies && ies_len) {
2255 buf = kmalloc(ies_len, GFP_KERNEL);
2256 if (buf == NULL)
2257 return -ENOMEM;
2258 pos = ies;
2259 while (pos + 1 < ies + ies_len) {
2260 if (pos + 2 + pos[1] > ies + ies_len)
2261 break;
2262 if (!ath6kl_is_p2p_ie(pos)) {
2263 memcpy(buf + len, pos, 2 + pos[1]);
2264 len += 2 + pos[1];
2265 }
2266 pos += 2 + pos[1];
2267 }
2268 }
2269
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302270 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2271 WMI_FRAME_PROBE_RESP, buf, len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002272 kfree(buf);
2273 return ret;
2274}
2275
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002276static int ath6kl_ap_beacon(struct wiphy *wiphy, struct net_device *dev,
2277 struct beacon_parameters *info, bool add)
2278{
2279 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302280 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002281 struct ieee80211_mgmt *mgmt;
2282 u8 *ies;
2283 int ies_len;
2284 struct wmi_connect_cmd p;
2285 int res;
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302286 int i, ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002287
2288 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: add=%d\n", __func__, add);
2289
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302290 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002291 return -EIO;
2292
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302293 if (vif->next_mode != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002294 return -EOPNOTSUPP;
2295
2296 if (info->beacon_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302297 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2298 WMI_FRAME_BEACON,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002299 info->beacon_ies,
2300 info->beacon_ies_len);
2301 if (res)
2302 return res;
2303 }
2304 if (info->proberesp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302305 res = ath6kl_set_ap_probe_resp_ies(vif, info->proberesp_ies,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002306 info->proberesp_ies_len);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002307 if (res)
2308 return res;
2309 }
2310 if (info->assocresp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302311 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2312 WMI_FRAME_ASSOC_RESP,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002313 info->assocresp_ies,
2314 info->assocresp_ies_len);
2315 if (res)
2316 return res;
2317 }
2318
2319 if (!add)
2320 return 0;
2321
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002322 ar->ap_mode_bkey.valid = false;
2323
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002324 /* TODO:
2325 * info->interval
2326 * info->dtim_period
2327 */
2328
2329 if (info->head == NULL)
2330 return -EINVAL;
2331 mgmt = (struct ieee80211_mgmt *) info->head;
2332 ies = mgmt->u.beacon.variable;
2333 if (ies > info->head + info->head_len)
2334 return -EINVAL;
2335 ies_len = info->head + info->head_len - ies;
2336
2337 if (info->ssid == NULL)
2338 return -EINVAL;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302339 memcpy(vif->ssid, info->ssid, info->ssid_len);
2340 vif->ssid_len = info->ssid_len;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002341 if (info->hidden_ssid != NL80211_HIDDEN_SSID_NOT_IN_USE)
2342 return -EOPNOTSUPP; /* TODO */
2343
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302344 ret = ath6kl_set_auth_type(vif, info->auth_type);
2345 if (ret)
2346 return ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002347
2348 memset(&p, 0, sizeof(p));
2349
2350 for (i = 0; i < info->crypto.n_akm_suites; i++) {
2351 switch (info->crypto.akm_suites[i]) {
2352 case WLAN_AKM_SUITE_8021X:
2353 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2354 p.auth_mode |= WPA_AUTH;
2355 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2356 p.auth_mode |= WPA2_AUTH;
2357 break;
2358 case WLAN_AKM_SUITE_PSK:
2359 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2360 p.auth_mode |= WPA_PSK_AUTH;
2361 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2362 p.auth_mode |= WPA2_PSK_AUTH;
2363 break;
2364 }
2365 }
2366 if (p.auth_mode == 0)
2367 p.auth_mode = NONE_AUTH;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302368 vif->auth_mode = p.auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002369
2370 for (i = 0; i < info->crypto.n_ciphers_pairwise; i++) {
2371 switch (info->crypto.ciphers_pairwise[i]) {
2372 case WLAN_CIPHER_SUITE_WEP40:
2373 case WLAN_CIPHER_SUITE_WEP104:
2374 p.prwise_crypto_type |= WEP_CRYPT;
2375 break;
2376 case WLAN_CIPHER_SUITE_TKIP:
2377 p.prwise_crypto_type |= TKIP_CRYPT;
2378 break;
2379 case WLAN_CIPHER_SUITE_CCMP:
2380 p.prwise_crypto_type |= AES_CRYPT;
2381 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002382 case WLAN_CIPHER_SUITE_SMS4:
2383 p.prwise_crypto_type |= WAPI_CRYPT;
2384 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002385 }
2386 }
Edward Lu229ed6b2011-08-30 21:58:07 +03002387 if (p.prwise_crypto_type == 0) {
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002388 p.prwise_crypto_type = NONE_CRYPT;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302389 ath6kl_set_cipher(vif, 0, true);
Edward Lu229ed6b2011-08-30 21:58:07 +03002390 } else if (info->crypto.n_ciphers_pairwise == 1)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302391 ath6kl_set_cipher(vif, info->crypto.ciphers_pairwise[0], true);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002392
2393 switch (info->crypto.cipher_group) {
2394 case WLAN_CIPHER_SUITE_WEP40:
2395 case WLAN_CIPHER_SUITE_WEP104:
2396 p.grp_crypto_type = WEP_CRYPT;
2397 break;
2398 case WLAN_CIPHER_SUITE_TKIP:
2399 p.grp_crypto_type = TKIP_CRYPT;
2400 break;
2401 case WLAN_CIPHER_SUITE_CCMP:
2402 p.grp_crypto_type = AES_CRYPT;
2403 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002404 case WLAN_CIPHER_SUITE_SMS4:
2405 p.grp_crypto_type = WAPI_CRYPT;
2406 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002407 default:
2408 p.grp_crypto_type = NONE_CRYPT;
2409 break;
2410 }
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302411 ath6kl_set_cipher(vif, info->crypto.cipher_group, false);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002412
2413 p.nw_type = AP_NETWORK;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302414 vif->nw_type = vif->next_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002415
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302416 p.ssid_len = vif->ssid_len;
2417 memcpy(p.ssid, vif->ssid, vif->ssid_len);
2418 p.dot11_auth_mode = vif->dot11_auth_mode;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302419 p.ch = cpu_to_le16(vif->next_chan);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002420
Thirumalai Pachamuthuc1762a32012-01-12 18:21:39 +05302421 /* Enable uAPSD support by default */
2422 res = ath6kl_wmi_ap_set_apsd(ar->wmi, vif->fw_vif_idx, true);
2423 if (res < 0)
2424 return res;
2425
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002426 if (vif->wdev.iftype == NL80211_IFTYPE_P2P_GO) {
2427 p.nw_subtype = SUBTYPE_P2PGO;
2428 } else {
2429 /*
2430 * Due to firmware limitation, it is not possible to
2431 * do P2P mgmt operations in AP mode
2432 */
2433 p.nw_subtype = SUBTYPE_NONE;
2434 }
2435
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302436 res = ath6kl_wmi_ap_profile_commit(ar->wmi, vif->fw_vif_idx, &p);
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002437 if (res < 0)
2438 return res;
2439
2440 return 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002441}
2442
2443static int ath6kl_add_beacon(struct wiphy *wiphy, struct net_device *dev,
2444 struct beacon_parameters *info)
2445{
2446 return ath6kl_ap_beacon(wiphy, dev, info, true);
2447}
2448
2449static int ath6kl_set_beacon(struct wiphy *wiphy, struct net_device *dev,
2450 struct beacon_parameters *info)
2451{
2452 return ath6kl_ap_beacon(wiphy, dev, info, false);
2453}
2454
2455static int ath6kl_del_beacon(struct wiphy *wiphy, struct net_device *dev)
2456{
2457 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302458 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002459
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302460 if (vif->nw_type != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002461 return -EOPNOTSUPP;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302462 if (!test_bit(CONNECTED, &vif->flags))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002463 return -ENOTCONN;
2464
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302465 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302466 clear_bit(CONNECTED, &vif->flags);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002467
2468 return 0;
2469}
2470
Jouni Malinen33e53082011-12-27 11:02:56 +02002471static const u8 bcast_addr[ETH_ALEN] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
2472
2473static int ath6kl_del_station(struct wiphy *wiphy, struct net_device *dev,
2474 u8 *mac)
2475{
2476 struct ath6kl *ar = ath6kl_priv(dev);
2477 struct ath6kl_vif *vif = netdev_priv(dev);
2478 const u8 *addr = mac ? mac : bcast_addr;
2479
2480 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx, WMI_AP_DEAUTH,
2481 addr, WLAN_REASON_PREV_AUTH_NOT_VALID);
2482}
2483
Jouni Malinen23875132011-08-30 21:57:53 +03002484static int ath6kl_change_station(struct wiphy *wiphy, struct net_device *dev,
2485 u8 *mac, struct station_parameters *params)
2486{
2487 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302488 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen23875132011-08-30 21:57:53 +03002489
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302490 if (vif->nw_type != AP_NETWORK)
Jouni Malinen23875132011-08-30 21:57:53 +03002491 return -EOPNOTSUPP;
2492
2493 /* Use this only for authorizing/unauthorizing a station */
2494 if (!(params->sta_flags_mask & BIT(NL80211_STA_FLAG_AUTHORIZED)))
2495 return -EOPNOTSUPP;
2496
2497 if (params->sta_flags_set & BIT(NL80211_STA_FLAG_AUTHORIZED))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302498 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2499 WMI_AP_MLME_AUTHORIZE, mac, 0);
2500 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2501 WMI_AP_MLME_UNAUTHORIZE, mac, 0);
Jouni Malinen23875132011-08-30 21:57:53 +03002502}
2503
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002504static int ath6kl_remain_on_channel(struct wiphy *wiphy,
2505 struct net_device *dev,
2506 struct ieee80211_channel *chan,
2507 enum nl80211_channel_type channel_type,
2508 unsigned int duration,
2509 u64 *cookie)
2510{
2511 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302512 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen10522612011-10-27 16:00:13 +03002513 u32 id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002514
2515 /* TODO: if already pending or ongoing remain-on-channel,
2516 * return -EBUSY */
Jouni Malinen10522612011-10-27 16:00:13 +03002517 id = ++vif->last_roc_id;
2518 if (id == 0) {
2519 /* Do not use 0 as the cookie value */
2520 id = ++vif->last_roc_id;
2521 }
2522 *cookie = id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002523
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302524 return ath6kl_wmi_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx,
2525 chan->center_freq, duration);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002526}
2527
2528static int ath6kl_cancel_remain_on_channel(struct wiphy *wiphy,
2529 struct net_device *dev,
2530 u64 cookie)
2531{
2532 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302533 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002534
Jouni Malinen10522612011-10-27 16:00:13 +03002535 if (cookie != vif->last_roc_id)
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002536 return -ENOENT;
Jouni Malinen10522612011-10-27 16:00:13 +03002537 vif->last_cancel_roc_id = cookie;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002538
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302539 return ath6kl_wmi_cancel_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002540}
2541
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302542static int ath6kl_send_go_probe_resp(struct ath6kl_vif *vif,
2543 const u8 *buf, size_t len,
2544 unsigned int freq)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002545{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302546 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002547 const u8 *pos;
2548 u8 *p2p;
2549 int p2p_len;
2550 int ret;
2551 const struct ieee80211_mgmt *mgmt;
2552
2553 mgmt = (const struct ieee80211_mgmt *) buf;
2554
2555 /* Include P2P IE(s) from the frame generated in user space. */
2556
2557 p2p = kmalloc(len, GFP_KERNEL);
2558 if (p2p == NULL)
2559 return -ENOMEM;
2560 p2p_len = 0;
2561
2562 pos = mgmt->u.probe_resp.variable;
2563 while (pos + 1 < buf + len) {
2564 if (pos + 2 + pos[1] > buf + len)
2565 break;
2566 if (ath6kl_is_p2p_ie(pos)) {
2567 memcpy(p2p + p2p_len, pos, 2 + pos[1]);
2568 p2p_len += 2 + pos[1];
2569 }
2570 pos += 2 + pos[1];
2571 }
2572
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302573 ret = ath6kl_wmi_send_probe_response_cmd(ar->wmi, vif->fw_vif_idx, freq,
2574 mgmt->da, p2p, p2p_len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002575 kfree(p2p);
2576 return ret;
2577}
2578
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08002579static bool ath6kl_mgmt_powersave_ap(struct ath6kl_vif *vif,
2580 u32 id,
2581 u32 freq,
2582 u32 wait,
2583 const u8 *buf,
2584 size_t len,
2585 bool *more_data,
2586 bool no_cck)
2587{
2588 struct ieee80211_mgmt *mgmt;
2589 struct ath6kl_sta *conn;
2590 bool is_psq_empty = false;
2591 struct ath6kl_mgmt_buff *mgmt_buf;
2592 size_t mgmt_buf_size;
2593 struct ath6kl *ar = vif->ar;
2594
2595 mgmt = (struct ieee80211_mgmt *) buf;
2596 if (is_multicast_ether_addr(mgmt->da))
2597 return false;
2598
2599 conn = ath6kl_find_sta(vif, mgmt->da);
2600 if (!conn)
2601 return false;
2602
2603 if (conn->sta_flags & STA_PS_SLEEP) {
2604 if (!(conn->sta_flags & STA_PS_POLLED)) {
2605 /* Queue the frames if the STA is sleeping */
2606 mgmt_buf_size = len + sizeof(struct ath6kl_mgmt_buff);
2607 mgmt_buf = kmalloc(mgmt_buf_size, GFP_KERNEL);
2608 if (!mgmt_buf)
2609 return false;
2610
2611 INIT_LIST_HEAD(&mgmt_buf->list);
2612 mgmt_buf->id = id;
2613 mgmt_buf->freq = freq;
2614 mgmt_buf->wait = wait;
2615 mgmt_buf->len = len;
2616 mgmt_buf->no_cck = no_cck;
2617 memcpy(mgmt_buf->buf, buf, len);
2618 spin_lock_bh(&conn->psq_lock);
2619 is_psq_empty = skb_queue_empty(&conn->psq) &&
2620 (conn->mgmt_psq_len == 0);
2621 list_add_tail(&mgmt_buf->list, &conn->mgmt_psq);
2622 conn->mgmt_psq_len++;
2623 spin_unlock_bh(&conn->psq_lock);
2624
2625 /*
2626 * If this is the first pkt getting queued
2627 * for this STA, update the PVB for this
2628 * STA.
2629 */
2630 if (is_psq_empty)
2631 ath6kl_wmi_set_pvb_cmd(ar->wmi, vif->fw_vif_idx,
2632 conn->aid, 1);
2633 return true;
2634 }
2635
2636 /*
2637 * This tx is because of a PsPoll.
2638 * Determine if MoreData bit has to be set.
2639 */
2640 spin_lock_bh(&conn->psq_lock);
2641 if (!skb_queue_empty(&conn->psq) || (conn->mgmt_psq_len != 0))
2642 *more_data = true;
2643 spin_unlock_bh(&conn->psq_lock);
2644 }
2645
2646 return false;
2647}
2648
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002649static int ath6kl_mgmt_tx(struct wiphy *wiphy, struct net_device *dev,
2650 struct ieee80211_channel *chan, bool offchan,
2651 enum nl80211_channel_type channel_type,
2652 bool channel_type_valid, unsigned int wait,
Johannes Berge247bd902011-11-04 11:18:21 +01002653 const u8 *buf, size_t len, bool no_cck,
2654 bool dont_wait_for_ack, u64 *cookie)
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002655{
2656 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302657 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002658 u32 id;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002659 const struct ieee80211_mgmt *mgmt;
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08002660 bool more_data, queued;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002661
2662 mgmt = (const struct ieee80211_mgmt *) buf;
2663 if (buf + len >= mgmt->u.probe_resp.variable &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302664 vif->nw_type == AP_NETWORK && test_bit(CONNECTED, &vif->flags) &&
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002665 ieee80211_is_probe_resp(mgmt->frame_control)) {
2666 /*
2667 * Send Probe Response frame in AP mode using a separate WMI
2668 * command to allow the target to fill in the generic IEs.
2669 */
2670 *cookie = 0; /* TX status not supported */
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302671 return ath6kl_send_go_probe_resp(vif, buf, len,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002672 chan->center_freq);
2673 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002674
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302675 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002676 if (id == 0) {
2677 /*
2678 * 0 is a reserved value in the WMI command and shall not be
2679 * used for the command.
2680 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302681 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002682 }
2683
2684 *cookie = id;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002685
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08002686 /* AP mode Power saving processing */
2687 if (vif->nw_type == AP_NETWORK) {
2688 queued = ath6kl_mgmt_powersave_ap(vif,
2689 id, chan->center_freq,
2690 wait, buf,
2691 len, &more_data, no_cck);
2692 if (queued)
2693 return 0;
Aarthi Thiruvengadam3ca9d1f2011-12-13 13:32:12 -08002694 }
Naveen Gangadharand0ff7382012-02-08 17:51:36 -08002695
2696 return ath6kl_wmi_send_mgmt_cmd(ar->wmi, vif->fw_vif_idx, id,
2697 chan->center_freq, wait,
2698 buf, len, no_cck);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002699}
2700
Jouni Malinenae32c302011-08-30 21:58:01 +03002701static void ath6kl_mgmt_frame_register(struct wiphy *wiphy,
2702 struct net_device *dev,
2703 u16 frame_type, bool reg)
2704{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302705 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinenae32c302011-08-30 21:58:01 +03002706
2707 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: frame_type=0x%x reg=%d\n",
2708 __func__, frame_type, reg);
2709 if (frame_type == IEEE80211_STYPE_PROBE_REQ) {
2710 /*
2711 * Note: This notification callback is not allowed to sleep, so
2712 * we cannot send WMI_PROBE_REQ_REPORT_CMD here. Instead, we
2713 * hardcode target to report Probe Request frames all the time.
2714 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302715 vif->probe_req_report = reg;
Jouni Malinenae32c302011-08-30 21:58:01 +03002716 }
2717}
2718
Kalle Valo10509f92011-12-13 14:52:07 +02002719static int ath6kl_cfg80211_sscan_start(struct wiphy *wiphy,
2720 struct net_device *dev,
2721 struct cfg80211_sched_scan_request *request)
2722{
2723 struct ath6kl *ar = ath6kl_priv(dev);
2724 struct ath6kl_vif *vif = netdev_priv(dev);
2725 u16 interval;
2726 int ret;
2727 u8 i;
2728
2729 if (ar->state != ATH6KL_STATE_ON)
2730 return -EIO;
2731
2732 if (vif->sme_state != SME_DISCONNECTED)
2733 return -EBUSY;
2734
2735 for (i = 0; i < ar->wiphy->max_sched_scan_ssids; i++) {
2736 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
2737 i, DISABLE_SSID_FLAG,
2738 0, NULL);
2739 }
2740
2741 /* fw uses seconds, also make sure that it's >0 */
2742 interval = max_t(u16, 1, request->interval / 1000);
2743
2744 ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx,
2745 interval, interval,
2746 10, 0, 0, 0, 3, 0, 0, 0);
2747
2748 if (request->n_ssids && request->ssids[0].ssid_len) {
2749 for (i = 0; i < request->n_ssids; i++) {
2750 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
2751 i, SPECIFIC_SSID_FLAG,
2752 request->ssids[i].ssid_len,
2753 request->ssids[i].ssid);
2754 }
2755 }
2756
2757 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
2758 ATH6KL_WOW_MODE_ENABLE,
2759 WOW_FILTER_SSID,
2760 WOW_HOST_REQ_DELAY);
2761 if (ret) {
2762 ath6kl_warn("Failed to enable wow with ssid filter: %d\n", ret);
2763 return ret;
2764 }
2765
2766 /* this also clears IE in fw if it's not set */
2767 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2768 WMI_FRAME_PROBE_REQ,
2769 request->ie, request->ie_len);
2770 if (ret) {
2771 ath6kl_warn("Failed to set probe request IE for scheduled scan: %d",
2772 ret);
2773 return ret;
2774 }
2775
2776 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
2777 ATH6KL_HOST_MODE_ASLEEP);
2778 if (ret) {
2779 ath6kl_warn("Failed to enable host sleep mode for sched scan: %d\n",
2780 ret);
2781 return ret;
2782 }
2783
2784 ar->state = ATH6KL_STATE_SCHED_SCAN;
2785
2786 return ret;
2787}
2788
2789static int ath6kl_cfg80211_sscan_stop(struct wiphy *wiphy,
2790 struct net_device *dev)
2791{
2792 struct ath6kl_vif *vif = netdev_priv(dev);
2793 bool stopped;
2794
2795 stopped = __ath6kl_cfg80211_sscan_stop(vif);
2796
2797 if (!stopped)
2798 return -EIO;
2799
2800 return 0;
2801}
2802
Jouni Malinenf80574a2011-08-30 21:58:04 +03002803static const struct ieee80211_txrx_stypes
2804ath6kl_mgmt_stypes[NUM_NL80211_IFTYPES] = {
2805 [NL80211_IFTYPE_STATION] = {
2806 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2807 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2808 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2809 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2810 },
Jouni Malinenba1f6fe2011-12-27 11:03:53 +02002811 [NL80211_IFTYPE_AP] = {
2812 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2813 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2814 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2815 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2816 },
Jouni Malinenf80574a2011-08-30 21:58:04 +03002817 [NL80211_IFTYPE_P2P_CLIENT] = {
2818 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2819 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2820 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2821 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2822 },
2823 [NL80211_IFTYPE_P2P_GO] = {
2824 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2825 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2826 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2827 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2828 },
2829};
2830
Kalle Valobdcd8172011-07-18 00:22:30 +03002831static struct cfg80211_ops ath6kl_cfg80211_ops = {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302832 .add_virtual_intf = ath6kl_cfg80211_add_iface,
2833 .del_virtual_intf = ath6kl_cfg80211_del_iface,
Kalle Valobdcd8172011-07-18 00:22:30 +03002834 .change_virtual_intf = ath6kl_cfg80211_change_iface,
2835 .scan = ath6kl_cfg80211_scan,
2836 .connect = ath6kl_cfg80211_connect,
2837 .disconnect = ath6kl_cfg80211_disconnect,
2838 .add_key = ath6kl_cfg80211_add_key,
2839 .get_key = ath6kl_cfg80211_get_key,
2840 .del_key = ath6kl_cfg80211_del_key,
2841 .set_default_key = ath6kl_cfg80211_set_default_key,
2842 .set_wiphy_params = ath6kl_cfg80211_set_wiphy_params,
2843 .set_tx_power = ath6kl_cfg80211_set_txpower,
2844 .get_tx_power = ath6kl_cfg80211_get_txpower,
2845 .set_power_mgmt = ath6kl_cfg80211_set_power_mgmt,
2846 .join_ibss = ath6kl_cfg80211_join_ibss,
2847 .leave_ibss = ath6kl_cfg80211_leave_ibss,
2848 .get_station = ath6kl_get_station,
2849 .set_pmksa = ath6kl_set_pmksa,
2850 .del_pmksa = ath6kl_del_pmksa,
2851 .flush_pmksa = ath6kl_flush_pmksa,
Kalle Valo003353b0d2011-09-01 10:14:21 +03002852 CFG80211_TESTMODE_CMD(ath6kl_tm_cmd)
Kalle Valoabcb3442011-07-22 08:26:20 +03002853#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02002854 .suspend = __ath6kl_cfg80211_suspend,
2855 .resume = __ath6kl_cfg80211_resume,
Kalle Valoabcb3442011-07-22 08:26:20 +03002856#endif
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002857 .set_channel = ath6kl_set_channel,
2858 .add_beacon = ath6kl_add_beacon,
2859 .set_beacon = ath6kl_set_beacon,
2860 .del_beacon = ath6kl_del_beacon,
Jouni Malinen33e53082011-12-27 11:02:56 +02002861 .del_station = ath6kl_del_station,
Jouni Malinen23875132011-08-30 21:57:53 +03002862 .change_station = ath6kl_change_station,
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002863 .remain_on_channel = ath6kl_remain_on_channel,
2864 .cancel_remain_on_channel = ath6kl_cancel_remain_on_channel,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002865 .mgmt_tx = ath6kl_mgmt_tx,
Jouni Malinenae32c302011-08-30 21:58:01 +03002866 .mgmt_frame_register = ath6kl_mgmt_frame_register,
Kalle Valo10509f92011-12-13 14:52:07 +02002867 .sched_scan_start = ath6kl_cfg80211_sscan_start,
2868 .sched_scan_stop = ath6kl_cfg80211_sscan_stop,
Kalle Valobdcd8172011-07-18 00:22:30 +03002869};
2870
Kalle Valo7125f012011-12-13 14:51:37 +02002871void ath6kl_cfg80211_stop(struct ath6kl_vif *vif)
Kalle Valoec4b7f62011-11-01 08:44:04 +02002872{
Kalle Valo10509f92011-12-13 14:52:07 +02002873 ath6kl_cfg80211_sscan_disable(vif);
2874
Kalle Valoec4b7f62011-11-01 08:44:04 +02002875 switch (vif->sme_state) {
Kalle Valoc97a31b2011-12-13 14:51:10 +02002876 case SME_DISCONNECTED:
2877 break;
Kalle Valoec4b7f62011-11-01 08:44:04 +02002878 case SME_CONNECTING:
2879 cfg80211_connect_result(vif->ndev, vif->bssid, NULL, 0,
2880 NULL, 0,
2881 WLAN_STATUS_UNSPECIFIED_FAILURE,
2882 GFP_KERNEL);
2883 break;
2884 case SME_CONNECTED:
Kalle Valoec4b7f62011-11-01 08:44:04 +02002885 cfg80211_disconnected(vif->ndev, 0, NULL, 0, GFP_KERNEL);
2886 break;
2887 }
2888
2889 if (test_bit(CONNECTED, &vif->flags) ||
2890 test_bit(CONNECT_PEND, &vif->flags))
Kalle Valo7125f012011-12-13 14:51:37 +02002891 ath6kl_wmi_disconnect_cmd(vif->ar->wmi, vif->fw_vif_idx);
Kalle Valoec4b7f62011-11-01 08:44:04 +02002892
2893 vif->sme_state = SME_DISCONNECTED;
Kalle Valo1f40525512011-11-01 08:44:13 +02002894 clear_bit(CONNECTED, &vif->flags);
2895 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valoec4b7f62011-11-01 08:44:04 +02002896
2897 /* disable scanning */
Kalle Valo7125f012011-12-13 14:51:37 +02002898 if (ath6kl_wmi_scanparams_cmd(vif->ar->wmi, vif->fw_vif_idx, 0xFFFF,
2899 0, 0, 0, 0, 0, 0, 0, 0, 0) != 0)
2900 ath6kl_warn("failed to disable scan during stop\n");
Kalle Valoec4b7f62011-11-01 08:44:04 +02002901
2902 ath6kl_cfg80211_scan_complete_event(vif, true);
2903}
2904
Kalle Valo7125f012011-12-13 14:51:37 +02002905void ath6kl_cfg80211_stop_all(struct ath6kl *ar)
2906{
2907 struct ath6kl_vif *vif;
2908
2909 vif = ath6kl_vif_first(ar);
2910 if (!vif) {
2911 /* save the current power mode before enabling power save */
2912 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
2913
2914 if (ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER) != 0)
2915 ath6kl_warn("ath6kl_deep_sleep_enable: "
2916 "wmi_powermode_cmd failed\n");
2917 return;
2918 }
2919
2920 /*
2921 * FIXME: we should take ar->list_lock to protect changes in the
2922 * vif_list, but that's not trivial to do as ath6kl_cfg80211_stop()
2923 * sleeps.
2924 */
2925 list_for_each_entry(vif, &ar->vif_list, list)
2926 ath6kl_cfg80211_stop(vif);
2927}
2928
Kalle Valoc25889e2012-01-17 20:08:27 +02002929static int ath6kl_cfg80211_vif_init(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +03002930{
Vasanthakumar Thiagarajan7baef812012-01-21 15:22:50 +05302931 vif->aggr_cntxt = aggr_init(vif);
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302932 if (!vif->aggr_cntxt) {
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302933 ath6kl_err("failed to initialize aggr\n");
2934 return -ENOMEM;
2935 }
Kalle Valobdcd8172011-07-18 00:22:30 +03002936
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302937 setup_timer(&vif->disconnect_timer, disconnect_timer_handler,
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302938 (unsigned long) vif->ndev);
Kalle Valo10509f92011-12-13 14:52:07 +02002939 setup_timer(&vif->sched_scan_timer, ath6kl_wmi_sscan_timer,
2940 (unsigned long) vif);
2941
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302942 set_bit(WMM_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan478ac022011-10-25 19:34:19 +05302943 spin_lock_init(&vif->if_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302944
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05302945 INIT_LIST_HEAD(&vif->mc_filter);
2946
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302947 return 0;
2948}
2949
Kalle Valoc25889e2012-01-17 20:08:27 +02002950void ath6kl_cfg80211_vif_cleanup(struct ath6kl_vif *vif)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302951{
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302952 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05302953 struct ath6kl_mc_filter *mc_filter, *tmp;
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302954
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302955 aggr_module_destroy(vif->aggr_cntxt);
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302956
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302957 ar->avail_idx_map |= BIT(vif->fw_vif_idx);
2958
2959 if (vif->nw_type == ADHOC_NETWORK)
2960 ar->ibss_if_active = false;
2961
Vasanthakumar Thiagarajan80abaf92012-01-03 14:42:01 +05302962 list_for_each_entry_safe(mc_filter, tmp, &vif->mc_filter, list) {
2963 list_del(&mc_filter->list);
2964 kfree(mc_filter);
2965 }
2966
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302967 unregister_netdevice(vif->ndev);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302968
2969 ar->num_vif--;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302970}
2971
2972struct net_device *ath6kl_interface_add(struct ath6kl *ar, char *name,
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302973 enum nl80211_iftype type, u8 fw_vif_idx,
2974 u8 nw_type)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302975{
2976 struct net_device *ndev;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302977 struct ath6kl_vif *vif;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302978
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302979 ndev = alloc_netdev(sizeof(*vif), name, ether_setup);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302980 if (!ndev)
2981 return NULL;
2982
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302983 vif = netdev_priv(ndev);
2984 ndev->ieee80211_ptr = &vif->wdev;
2985 vif->wdev.wiphy = ar->wiphy;
2986 vif->ar = ar;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302987 vif->ndev = ndev;
2988 SET_NETDEV_DEV(ndev, wiphy_dev(vif->wdev.wiphy));
2989 vif->wdev.netdev = ndev;
2990 vif->wdev.iftype = type;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302991 vif->fw_vif_idx = fw_vif_idx;
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302992 vif->nw_type = vif->next_mode = nw_type;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302993
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302994 memcpy(ndev->dev_addr, ar->mac_addr, ETH_ALEN);
2995 if (fw_vif_idx != 0)
2996 ndev->dev_addr[0] = (ndev->dev_addr[0] ^ (1 << fw_vif_idx)) |
2997 0x2;
2998
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302999 init_netdev(ndev);
3000
Vasanthakumar Thiagarajane29f25f2011-10-25 19:34:15 +05303001 ath6kl_init_control_info(vif);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303002
Kalle Valoc25889e2012-01-17 20:08:27 +02003003 if (ath6kl_cfg80211_vif_init(vif))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303004 goto err;
3005
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05303006 if (register_netdevice(ndev))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303007 goto err;
3008
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303009 ar->avail_idx_map &= ~BIT(fw_vif_idx);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05303010 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05303011 set_bit(WLAN_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303012 ar->wlan_pwr_state = WLAN_POWER_STATE_ON;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05303013 set_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303014
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05303015 if (type == NL80211_IFTYPE_ADHOC)
3016 ar->ibss_if_active = true;
3017
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05303018 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05303019 list_add_tail(&vif->list, &ar->vif_list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05303020 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05303021
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303022 return ndev;
3023
3024err:
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05303025 aggr_module_destroy(vif->aggr_cntxt);
3026 free_netdev(ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303027 return NULL;
3028}
3029
Kalle Valo46d33a22012-01-17 20:08:40 +02003030int ath6kl_cfg80211_init(struct ath6kl *ar)
3031{
3032 struct wiphy *wiphy = ar->wiphy;
3033 int ret;
3034
3035 wiphy->mgmt_stypes = ath6kl_mgmt_stypes;
3036
3037 wiphy->max_remain_on_channel_duration = 5000;
3038
3039 /* set device pointer for wiphy */
3040 set_wiphy_dev(wiphy, ar->dev);
3041
3042 wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
3043 BIT(NL80211_IFTYPE_ADHOC) |
3044 BIT(NL80211_IFTYPE_AP);
3045 if (ar->p2p) {
3046 wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_GO) |
3047 BIT(NL80211_IFTYPE_P2P_CLIENT);
3048 }
3049
3050 /* max num of ssids that can be probed during scanning */
3051 wiphy->max_scan_ssids = MAX_PROBED_SSID_INDEX;
3052 wiphy->max_scan_ie_len = 1000; /* FIX: what is correct limit? */
3053 wiphy->bands[IEEE80211_BAND_2GHZ] = &ath6kl_band_2ghz;
3054 wiphy->bands[IEEE80211_BAND_5GHZ] = &ath6kl_band_5ghz;
3055 wiphy->signal_type = CFG80211_SIGNAL_TYPE_MBM;
3056
3057 wiphy->cipher_suites = cipher_suites;
3058 wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
3059
3060 wiphy->wowlan.flags = WIPHY_WOWLAN_MAGIC_PKT |
3061 WIPHY_WOWLAN_DISCONNECT |
3062 WIPHY_WOWLAN_GTK_REKEY_FAILURE |
3063 WIPHY_WOWLAN_SUPPORTS_GTK_REKEY |
3064 WIPHY_WOWLAN_EAP_IDENTITY_REQ |
3065 WIPHY_WOWLAN_4WAY_HANDSHAKE;
3066 wiphy->wowlan.n_patterns = WOW_MAX_FILTERS_PER_LIST;
3067 wiphy->wowlan.pattern_min_len = 1;
3068 wiphy->wowlan.pattern_max_len = WOW_PATTERN_SIZE;
3069
3070 wiphy->max_sched_scan_ssids = 10;
3071
3072 ret = wiphy_register(wiphy);
3073 if (ret < 0) {
3074 ath6kl_err("couldn't register wiphy device\n");
3075 return ret;
3076 }
3077
3078 return 0;
3079}
3080
3081void ath6kl_cfg80211_cleanup(struct ath6kl *ar)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05303082{
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05303083 wiphy_unregister(ar->wiphy);
Kalle Valo45eaa782012-01-17 20:09:05 +02003084}
Kalle Valo46d33a22012-01-17 20:08:40 +02003085
Kalle Valo45eaa782012-01-17 20:09:05 +02003086struct ath6kl *ath6kl_cfg80211_create(void)
3087{
3088 struct ath6kl *ar;
3089 struct wiphy *wiphy;
3090
3091 /* create a new wiphy for use with cfg80211 */
3092 wiphy = wiphy_new(&ath6kl_cfg80211_ops, sizeof(struct ath6kl));
3093
3094 if (!wiphy) {
3095 ath6kl_err("couldn't allocate wiphy device\n");
3096 return NULL;
3097 }
3098
3099 ar = wiphy_priv(wiphy);
3100 ar->wiphy = wiphy;
3101
3102 return ar;
3103}
3104
3105/* Note: ar variable must not be accessed after calling this! */
3106void ath6kl_cfg80211_destroy(struct ath6kl *ar)
3107{
Vasanthakumar Thiagarajan1d2a4452012-01-21 15:22:53 +05303108 int i;
3109
3110 for (i = 0; i < AP_MAX_NUM_STA; i++)
3111 kfree(ar->sta_list[i].aggr_conn);
3112
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05303113 wiphy_free(ar->wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03003114}
Kalle Valo45eaa782012-01-17 20:09:05 +02003115