blob: a5c0a580d3927cf1499a67d41a9fc521a069e69e [file] [log] [blame]
Kalle Valobdcd8172011-07-18 00:22:30 +03001/*
2 * Copyright (c) 2004-2011 Atheros Communications Inc.
3 *
4 * Permission to use, copy, modify, and/or distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
17#include "core.h"
18#include "cfg80211.h"
19#include "debug.h"
Kalle Valoabcb3442011-07-22 08:26:20 +030020#include "hif-ops.h"
Kalle Valo003353b0d2011-09-01 10:14:21 +030021#include "testmode.h"
Kalle Valobdcd8172011-07-18 00:22:30 +030022
Jouni Malinen6bbc7c32011-09-05 17:38:47 +030023static unsigned int ath6kl_p2p;
24
25module_param(ath6kl_p2p, uint, 0644);
26
Kalle Valobdcd8172011-07-18 00:22:30 +030027#define RATETAB_ENT(_rate, _rateid, _flags) { \
28 .bitrate = (_rate), \
29 .flags = (_flags), \
30 .hw_value = (_rateid), \
31}
32
33#define CHAN2G(_channel, _freq, _flags) { \
34 .band = IEEE80211_BAND_2GHZ, \
35 .hw_value = (_channel), \
36 .center_freq = (_freq), \
37 .flags = (_flags), \
38 .max_antenna_gain = 0, \
39 .max_power = 30, \
40}
41
42#define CHAN5G(_channel, _flags) { \
43 .band = IEEE80211_BAND_5GHZ, \
44 .hw_value = (_channel), \
45 .center_freq = 5000 + (5 * (_channel)), \
46 .flags = (_flags), \
47 .max_antenna_gain = 0, \
48 .max_power = 30, \
49}
50
51static struct ieee80211_rate ath6kl_rates[] = {
52 RATETAB_ENT(10, 0x1, 0),
53 RATETAB_ENT(20, 0x2, 0),
54 RATETAB_ENT(55, 0x4, 0),
55 RATETAB_ENT(110, 0x8, 0),
56 RATETAB_ENT(60, 0x10, 0),
57 RATETAB_ENT(90, 0x20, 0),
58 RATETAB_ENT(120, 0x40, 0),
59 RATETAB_ENT(180, 0x80, 0),
60 RATETAB_ENT(240, 0x100, 0),
61 RATETAB_ENT(360, 0x200, 0),
62 RATETAB_ENT(480, 0x400, 0),
63 RATETAB_ENT(540, 0x800, 0),
64};
65
66#define ath6kl_a_rates (ath6kl_rates + 4)
67#define ath6kl_a_rates_size 8
68#define ath6kl_g_rates (ath6kl_rates + 0)
69#define ath6kl_g_rates_size 12
70
71static struct ieee80211_channel ath6kl_2ghz_channels[] = {
72 CHAN2G(1, 2412, 0),
73 CHAN2G(2, 2417, 0),
74 CHAN2G(3, 2422, 0),
75 CHAN2G(4, 2427, 0),
76 CHAN2G(5, 2432, 0),
77 CHAN2G(6, 2437, 0),
78 CHAN2G(7, 2442, 0),
79 CHAN2G(8, 2447, 0),
80 CHAN2G(9, 2452, 0),
81 CHAN2G(10, 2457, 0),
82 CHAN2G(11, 2462, 0),
83 CHAN2G(12, 2467, 0),
84 CHAN2G(13, 2472, 0),
85 CHAN2G(14, 2484, 0),
86};
87
88static struct ieee80211_channel ath6kl_5ghz_a_channels[] = {
89 CHAN5G(34, 0), CHAN5G(36, 0),
90 CHAN5G(38, 0), CHAN5G(40, 0),
91 CHAN5G(42, 0), CHAN5G(44, 0),
92 CHAN5G(46, 0), CHAN5G(48, 0),
93 CHAN5G(52, 0), CHAN5G(56, 0),
94 CHAN5G(60, 0), CHAN5G(64, 0),
95 CHAN5G(100, 0), CHAN5G(104, 0),
96 CHAN5G(108, 0), CHAN5G(112, 0),
97 CHAN5G(116, 0), CHAN5G(120, 0),
98 CHAN5G(124, 0), CHAN5G(128, 0),
99 CHAN5G(132, 0), CHAN5G(136, 0),
100 CHAN5G(140, 0), CHAN5G(149, 0),
101 CHAN5G(153, 0), CHAN5G(157, 0),
102 CHAN5G(161, 0), CHAN5G(165, 0),
103 CHAN5G(184, 0), CHAN5G(188, 0),
104 CHAN5G(192, 0), CHAN5G(196, 0),
105 CHAN5G(200, 0), CHAN5G(204, 0),
106 CHAN5G(208, 0), CHAN5G(212, 0),
107 CHAN5G(216, 0),
108};
109
110static struct ieee80211_supported_band ath6kl_band_2ghz = {
111 .n_channels = ARRAY_SIZE(ath6kl_2ghz_channels),
112 .channels = ath6kl_2ghz_channels,
113 .n_bitrates = ath6kl_g_rates_size,
114 .bitrates = ath6kl_g_rates,
115};
116
117static struct ieee80211_supported_band ath6kl_band_5ghz = {
118 .n_channels = ARRAY_SIZE(ath6kl_5ghz_a_channels),
119 .channels = ath6kl_5ghz_a_channels,
120 .n_bitrates = ath6kl_a_rates_size,
121 .bitrates = ath6kl_a_rates,
122};
123
Jouni Malinen837cb972011-10-11 17:31:57 +0300124#define CCKM_KRK_CIPHER_SUITE 0x004096ff /* use for KRK */
125
Kalle Valobdcd8172011-07-18 00:22:30 +0300126static int ath6kl_set_wpa_version(struct ath6kl *ar,
127 enum nl80211_wpa_versions wpa_version)
128{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530129 /* TODO: Findout vif */
130 struct ath6kl_vif *vif = ar->vif;
131
Kalle Valobdcd8172011-07-18 00:22:30 +0300132 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: %u\n", __func__, wpa_version);
133
134 if (!wpa_version) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530135 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300136 } else if (wpa_version & NL80211_WPA_VERSION_2) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530137 vif->auth_mode = WPA2_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300138 } else if (wpa_version & NL80211_WPA_VERSION_1) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530139 vif->auth_mode = WPA_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300140 } else {
141 ath6kl_err("%s: %u not supported\n", __func__, wpa_version);
142 return -ENOTSUPP;
143 }
144
145 return 0;
146}
147
148static int ath6kl_set_auth_type(struct ath6kl *ar,
149 enum nl80211_auth_type auth_type)
150{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530151 /* TODO: Findout vif */
152 struct ath6kl_vif *vif = ar->vif;
Kalle Valobdcd8172011-07-18 00:22:30 +0300153
154 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, auth_type);
155
156 switch (auth_type) {
157 case NL80211_AUTHTYPE_OPEN_SYSTEM:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530158 vif->dot11_auth_mode = OPEN_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300159 break;
160 case NL80211_AUTHTYPE_SHARED_KEY:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530161 vif->dot11_auth_mode = SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300162 break;
163 case NL80211_AUTHTYPE_NETWORK_EAP:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530164 vif->dot11_auth_mode = LEAP_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300165 break;
166
167 case NL80211_AUTHTYPE_AUTOMATIC:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530168 vif->dot11_auth_mode = OPEN_AUTH | SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300169 break;
170
171 default:
172 ath6kl_err("%s: 0x%x not spported\n", __func__, auth_type);
173 return -ENOTSUPP;
174 }
175
176 return 0;
177}
178
179static int ath6kl_set_cipher(struct ath6kl *ar, u32 cipher, bool ucast)
180{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530181 /* TODO: Findout vif */
182 struct ath6kl_vif *vif = ar->vif;
183
184 u8 *ar_cipher = ucast ? &vif->prwise_crypto : &vif->grp_crypto;
185 u8 *ar_cipher_len = ucast ? &vif->prwise_crypto_len :
186 &vif->grp_crypto_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300187
188 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: cipher 0x%x, ucast %u\n",
189 __func__, cipher, ucast);
190
191 switch (cipher) {
192 case 0:
193 /* our own hack to use value 0 as no crypto used */
194 *ar_cipher = NONE_CRYPT;
195 *ar_cipher_len = 0;
196 break;
197 case WLAN_CIPHER_SUITE_WEP40:
198 *ar_cipher = WEP_CRYPT;
199 *ar_cipher_len = 5;
200 break;
201 case WLAN_CIPHER_SUITE_WEP104:
202 *ar_cipher = WEP_CRYPT;
203 *ar_cipher_len = 13;
204 break;
205 case WLAN_CIPHER_SUITE_TKIP:
206 *ar_cipher = TKIP_CRYPT;
207 *ar_cipher_len = 0;
208 break;
209 case WLAN_CIPHER_SUITE_CCMP:
210 *ar_cipher = AES_CRYPT;
211 *ar_cipher_len = 0;
212 break;
213 default:
214 ath6kl_err("cipher 0x%x not supported\n", cipher);
215 return -ENOTSUPP;
216 }
217
218 return 0;
219}
220
221static void ath6kl_set_key_mgmt(struct ath6kl *ar, u32 key_mgmt)
222{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530223 /* TODO: Findout vif */
224 struct ath6kl_vif *vif = ar->vif;
225
Kalle Valobdcd8172011-07-18 00:22:30 +0300226 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, key_mgmt);
227
228 if (key_mgmt == WLAN_AKM_SUITE_PSK) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530229 if (vif->auth_mode == WPA_AUTH)
230 vif->auth_mode = WPA_PSK_AUTH;
231 else if (vif->auth_mode == WPA2_AUTH)
232 vif->auth_mode = WPA2_PSK_AUTH;
Jouni Malinen837cb972011-10-11 17:31:57 +0300233 } else if (key_mgmt == 0x00409600) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530234 if (vif->auth_mode == WPA_AUTH)
235 vif->auth_mode = WPA_AUTH_CCKM;
236 else if (vif->auth_mode == WPA2_AUTH)
237 vif->auth_mode = WPA2_AUTH_CCKM;
Kalle Valobdcd8172011-07-18 00:22:30 +0300238 } else if (key_mgmt != WLAN_AKM_SUITE_8021X) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530239 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300240 }
241}
242
243static bool ath6kl_cfg80211_ready(struct ath6kl *ar)
244{
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530245 struct ath6kl_vif *vif = ar->vif;
246
Kalle Valobdcd8172011-07-18 00:22:30 +0300247 if (!test_bit(WMI_READY, &ar->flag)) {
248 ath6kl_err("wmi is not ready\n");
249 return false;
250 }
251
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530252 if (!test_bit(WLAN_ENABLED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300253 ath6kl_err("wlan disabled\n");
254 return false;
255 }
256
257 return true;
258}
259
Kevin Fang6981ffd2011-10-07 08:51:19 +0800260static bool ath6kl_is_wpa_ie(const u8 *pos)
261{
262 return pos[0] == WLAN_EID_WPA && pos[1] >= 4 &&
263 pos[2] == 0x00 && pos[3] == 0x50 &&
264 pos[4] == 0xf2 && pos[5] == 0x01;
265}
266
267static bool ath6kl_is_rsn_ie(const u8 *pos)
268{
269 return pos[0] == WLAN_EID_RSN;
270}
271
272static int ath6kl_set_assoc_req_ies(struct ath6kl *ar, const u8 *ies,
273 size_t ies_len)
274{
275 const u8 *pos;
276 u8 *buf = NULL;
277 size_t len = 0;
278 int ret;
279
280 /*
281 * Filter out RSN/WPA IE(s)
282 */
283
284 if (ies && ies_len) {
285 buf = kmalloc(ies_len, GFP_KERNEL);
286 if (buf == NULL)
287 return -ENOMEM;
288 pos = ies;
289
290 while (pos + 1 < ies + ies_len) {
291 if (pos + 2 + pos[1] > ies + ies_len)
292 break;
293 if (!(ath6kl_is_wpa_ie(pos) || ath6kl_is_rsn_ie(pos))) {
294 memcpy(buf + len, pos, 2 + pos[1]);
295 len += 2 + pos[1];
296 }
297 pos += 2 + pos[1];
298 }
299 }
300
301 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, WMI_FRAME_ASSOC_REQ,
302 buf, len);
303 kfree(buf);
304 return ret;
305}
306
Kalle Valobdcd8172011-07-18 00:22:30 +0300307static int ath6kl_cfg80211_connect(struct wiphy *wiphy, struct net_device *dev,
308 struct cfg80211_connect_params *sme)
309{
310 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530311 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300312 int status;
313
314 ar->sme_state = SME_CONNECTING;
315
316 if (!ath6kl_cfg80211_ready(ar))
317 return -EIO;
318
319 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
320 ath6kl_err("destroy in progress\n");
321 return -EBUSY;
322 }
323
324 if (test_bit(SKIP_SCAN, &ar->flag) &&
325 ((sme->channel && sme->channel->center_freq == 0) ||
326 (sme->bssid && is_zero_ether_addr(sme->bssid)))) {
327 ath6kl_err("SkipScan: channel or bssid invalid\n");
328 return -EINVAL;
329 }
330
331 if (down_interruptible(&ar->sem)) {
332 ath6kl_err("busy, couldn't get access\n");
333 return -ERESTARTSYS;
334 }
335
336 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
337 ath6kl_err("busy, destroy in progress\n");
338 up(&ar->sem);
339 return -EBUSY;
340 }
341
342 if (ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)]) {
343 /*
344 * sleep until the command queue drains
345 */
346 wait_event_interruptible_timeout(ar->event_wq,
347 ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)] == 0,
348 WMI_TIMEOUT);
349 if (signal_pending(current)) {
350 ath6kl_err("cmd queue drain timeout\n");
351 up(&ar->sem);
352 return -EINTR;
353 }
354 }
355
Kevin Fang6981ffd2011-10-07 08:51:19 +0800356 if (sme->ie && (sme->ie_len > 0)) {
357 status = ath6kl_set_assoc_req_ies(ar, sme->ie, sme->ie_len);
358 if (status)
359 return status;
360 }
361
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530362 if (test_bit(CONNECTED, &vif->flags) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530363 vif->ssid_len == sme->ssid_len &&
364 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300365 ar->reconnect_flag = true;
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530366 status = ath6kl_wmi_reconnect_cmd(ar->wmi, vif->req_bssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530367 vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300368
369 up(&ar->sem);
370 if (status) {
371 ath6kl_err("wmi_reconnect_cmd failed\n");
372 return -EIO;
373 }
374 return 0;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530375 } else if (vif->ssid_len == sme->ssid_len &&
376 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300377 ath6kl_disconnect(ar);
378 }
379
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530380 memset(vif->ssid, 0, sizeof(vif->ssid));
381 vif->ssid_len = sme->ssid_len;
382 memcpy(vif->ssid, sme->ssid, sme->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +0300383
384 if (sme->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530385 vif->ch_hint = sme->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +0300386
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530387 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300388 if (sme->bssid && !is_broadcast_ether_addr(sme->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530389 memcpy(vif->req_bssid, sme->bssid, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300390
391 ath6kl_set_wpa_version(ar, sme->crypto.wpa_versions);
392
393 status = ath6kl_set_auth_type(ar, sme->auth_type);
394 if (status) {
395 up(&ar->sem);
396 return status;
397 }
398
399 if (sme->crypto.n_ciphers_pairwise)
400 ath6kl_set_cipher(ar, sme->crypto.ciphers_pairwise[0], true);
401 else
402 ath6kl_set_cipher(ar, 0, true);
403
404 ath6kl_set_cipher(ar, sme->crypto.cipher_group, false);
405
406 if (sme->crypto.n_akm_suites)
407 ath6kl_set_key_mgmt(ar, sme->crypto.akm_suites[0]);
408
409 if ((sme->key_len) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530410 (vif->auth_mode == NONE_AUTH) &&
411 (vif->prwise_crypto == WEP_CRYPT)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300412 struct ath6kl_key *key = NULL;
413
414 if (sme->key_idx < WMI_MIN_KEY_INDEX ||
415 sme->key_idx > WMI_MAX_KEY_INDEX) {
416 ath6kl_err("key index %d out of bounds\n",
417 sme->key_idx);
418 up(&ar->sem);
419 return -ENOENT;
420 }
421
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530422 key = &vif->keys[sme->key_idx];
Kalle Valobdcd8172011-07-18 00:22:30 +0300423 key->key_len = sme->key_len;
424 memcpy(key->key, sme->key, key->key_len);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530425 key->cipher = vif->prwise_crypto;
426 vif->def_txkey_index = sme->key_idx;
Kalle Valobdcd8172011-07-18 00:22:30 +0300427
428 ath6kl_wmi_addkey_cmd(ar->wmi, sme->key_idx,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530429 vif->prwise_crypto,
Kalle Valobdcd8172011-07-18 00:22:30 +0300430 GROUP_USAGE | TX_USAGE,
431 key->key_len,
432 NULL,
433 key->key, KEY_OP_INIT_VAL, NULL,
434 NO_SYNC_WMIFLAG);
435 }
436
437 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530438 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300439 if (ath6kl_wmi_bssfilter_cmd(ar->wmi, ALL_BSS_FILTER, 0) != 0) {
440 ath6kl_err("couldn't set bss filtering\n");
441 up(&ar->sem);
442 return -EIO;
443 }
444 }
445
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530446 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +0300447
448 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
449 "%s: connect called with authmode %d dot11 auth %d"
450 " PW crypto %d PW crypto len %d GRP crypto %d"
451 " GRP crypto len %d channel hint %u\n",
452 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530453 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
454 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530455 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300456
457 ar->reconnect_flag = 0;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530458 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530459 vif->dot11_auth_mode, vif->auth_mode,
460 vif->prwise_crypto,
461 vif->prwise_crypto_len,
462 vif->grp_crypto, vif->grp_crypto_len,
463 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530464 vif->req_bssid, vif->ch_hint,
Kalle Valobdcd8172011-07-18 00:22:30 +0300465 ar->connect_ctrl_flags);
466
467 up(&ar->sem);
468
469 if (status == -EINVAL) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530470 memset(vif->ssid, 0, sizeof(vif->ssid));
471 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300472 ath6kl_err("invalid request\n");
473 return -ENOENT;
474 } else if (status) {
475 ath6kl_err("ath6kl_wmi_connect_cmd failed\n");
476 return -EIO;
477 }
478
479 if ((!(ar->connect_ctrl_flags & CONNECT_DO_WPA_OFFLOAD)) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530480 ((vif->auth_mode == WPA_PSK_AUTH)
481 || (vif->auth_mode == WPA2_PSK_AUTH))) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300482 mod_timer(&ar->disconnect_timer,
483 jiffies + msecs_to_jiffies(DISCON_TIMER_INTVAL));
484 }
485
486 ar->connect_ctrl_flags &= ~CONNECT_DO_WPA_OFFLOAD;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530487 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300488
489 return 0;
490}
491
Jouni Malinen01cac472011-09-19 19:14:59 +0300492static int ath6kl_add_bss_if_needed(struct ath6kl *ar, const u8 *bssid,
493 struct ieee80211_channel *chan,
494 const u8 *beacon_ie, size_t beacon_ie_len)
495{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530496 /* TODO: Findout vif */
497 struct ath6kl_vif *vif = ar->vif;
Jouni Malinen01cac472011-09-19 19:14:59 +0300498 struct cfg80211_bss *bss;
499 u8 *ie;
500
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530501 bss = cfg80211_get_bss(ar->wiphy, chan, bssid,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530502 vif->ssid, vif->ssid_len, WLAN_CAPABILITY_ESS,
Jouni Malinen01cac472011-09-19 19:14:59 +0300503 WLAN_CAPABILITY_ESS);
504 if (bss == NULL) {
505 /*
506 * Since cfg80211 may not yet know about the BSS,
507 * generate a partial entry until the first BSS info
508 * event becomes available.
509 *
510 * Prepend SSID element since it is not included in the Beacon
511 * IEs from the target.
512 */
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530513 ie = kmalloc(2 + vif->ssid_len + beacon_ie_len, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300514 if (ie == NULL)
515 return -ENOMEM;
516 ie[0] = WLAN_EID_SSID;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530517 ie[1] = vif->ssid_len;
518 memcpy(ie + 2, vif->ssid, vif->ssid_len);
519 memcpy(ie + 2 + vif->ssid_len, beacon_ie, beacon_ie_len);
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530520 bss = cfg80211_inform_bss(ar->wiphy, chan,
Jouni Malinen01cac472011-09-19 19:14:59 +0300521 bssid, 0, WLAN_CAPABILITY_ESS, 100,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530522 ie, 2 + vif->ssid_len + beacon_ie_len,
Jouni Malinen01cac472011-09-19 19:14:59 +0300523 0, GFP_KERNEL);
524 if (bss)
525 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "added dummy bss for "
526 "%pM prior to indicating connect/roamed "
527 "event\n", bssid);
528 kfree(ie);
529 } else
530 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "cfg80211 already has a bss "
531 "entry\n");
532
533 if (bss == NULL)
534 return -ENOMEM;
535
536 cfg80211_put_bss(bss);
537
538 return 0;
539}
540
Kalle Valobdcd8172011-07-18 00:22:30 +0300541void ath6kl_cfg80211_connect_event(struct ath6kl *ar, u16 channel,
542 u8 *bssid, u16 listen_intvl,
543 u16 beacon_intvl,
544 enum network_type nw_type,
545 u8 beacon_ie_len, u8 assoc_req_len,
546 u8 assoc_resp_len, u8 *assoc_info)
547{
Jouni Malinen01cac472011-09-19 19:14:59 +0300548 struct ieee80211_channel *chan;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530549 /* TODO: Findout vif */
550 struct ath6kl_vif *vif = ar->vif;
Kalle Valobdcd8172011-07-18 00:22:30 +0300551
552 /* capinfo + listen interval */
553 u8 assoc_req_ie_offset = sizeof(u16) + sizeof(u16);
554
555 /* capinfo + status code + associd */
556 u8 assoc_resp_ie_offset = sizeof(u16) + sizeof(u16) + sizeof(u16);
557
558 u8 *assoc_req_ie = assoc_info + beacon_ie_len + assoc_req_ie_offset;
559 u8 *assoc_resp_ie = assoc_info + beacon_ie_len + assoc_req_len +
560 assoc_resp_ie_offset;
561
562 assoc_req_len -= assoc_req_ie_offset;
563 assoc_resp_len -= assoc_resp_ie_offset;
564
Jouni Malinen32c10872011-09-19 19:15:07 +0300565 /*
566 * Store Beacon interval here; DTIM period will be available only once
567 * a Beacon frame from the AP is seen.
568 */
569 ar->assoc_bss_beacon_int = beacon_intvl;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530570 clear_bit(DTIM_PERIOD_AVAIL, &vif->flags);
Jouni Malinen32c10872011-09-19 19:15:07 +0300571
Kalle Valobdcd8172011-07-18 00:22:30 +0300572 if (nw_type & ADHOC_NETWORK) {
573 if (ar->wdev->iftype != NL80211_IFTYPE_ADHOC) {
574 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
575 "%s: ath6k not in ibss mode\n", __func__);
576 return;
577 }
578 }
579
580 if (nw_type & INFRA_NETWORK) {
Jouni Malinen6b5e5d22011-08-30 21:58:05 +0300581 if (ar->wdev->iftype != NL80211_IFTYPE_STATION &&
582 ar->wdev->iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300583 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
584 "%s: ath6k not in station mode\n", __func__);
585 return;
586 }
587 }
588
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530589 chan = ieee80211_get_channel(ar->wiphy, (int) channel);
Kalle Valobdcd8172011-07-18 00:22:30 +0300590
Kalle Valobdcd8172011-07-18 00:22:30 +0300591
592 if (nw_type & ADHOC_NETWORK) {
593 cfg80211_ibss_joined(ar->net_dev, bssid, GFP_KERNEL);
594 return;
595 }
596
Jouni Malinen01cac472011-09-19 19:14:59 +0300597 if (ath6kl_add_bss_if_needed(ar, bssid, chan, assoc_info,
598 beacon_ie_len) < 0) {
599 ath6kl_err("could not add cfg80211 bss entry for "
600 "connect/roamed notification\n");
601 return;
602 }
603
Raja Mani9aa60352011-08-04 19:26:29 +0530604 if (ar->sme_state == SME_CONNECTING) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300605 /* inform connect result to cfg80211 */
Raja Mani9aa60352011-08-04 19:26:29 +0530606 ar->sme_state = SME_CONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300607 cfg80211_connect_result(ar->net_dev, bssid,
608 assoc_req_ie, assoc_req_len,
609 assoc_resp_ie, assoc_resp_len,
610 WLAN_STATUS_SUCCESS, GFP_KERNEL);
Raja Mani9aa60352011-08-04 19:26:29 +0530611 } else if (ar->sme_state == SME_CONNECTED) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300612 /* inform roam event to cfg80211 */
Jouni Malinen01cac472011-09-19 19:14:59 +0300613 cfg80211_roamed(ar->net_dev, chan, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300614 assoc_req_ie, assoc_req_len,
615 assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
616 }
617}
618
619static int ath6kl_cfg80211_disconnect(struct wiphy *wiphy,
620 struct net_device *dev, u16 reason_code)
621{
622 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530623 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300624
625 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: reason=%u\n", __func__,
626 reason_code);
627
628 if (!ath6kl_cfg80211_ready(ar))
629 return -EIO;
630
631 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
632 ath6kl_err("busy, destroy in progress\n");
633 return -EBUSY;
634 }
635
636 if (down_interruptible(&ar->sem)) {
637 ath6kl_err("busy, couldn't get access\n");
638 return -ERESTARTSYS;
639 }
640
641 ar->reconnect_flag = 0;
642 ath6kl_disconnect(ar);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530643 memset(vif->ssid, 0, sizeof(vif->ssid));
644 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300645
646 if (!test_bit(SKIP_SCAN, &ar->flag))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530647 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300648
649 up(&ar->sem);
650
Vasanthakumar Thiagarajan170826d2011-09-10 15:26:35 +0530651 ar->sme_state = SME_DISCONNECTED;
652
Kalle Valobdcd8172011-07-18 00:22:30 +0300653 return 0;
654}
655
656void ath6kl_cfg80211_disconnect_event(struct ath6kl *ar, u8 reason,
657 u8 *bssid, u8 assoc_resp_len,
658 u8 *assoc_info, u16 proto_reason)
659{
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530660 /* TODO: Findout vif */
661 struct ath6kl_vif *vif = ar->vif;
662
Kalle Valobdcd8172011-07-18 00:22:30 +0300663 if (ar->scan_req) {
664 cfg80211_scan_done(ar->scan_req, true);
665 ar->scan_req = NULL;
666 }
667
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530668 if (vif->nw_type & ADHOC_NETWORK) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300669 if (ar->wdev->iftype != NL80211_IFTYPE_ADHOC) {
670 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
671 "%s: ath6k not in ibss mode\n", __func__);
672 return;
673 }
674 memset(bssid, 0, ETH_ALEN);
675 cfg80211_ibss_joined(ar->net_dev, bssid, GFP_KERNEL);
676 return;
677 }
678
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530679 if (vif->nw_type & INFRA_NETWORK) {
Jouni Malinen6b5e5d22011-08-30 21:58:05 +0300680 if (ar->wdev->iftype != NL80211_IFTYPE_STATION &&
681 ar->wdev->iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300682 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
683 "%s: ath6k not in station mode\n", __func__);
684 return;
685 }
686 }
687
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530688 /*
689 * Send a disconnect command to target when a disconnect event is
690 * received with reason code other than 3 (DISCONNECT_CMD - disconnect
691 * request from host) to make the firmware stop trying to connect even
692 * after giving disconnect event. There will be one more disconnect
693 * event for this disconnect command with reason code DISCONNECT_CMD
694 * which will be notified to cfg80211.
695 */
Kalle Valobdcd8172011-07-18 00:22:30 +0300696
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530697 if (reason != DISCONNECT_CMD) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300698 ath6kl_wmi_disconnect_cmd(ar->wmi);
699 return;
700 }
701
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530702 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300703
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530704 if (ar->sme_state == SME_CONNECTING) {
705 cfg80211_connect_result(ar->net_dev,
706 bssid, NULL, 0,
707 NULL, 0,
708 WLAN_STATUS_UNSPECIFIED_FAILURE,
709 GFP_KERNEL);
710 } else if (ar->sme_state == SME_CONNECTED) {
711 cfg80211_disconnected(ar->net_dev, reason,
712 NULL, 0, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300713 }
714
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530715 ar->sme_state = SME_DISCONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300716}
717
Kalle Valobdcd8172011-07-18 00:22:30 +0300718static int ath6kl_cfg80211_scan(struct wiphy *wiphy, struct net_device *ndev,
719 struct cfg80211_scan_request *request)
720{
721 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530722 struct ath6kl_vif *vif = netdev_priv(ndev);
Edward Lu1276c9e2011-08-30 21:58:00 +0300723 s8 n_channels = 0;
724 u16 *channels = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300725 int ret = 0;
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530726 u32 force_fg_scan = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300727
728 if (!ath6kl_cfg80211_ready(ar))
729 return -EIO;
730
731 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530732 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300733 ret = ath6kl_wmi_bssfilter_cmd(
734 ar->wmi,
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530735 (test_bit(CONNECTED, &vif->flags) ?
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300736 ALL_BUT_BSS_FILTER : ALL_BSS_FILTER), 0);
737 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300738 ath6kl_err("couldn't set bss filtering\n");
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300739 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +0300740 }
741 }
742
743 if (request->n_ssids && request->ssids[0].ssid_len) {
744 u8 i;
745
746 if (request->n_ssids > (MAX_PROBED_SSID_INDEX - 1))
747 request->n_ssids = MAX_PROBED_SSID_INDEX - 1;
748
749 for (i = 0; i < request->n_ssids; i++)
750 ath6kl_wmi_probedssid_cmd(ar->wmi, i + 1,
751 SPECIFIC_SSID_FLAG,
752 request->ssids[i].ssid_len,
753 request->ssids[i].ssid);
754 }
755
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300756 if (request->ie) {
757 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, WMI_FRAME_PROBE_REQ,
758 request->ie, request->ie_len);
759 if (ret) {
760 ath6kl_err("failed to set Probe Request appie for "
761 "scan");
762 return ret;
763 }
764 }
765
Jouni Malinen11869be2011-09-02 20:07:06 +0300766 /*
767 * Scan only the requested channels if the request specifies a set of
768 * channels. If the list is longer than the target supports, do not
769 * configure the list and instead, scan all available channels.
770 */
771 if (request->n_channels > 0 &&
772 request->n_channels <= WMI_MAX_CHANNELS) {
Edward Lu1276c9e2011-08-30 21:58:00 +0300773 u8 i;
774
Jouni Malinen11869be2011-09-02 20:07:06 +0300775 n_channels = request->n_channels;
Edward Lu1276c9e2011-08-30 21:58:00 +0300776
777 channels = kzalloc(n_channels * sizeof(u16), GFP_KERNEL);
778 if (channels == NULL) {
779 ath6kl_warn("failed to set scan channels, "
780 "scan all channels");
781 n_channels = 0;
782 }
783
784 for (i = 0; i < n_channels; i++)
785 channels[i] = request->channels[i]->center_freq;
786 }
787
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530788 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530789 force_fg_scan = 1;
790
791 ret = ath6kl_wmi_startscan_cmd(ar->wmi, WMI_LONG_SCAN, force_fg_scan,
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300792 false, 0, 0, n_channels, channels);
793 if (ret)
Kalle Valobdcd8172011-07-18 00:22:30 +0300794 ath6kl_err("wmi_startscan_cmd failed\n");
Jouni Malinen11869be2011-09-02 20:07:06 +0300795 else
796 ar->scan_req = request;
Kalle Valobdcd8172011-07-18 00:22:30 +0300797
Edward Lu1276c9e2011-08-30 21:58:00 +0300798 kfree(channels);
799
Kalle Valobdcd8172011-07-18 00:22:30 +0300800 return ret;
801}
802
803void ath6kl_cfg80211_scan_complete_event(struct ath6kl *ar, int status)
804{
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300805 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +0300806
807 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: status %d\n", __func__, status);
808
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300809 if (!ar->scan_req)
810 return;
Kalle Valobdcd8172011-07-18 00:22:30 +0300811
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300812 if ((status == -ECANCELED) || (status == -EBUSY)) {
813 cfg80211_scan_done(ar->scan_req, true);
814 goto out;
Kalle Valobdcd8172011-07-18 00:22:30 +0300815 }
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300816
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300817 cfg80211_scan_done(ar->scan_req, false);
818
819 if (ar->scan_req->n_ssids && ar->scan_req->ssids[0].ssid_len) {
820 for (i = 0; i < ar->scan_req->n_ssids; i++) {
821 ath6kl_wmi_probedssid_cmd(ar->wmi, i + 1,
822 DISABLE_SSID_FLAG,
823 0, NULL);
824 }
825 }
826
827out:
828 ar->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300829}
830
831static int ath6kl_cfg80211_add_key(struct wiphy *wiphy, struct net_device *ndev,
832 u8 key_index, bool pairwise,
833 const u8 *mac_addr,
834 struct key_params *params)
835{
836 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530837 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300838 struct ath6kl_key *key = NULL;
839 u8 key_usage;
840 u8 key_type;
841 int status = 0;
842
843 if (!ath6kl_cfg80211_ready(ar))
844 return -EIO;
845
Jouni Malinen837cb972011-10-11 17:31:57 +0300846 if (params->cipher == CCKM_KRK_CIPHER_SUITE) {
847 if (params->key_len != WMI_KRK_LEN)
848 return -EINVAL;
849 return ath6kl_wmi_add_krk_cmd(ar->wmi, params->key);
850 }
851
Kalle Valobdcd8172011-07-18 00:22:30 +0300852 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
853 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
854 "%s: key index %d out of bounds\n", __func__,
855 key_index);
856 return -ENOENT;
857 }
858
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530859 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +0300860 memset(key, 0, sizeof(struct ath6kl_key));
861
862 if (pairwise)
863 key_usage = PAIRWISE_USAGE;
864 else
865 key_usage = GROUP_USAGE;
866
867 if (params) {
868 if (params->key_len > WLAN_MAX_KEY_LEN ||
869 params->seq_len > sizeof(key->seq))
870 return -EINVAL;
871
872 key->key_len = params->key_len;
873 memcpy(key->key, params->key, key->key_len);
874 key->seq_len = params->seq_len;
875 memcpy(key->seq, params->seq, key->seq_len);
876 key->cipher = params->cipher;
877 }
878
879 switch (key->cipher) {
880 case WLAN_CIPHER_SUITE_WEP40:
881 case WLAN_CIPHER_SUITE_WEP104:
882 key_type = WEP_CRYPT;
883 break;
884
885 case WLAN_CIPHER_SUITE_TKIP:
886 key_type = TKIP_CRYPT;
887 break;
888
889 case WLAN_CIPHER_SUITE_CCMP:
890 key_type = AES_CRYPT;
891 break;
892
893 default:
894 return -ENOTSUPP;
895 }
896
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530897 if (((vif->auth_mode == WPA_PSK_AUTH)
898 || (vif->auth_mode == WPA2_PSK_AUTH))
Kalle Valobdcd8172011-07-18 00:22:30 +0300899 && (key_usage & GROUP_USAGE))
900 del_timer(&ar->disconnect_timer);
901
902 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
903 "%s: index %d, key_len %d, key_type 0x%x, key_usage 0x%x, seq_len %d\n",
904 __func__, key_index, key->key_len, key_type,
905 key_usage, key->seq_len);
906
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530907 vif->def_txkey_index = key_index;
Jouni Malinen9a5b1312011-08-30 21:57:52 +0300908
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530909 if (vif->nw_type == AP_NETWORK && !pairwise &&
Jouni Malinen9a5b1312011-08-30 21:57:52 +0300910 (key_type == TKIP_CRYPT || key_type == AES_CRYPT) && params) {
911 ar->ap_mode_bkey.valid = true;
912 ar->ap_mode_bkey.key_index = key_index;
913 ar->ap_mode_bkey.key_type = key_type;
914 ar->ap_mode_bkey.key_len = key->key_len;
915 memcpy(ar->ap_mode_bkey.key, key->key, key->key_len);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530916 if (!test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +0300917 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay initial group "
918 "key configuration until AP mode has been "
919 "started\n");
920 /*
921 * The key will be set in ath6kl_connect_ap_mode() once
922 * the connected event is received from the target.
923 */
924 return 0;
925 }
926 }
927
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530928 if (vif->next_mode == AP_NETWORK && key_type == WEP_CRYPT &&
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530929 !test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen151411e2011-09-15 15:10:16 +0300930 /*
931 * Store the key locally so that it can be re-configured after
932 * the AP mode has properly started
933 * (ath6kl_install_statioc_wep_keys).
934 */
935 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay WEP key configuration "
936 "until AP mode has been started\n");
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530937 vif->wep_key_list[key_index].key_len = key->key_len;
938 memcpy(vif->wep_key_list[key_index].key, key->key,
939 key->key_len);
Jouni Malinen151411e2011-09-15 15:10:16 +0300940 return 0;
941 }
942
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530943 status = ath6kl_wmi_addkey_cmd(ar->wmi, vif->def_txkey_index,
Kalle Valobdcd8172011-07-18 00:22:30 +0300944 key_type, key_usage, key->key_len,
945 key->seq, key->key, KEY_OP_INIT_VAL,
946 (u8 *) mac_addr, SYNC_BOTH_WMIFLAG);
947
948 if (status)
949 return -EIO;
950
951 return 0;
952}
953
954static int ath6kl_cfg80211_del_key(struct wiphy *wiphy, struct net_device *ndev,
955 u8 key_index, bool pairwise,
956 const u8 *mac_addr)
957{
958 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530959 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300960
961 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
962
963 if (!ath6kl_cfg80211_ready(ar))
964 return -EIO;
965
966 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
967 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
968 "%s: key index %d out of bounds\n", __func__,
969 key_index);
970 return -ENOENT;
971 }
972
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530973 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300974 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
975 "%s: index %d is empty\n", __func__, key_index);
976 return 0;
977 }
978
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530979 vif->keys[key_index].key_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300980
981 return ath6kl_wmi_deletekey_cmd(ar->wmi, key_index);
982}
983
984static int ath6kl_cfg80211_get_key(struct wiphy *wiphy, struct net_device *ndev,
985 u8 key_index, bool pairwise,
986 const u8 *mac_addr, void *cookie,
987 void (*callback) (void *cookie,
988 struct key_params *))
989{
990 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530991 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300992 struct ath6kl_key *key = NULL;
993 struct key_params params;
994
995 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
996
997 if (!ath6kl_cfg80211_ready(ar))
998 return -EIO;
999
1000 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1001 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1002 "%s: key index %d out of bounds\n", __func__,
1003 key_index);
1004 return -ENOENT;
1005 }
1006
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301007 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001008 memset(&params, 0, sizeof(params));
1009 params.cipher = key->cipher;
1010 params.key_len = key->key_len;
1011 params.seq_len = key->seq_len;
1012 params.seq = key->seq;
1013 params.key = key->key;
1014
1015 callback(cookie, &params);
1016
1017 return key->key_len ? 0 : -ENOENT;
1018}
1019
1020static int ath6kl_cfg80211_set_default_key(struct wiphy *wiphy,
1021 struct net_device *ndev,
1022 u8 key_index, bool unicast,
1023 bool multicast)
1024{
1025 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301026 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001027 struct ath6kl_key *key = NULL;
1028 int status = 0;
1029 u8 key_usage;
Edward Lu229ed6b2011-08-30 21:58:07 +03001030 enum crypto_type key_type = NONE_CRYPT;
Kalle Valobdcd8172011-07-18 00:22:30 +03001031
1032 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1033
1034 if (!ath6kl_cfg80211_ready(ar))
1035 return -EIO;
1036
1037 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1038 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1039 "%s: key index %d out of bounds\n",
1040 __func__, key_index);
1041 return -ENOENT;
1042 }
1043
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301044 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001045 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: invalid key index %d\n",
1046 __func__, key_index);
1047 return -EINVAL;
1048 }
1049
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301050 vif->def_txkey_index = key_index;
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301051 key = &vif->keys[vif->def_txkey_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001052 key_usage = GROUP_USAGE;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301053 if (vif->prwise_crypto == WEP_CRYPT)
Kalle Valobdcd8172011-07-18 00:22:30 +03001054 key_usage |= TX_USAGE;
Edward Lu229ed6b2011-08-30 21:58:07 +03001055 if (unicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301056 key_type = vif->prwise_crypto;
Edward Lu229ed6b2011-08-30 21:58:07 +03001057 if (multicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301058 key_type = vif->grp_crypto;
Kalle Valobdcd8172011-07-18 00:22:30 +03001059
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301060 if (vif->next_mode == AP_NETWORK && !test_bit(CONNECTED, &vif->flags))
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001061 return 0; /* Delay until AP mode has been started */
1062
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301063 status = ath6kl_wmi_addkey_cmd(ar->wmi, vif->def_txkey_index,
Edward Lu229ed6b2011-08-30 21:58:07 +03001064 key_type, key_usage,
Kalle Valobdcd8172011-07-18 00:22:30 +03001065 key->key_len, key->seq, key->key,
1066 KEY_OP_INIT_VAL, NULL,
1067 SYNC_BOTH_WMIFLAG);
1068 if (status)
1069 return -EIO;
1070
1071 return 0;
1072}
1073
1074void ath6kl_cfg80211_tkip_micerr_event(struct ath6kl *ar, u8 keyid,
1075 bool ismcast)
1076{
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301077 /* TODO: Findout vif */
1078 struct ath6kl_vif *vif = ar->vif;
1079
Kalle Valobdcd8172011-07-18 00:22:30 +03001080 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1081 "%s: keyid %d, ismcast %d\n", __func__, keyid, ismcast);
1082
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301083 cfg80211_michael_mic_failure(ar->net_dev, vif->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001084 (ismcast ? NL80211_KEYTYPE_GROUP :
1085 NL80211_KEYTYPE_PAIRWISE), keyid, NULL,
1086 GFP_KERNEL);
1087}
1088
1089static int ath6kl_cfg80211_set_wiphy_params(struct wiphy *wiphy, u32 changed)
1090{
1091 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
1092 int ret;
1093
1094 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: changed 0x%x\n", __func__,
1095 changed);
1096
1097 if (!ath6kl_cfg80211_ready(ar))
1098 return -EIO;
1099
1100 if (changed & WIPHY_PARAM_RTS_THRESHOLD) {
1101 ret = ath6kl_wmi_set_rts_cmd(ar->wmi, wiphy->rts_threshold);
1102 if (ret != 0) {
1103 ath6kl_err("ath6kl_wmi_set_rts_cmd failed\n");
1104 return -EIO;
1105 }
1106 }
1107
1108 return 0;
1109}
1110
1111/*
1112 * The type nl80211_tx_power_setting replaces the following
1113 * data type from 2.6.36 onwards
1114*/
1115static int ath6kl_cfg80211_set_txpower(struct wiphy *wiphy,
1116 enum nl80211_tx_power_setting type,
1117 int dbm)
1118{
1119 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
1120 u8 ath6kl_dbm;
1121
1122 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x, dbm %d\n", __func__,
1123 type, dbm);
1124
1125 if (!ath6kl_cfg80211_ready(ar))
1126 return -EIO;
1127
1128 switch (type) {
1129 case NL80211_TX_POWER_AUTOMATIC:
1130 return 0;
1131 case NL80211_TX_POWER_LIMITED:
1132 ar->tx_pwr = ath6kl_dbm = dbm;
1133 break;
1134 default:
1135 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x not supported\n",
1136 __func__, type);
1137 return -EOPNOTSUPP;
1138 }
1139
1140 ath6kl_wmi_set_tx_pwr_cmd(ar->wmi, ath6kl_dbm);
1141
1142 return 0;
1143}
1144
1145static int ath6kl_cfg80211_get_txpower(struct wiphy *wiphy, int *dbm)
1146{
1147 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301148 struct ath6kl_vif *vif = ar->vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001149
1150 if (!ath6kl_cfg80211_ready(ar))
1151 return -EIO;
1152
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301153 if (test_bit(CONNECTED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001154 ar->tx_pwr = 0;
1155
1156 if (ath6kl_wmi_get_tx_pwr_cmd(ar->wmi) != 0) {
1157 ath6kl_err("ath6kl_wmi_get_tx_pwr_cmd failed\n");
1158 return -EIO;
1159 }
1160
1161 wait_event_interruptible_timeout(ar->event_wq, ar->tx_pwr != 0,
1162 5 * HZ);
1163
1164 if (signal_pending(current)) {
1165 ath6kl_err("target did not respond\n");
1166 return -EINTR;
1167 }
1168 }
1169
1170 *dbm = ar->tx_pwr;
1171 return 0;
1172}
1173
1174static int ath6kl_cfg80211_set_power_mgmt(struct wiphy *wiphy,
1175 struct net_device *dev,
1176 bool pmgmt, int timeout)
1177{
1178 struct ath6kl *ar = ath6kl_priv(dev);
1179 struct wmi_power_mode_cmd mode;
1180
1181 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: pmgmt %d, timeout %d\n",
1182 __func__, pmgmt, timeout);
1183
1184 if (!ath6kl_cfg80211_ready(ar))
1185 return -EIO;
1186
1187 if (pmgmt) {
1188 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: max perf\n", __func__);
1189 mode.pwr_mode = REC_POWER;
1190 } else {
1191 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: rec power\n", __func__);
1192 mode.pwr_mode = MAX_PERF_POWER;
1193 }
1194
1195 if (ath6kl_wmi_powermode_cmd(ar->wmi, mode.pwr_mode) != 0) {
1196 ath6kl_err("wmi_powermode_cmd failed\n");
1197 return -EIO;
1198 }
1199
1200 return 0;
1201}
1202
1203static int ath6kl_cfg80211_change_iface(struct wiphy *wiphy,
1204 struct net_device *ndev,
1205 enum nl80211_iftype type, u32 *flags,
1206 struct vif_params *params)
1207{
1208 struct ath6kl *ar = ath6kl_priv(ndev);
1209 struct wireless_dev *wdev = ar->wdev;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301210 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001211
1212 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type %u\n", __func__, type);
1213
1214 if (!ath6kl_cfg80211_ready(ar))
1215 return -EIO;
1216
1217 switch (type) {
1218 case NL80211_IFTYPE_STATION:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301219 vif->next_mode = INFRA_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001220 break;
1221 case NL80211_IFTYPE_ADHOC:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301222 vif->next_mode = ADHOC_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001223 break;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001224 case NL80211_IFTYPE_AP:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301225 vif->next_mode = AP_NETWORK;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001226 break;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001227 case NL80211_IFTYPE_P2P_CLIENT:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301228 vif->next_mode = INFRA_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001229 break;
1230 case NL80211_IFTYPE_P2P_GO:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301231 vif->next_mode = AP_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001232 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001233 default:
1234 ath6kl_err("invalid interface type %u\n", type);
1235 return -EOPNOTSUPP;
1236 }
1237
1238 wdev->iftype = type;
1239
1240 return 0;
1241}
1242
1243static int ath6kl_cfg80211_join_ibss(struct wiphy *wiphy,
1244 struct net_device *dev,
1245 struct cfg80211_ibss_params *ibss_param)
1246{
1247 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301248 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001249 int status;
1250
1251 if (!ath6kl_cfg80211_ready(ar))
1252 return -EIO;
1253
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301254 vif->ssid_len = ibss_param->ssid_len;
1255 memcpy(vif->ssid, ibss_param->ssid, vif->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +03001256
1257 if (ibss_param->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301258 vif->ch_hint = ibss_param->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +03001259
1260 if (ibss_param->channel_fixed) {
1261 /*
1262 * TODO: channel_fixed: The channel should be fixed, do not
1263 * search for IBSSs to join on other channels. Target
1264 * firmware does not support this feature, needs to be
1265 * updated.
1266 */
1267 return -EOPNOTSUPP;
1268 }
1269
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301270 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001271 if (ibss_param->bssid && !is_broadcast_ether_addr(ibss_param->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301272 memcpy(vif->req_bssid, ibss_param->bssid,
1273 sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001274
1275 ath6kl_set_wpa_version(ar, 0);
1276
1277 status = ath6kl_set_auth_type(ar, NL80211_AUTHTYPE_OPEN_SYSTEM);
1278 if (status)
1279 return status;
1280
1281 if (ibss_param->privacy) {
1282 ath6kl_set_cipher(ar, WLAN_CIPHER_SUITE_WEP40, true);
1283 ath6kl_set_cipher(ar, WLAN_CIPHER_SUITE_WEP40, false);
1284 } else {
1285 ath6kl_set_cipher(ar, 0, true);
1286 ath6kl_set_cipher(ar, 0, false);
1287 }
1288
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301289 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +03001290
1291 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1292 "%s: connect called with authmode %d dot11 auth %d"
1293 " PW crypto %d PW crypto len %d GRP crypto %d"
1294 " GRP crypto len %d channel hint %u\n",
1295 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301296 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
1297 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301298 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +03001299
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301300 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301301 vif->dot11_auth_mode, vif->auth_mode,
1302 vif->prwise_crypto,
1303 vif->prwise_crypto_len,
1304 vif->grp_crypto, vif->grp_crypto_len,
1305 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301306 vif->req_bssid, vif->ch_hint,
Kalle Valobdcd8172011-07-18 00:22:30 +03001307 ar->connect_ctrl_flags);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301308 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001309
1310 return 0;
1311}
1312
1313static int ath6kl_cfg80211_leave_ibss(struct wiphy *wiphy,
1314 struct net_device *dev)
1315{
1316 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301317 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001318
1319 if (!ath6kl_cfg80211_ready(ar))
1320 return -EIO;
1321
1322 ath6kl_disconnect(ar);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301323 memset(vif->ssid, 0, sizeof(vif->ssid));
1324 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001325
1326 return 0;
1327}
1328
1329static const u32 cipher_suites[] = {
1330 WLAN_CIPHER_SUITE_WEP40,
1331 WLAN_CIPHER_SUITE_WEP104,
1332 WLAN_CIPHER_SUITE_TKIP,
1333 WLAN_CIPHER_SUITE_CCMP,
Jouni Malinen837cb972011-10-11 17:31:57 +03001334 CCKM_KRK_CIPHER_SUITE,
Kalle Valobdcd8172011-07-18 00:22:30 +03001335};
1336
1337static bool is_rate_legacy(s32 rate)
1338{
1339 static const s32 legacy[] = { 1000, 2000, 5500, 11000,
1340 6000, 9000, 12000, 18000, 24000,
1341 36000, 48000, 54000
1342 };
1343 u8 i;
1344
1345 for (i = 0; i < ARRAY_SIZE(legacy); i++)
1346 if (rate == legacy[i])
1347 return true;
1348
1349 return false;
1350}
1351
1352static bool is_rate_ht20(s32 rate, u8 *mcs, bool *sgi)
1353{
1354 static const s32 ht20[] = { 6500, 13000, 19500, 26000, 39000,
1355 52000, 58500, 65000, 72200
1356 };
1357 u8 i;
1358
1359 for (i = 0; i < ARRAY_SIZE(ht20); i++) {
1360 if (rate == ht20[i]) {
1361 if (i == ARRAY_SIZE(ht20) - 1)
1362 /* last rate uses sgi */
1363 *sgi = true;
1364 else
1365 *sgi = false;
1366
1367 *mcs = i;
1368 return true;
1369 }
1370 }
1371 return false;
1372}
1373
1374static bool is_rate_ht40(s32 rate, u8 *mcs, bool *sgi)
1375{
1376 static const s32 ht40[] = { 13500, 27000, 40500, 54000,
1377 81000, 108000, 121500, 135000,
1378 150000
1379 };
1380 u8 i;
1381
1382 for (i = 0; i < ARRAY_SIZE(ht40); i++) {
1383 if (rate == ht40[i]) {
1384 if (i == ARRAY_SIZE(ht40) - 1)
1385 /* last rate uses sgi */
1386 *sgi = true;
1387 else
1388 *sgi = false;
1389
1390 *mcs = i;
1391 return true;
1392 }
1393 }
1394
1395 return false;
1396}
1397
1398static int ath6kl_get_station(struct wiphy *wiphy, struct net_device *dev,
1399 u8 *mac, struct station_info *sinfo)
1400{
1401 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301402 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001403 long left;
1404 bool sgi;
1405 s32 rate;
1406 int ret;
1407 u8 mcs;
1408
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301409 if (memcmp(mac, vif->bssid, ETH_ALEN) != 0)
Kalle Valobdcd8172011-07-18 00:22:30 +03001410 return -ENOENT;
1411
1412 if (down_interruptible(&ar->sem))
1413 return -EBUSY;
1414
1415 set_bit(STATS_UPDATE_PEND, &ar->flag);
1416
1417 ret = ath6kl_wmi_get_stats_cmd(ar->wmi);
1418
1419 if (ret != 0) {
1420 up(&ar->sem);
1421 return -EIO;
1422 }
1423
1424 left = wait_event_interruptible_timeout(ar->event_wq,
1425 !test_bit(STATS_UPDATE_PEND,
1426 &ar->flag),
1427 WMI_TIMEOUT);
1428
1429 up(&ar->sem);
1430
1431 if (left == 0)
1432 return -ETIMEDOUT;
1433 else if (left < 0)
1434 return left;
1435
1436 if (ar->target_stats.rx_byte) {
1437 sinfo->rx_bytes = ar->target_stats.rx_byte;
1438 sinfo->filled |= STATION_INFO_RX_BYTES;
1439 sinfo->rx_packets = ar->target_stats.rx_pkt;
1440 sinfo->filled |= STATION_INFO_RX_PACKETS;
1441 }
1442
1443 if (ar->target_stats.tx_byte) {
1444 sinfo->tx_bytes = ar->target_stats.tx_byte;
1445 sinfo->filled |= STATION_INFO_TX_BYTES;
1446 sinfo->tx_packets = ar->target_stats.tx_pkt;
1447 sinfo->filled |= STATION_INFO_TX_PACKETS;
1448 }
1449
1450 sinfo->signal = ar->target_stats.cs_rssi;
1451 sinfo->filled |= STATION_INFO_SIGNAL;
1452
1453 rate = ar->target_stats.tx_ucast_rate;
1454
1455 if (is_rate_legacy(rate)) {
1456 sinfo->txrate.legacy = rate / 100;
1457 } else if (is_rate_ht20(rate, &mcs, &sgi)) {
1458 if (sgi) {
1459 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1460 sinfo->txrate.mcs = mcs - 1;
1461 } else {
1462 sinfo->txrate.mcs = mcs;
1463 }
1464
1465 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1466 } else if (is_rate_ht40(rate, &mcs, &sgi)) {
1467 if (sgi) {
1468 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1469 sinfo->txrate.mcs = mcs - 1;
1470 } else {
1471 sinfo->txrate.mcs = mcs;
1472 }
1473
1474 sinfo->txrate.flags |= RATE_INFO_FLAGS_40_MHZ_WIDTH;
1475 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1476 } else {
Kalle Valo9a730832011-09-27 23:33:28 +03001477 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1478 "invalid rate from stats: %d\n", rate);
1479 ath6kl_debug_war(ar, ATH6KL_WAR_INVALID_RATE);
Kalle Valobdcd8172011-07-18 00:22:30 +03001480 return 0;
1481 }
1482
1483 sinfo->filled |= STATION_INFO_TX_BITRATE;
1484
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301485 if (test_bit(CONNECTED, &vif->flags) &&
1486 test_bit(DTIM_PERIOD_AVAIL, &vif->flags) &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301487 vif->nw_type == INFRA_NETWORK) {
Jouni Malinen32c10872011-09-19 19:15:07 +03001488 sinfo->filled |= STATION_INFO_BSS_PARAM;
1489 sinfo->bss_param.flags = 0;
1490 sinfo->bss_param.dtim_period = ar->assoc_bss_dtim_period;
1491 sinfo->bss_param.beacon_interval = ar->assoc_bss_beacon_int;
1492 }
1493
Kalle Valobdcd8172011-07-18 00:22:30 +03001494 return 0;
1495}
1496
1497static int ath6kl_set_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1498 struct cfg80211_pmksa *pmksa)
1499{
1500 struct ath6kl *ar = ath6kl_priv(netdev);
1501 return ath6kl_wmi_setpmkid_cmd(ar->wmi, pmksa->bssid,
1502 pmksa->pmkid, true);
1503}
1504
1505static int ath6kl_del_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1506 struct cfg80211_pmksa *pmksa)
1507{
1508 struct ath6kl *ar = ath6kl_priv(netdev);
1509 return ath6kl_wmi_setpmkid_cmd(ar->wmi, pmksa->bssid,
1510 pmksa->pmkid, false);
1511}
1512
1513static int ath6kl_flush_pmksa(struct wiphy *wiphy, struct net_device *netdev)
1514{
1515 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301516 struct ath6kl_vif *vif = netdev_priv(netdev);
1517
1518 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301519 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->bssid,
1520 NULL, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001521 return 0;
1522}
1523
Kalle Valoabcb3442011-07-22 08:26:20 +03001524#ifdef CONFIG_PM
1525static int ar6k_cfg80211_suspend(struct wiphy *wiphy,
1526 struct cfg80211_wowlan *wow)
1527{
1528 struct ath6kl *ar = wiphy_priv(wiphy);
1529
1530 return ath6kl_hif_suspend(ar);
1531}
Chilam Ngaa6cffc2011-10-05 10:12:52 +03001532
1533static int ar6k_cfg80211_resume(struct wiphy *wiphy)
1534{
1535 struct ath6kl *ar = wiphy_priv(wiphy);
1536
1537 return ath6kl_hif_resume(ar);
1538}
Kalle Valoabcb3442011-07-22 08:26:20 +03001539#endif
1540
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001541static int ath6kl_set_channel(struct wiphy *wiphy, struct net_device *dev,
1542 struct ieee80211_channel *chan,
1543 enum nl80211_channel_type channel_type)
1544{
1545 struct ath6kl *ar = ath6kl_priv(dev);
1546
1547 if (!ath6kl_cfg80211_ready(ar))
1548 return -EIO;
1549
1550 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: center_freq=%u hw_value=%u\n",
1551 __func__, chan->center_freq, chan->hw_value);
1552 ar->next_chan = chan->center_freq;
1553
1554 return 0;
1555}
1556
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001557static bool ath6kl_is_p2p_ie(const u8 *pos)
1558{
1559 return pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
1560 pos[2] == 0x50 && pos[3] == 0x6f &&
1561 pos[4] == 0x9a && pos[5] == 0x09;
1562}
1563
1564static int ath6kl_set_ap_probe_resp_ies(struct ath6kl *ar, const u8 *ies,
1565 size_t ies_len)
1566{
1567 const u8 *pos;
1568 u8 *buf = NULL;
1569 size_t len = 0;
1570 int ret;
1571
1572 /*
1573 * Filter out P2P IE(s) since they will be included depending on
1574 * the Probe Request frame in ath6kl_send_go_probe_resp().
1575 */
1576
1577 if (ies && ies_len) {
1578 buf = kmalloc(ies_len, GFP_KERNEL);
1579 if (buf == NULL)
1580 return -ENOMEM;
1581 pos = ies;
1582 while (pos + 1 < ies + ies_len) {
1583 if (pos + 2 + pos[1] > ies + ies_len)
1584 break;
1585 if (!ath6kl_is_p2p_ie(pos)) {
1586 memcpy(buf + len, pos, 2 + pos[1]);
1587 len += 2 + pos[1];
1588 }
1589 pos += 2 + pos[1];
1590 }
1591 }
1592
1593 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, WMI_FRAME_PROBE_RESP,
1594 buf, len);
1595 kfree(buf);
1596 return ret;
1597}
1598
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001599static int ath6kl_ap_beacon(struct wiphy *wiphy, struct net_device *dev,
1600 struct beacon_parameters *info, bool add)
1601{
1602 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301603 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001604 struct ieee80211_mgmt *mgmt;
1605 u8 *ies;
1606 int ies_len;
1607 struct wmi_connect_cmd p;
1608 int res;
1609 int i;
1610
1611 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: add=%d\n", __func__, add);
1612
1613 if (!ath6kl_cfg80211_ready(ar))
1614 return -EIO;
1615
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301616 if (vif->next_mode != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001617 return -EOPNOTSUPP;
1618
1619 if (info->beacon_ies) {
1620 res = ath6kl_wmi_set_appie_cmd(ar->wmi, WMI_FRAME_BEACON,
1621 info->beacon_ies,
1622 info->beacon_ies_len);
1623 if (res)
1624 return res;
1625 }
1626 if (info->proberesp_ies) {
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001627 res = ath6kl_set_ap_probe_resp_ies(ar, info->proberesp_ies,
1628 info->proberesp_ies_len);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001629 if (res)
1630 return res;
1631 }
1632 if (info->assocresp_ies) {
1633 res = ath6kl_wmi_set_appie_cmd(ar->wmi, WMI_FRAME_ASSOC_RESP,
1634 info->assocresp_ies,
1635 info->assocresp_ies_len);
1636 if (res)
1637 return res;
1638 }
1639
1640 if (!add)
1641 return 0;
1642
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001643 ar->ap_mode_bkey.valid = false;
1644
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001645 /* TODO:
1646 * info->interval
1647 * info->dtim_period
1648 */
1649
1650 if (info->head == NULL)
1651 return -EINVAL;
1652 mgmt = (struct ieee80211_mgmt *) info->head;
1653 ies = mgmt->u.beacon.variable;
1654 if (ies > info->head + info->head_len)
1655 return -EINVAL;
1656 ies_len = info->head + info->head_len - ies;
1657
1658 if (info->ssid == NULL)
1659 return -EINVAL;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301660 memcpy(vif->ssid, info->ssid, info->ssid_len);
1661 vif->ssid_len = info->ssid_len;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001662 if (info->hidden_ssid != NL80211_HIDDEN_SSID_NOT_IN_USE)
1663 return -EOPNOTSUPP; /* TODO */
1664
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301665 vif->dot11_auth_mode = OPEN_AUTH;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001666
1667 memset(&p, 0, sizeof(p));
1668
1669 for (i = 0; i < info->crypto.n_akm_suites; i++) {
1670 switch (info->crypto.akm_suites[i]) {
1671 case WLAN_AKM_SUITE_8021X:
1672 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
1673 p.auth_mode |= WPA_AUTH;
1674 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
1675 p.auth_mode |= WPA2_AUTH;
1676 break;
1677 case WLAN_AKM_SUITE_PSK:
1678 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
1679 p.auth_mode |= WPA_PSK_AUTH;
1680 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
1681 p.auth_mode |= WPA2_PSK_AUTH;
1682 break;
1683 }
1684 }
1685 if (p.auth_mode == 0)
1686 p.auth_mode = NONE_AUTH;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301687 vif->auth_mode = p.auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001688
1689 for (i = 0; i < info->crypto.n_ciphers_pairwise; i++) {
1690 switch (info->crypto.ciphers_pairwise[i]) {
1691 case WLAN_CIPHER_SUITE_WEP40:
1692 case WLAN_CIPHER_SUITE_WEP104:
1693 p.prwise_crypto_type |= WEP_CRYPT;
1694 break;
1695 case WLAN_CIPHER_SUITE_TKIP:
1696 p.prwise_crypto_type |= TKIP_CRYPT;
1697 break;
1698 case WLAN_CIPHER_SUITE_CCMP:
1699 p.prwise_crypto_type |= AES_CRYPT;
1700 break;
1701 }
1702 }
Edward Lu229ed6b2011-08-30 21:58:07 +03001703 if (p.prwise_crypto_type == 0) {
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001704 p.prwise_crypto_type = NONE_CRYPT;
Edward Lu229ed6b2011-08-30 21:58:07 +03001705 ath6kl_set_cipher(ar, 0, true);
1706 } else if (info->crypto.n_ciphers_pairwise == 1)
1707 ath6kl_set_cipher(ar, info->crypto.ciphers_pairwise[0], true);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001708
1709 switch (info->crypto.cipher_group) {
1710 case WLAN_CIPHER_SUITE_WEP40:
1711 case WLAN_CIPHER_SUITE_WEP104:
1712 p.grp_crypto_type = WEP_CRYPT;
1713 break;
1714 case WLAN_CIPHER_SUITE_TKIP:
1715 p.grp_crypto_type = TKIP_CRYPT;
1716 break;
1717 case WLAN_CIPHER_SUITE_CCMP:
1718 p.grp_crypto_type = AES_CRYPT;
1719 break;
1720 default:
1721 p.grp_crypto_type = NONE_CRYPT;
1722 break;
1723 }
Edward Lu229ed6b2011-08-30 21:58:07 +03001724 ath6kl_set_cipher(ar, info->crypto.cipher_group, false);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001725
1726 p.nw_type = AP_NETWORK;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301727 vif->nw_type = vif->next_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001728
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301729 p.ssid_len = vif->ssid_len;
1730 memcpy(p.ssid, vif->ssid, vif->ssid_len);
1731 p.dot11_auth_mode = vif->dot11_auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001732 p.ch = cpu_to_le16(ar->next_chan);
1733
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001734 res = ath6kl_wmi_ap_profile_commit(ar->wmi, &p);
1735 if (res < 0)
1736 return res;
1737
1738 return 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001739}
1740
1741static int ath6kl_add_beacon(struct wiphy *wiphy, struct net_device *dev,
1742 struct beacon_parameters *info)
1743{
1744 return ath6kl_ap_beacon(wiphy, dev, info, true);
1745}
1746
1747static int ath6kl_set_beacon(struct wiphy *wiphy, struct net_device *dev,
1748 struct beacon_parameters *info)
1749{
1750 return ath6kl_ap_beacon(wiphy, dev, info, false);
1751}
1752
1753static int ath6kl_del_beacon(struct wiphy *wiphy, struct net_device *dev)
1754{
1755 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301756 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001757
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301758 if (vif->nw_type != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001759 return -EOPNOTSUPP;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301760 if (!test_bit(CONNECTED, &vif->flags))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001761 return -ENOTCONN;
1762
1763 ath6kl_wmi_disconnect_cmd(ar->wmi);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301764 clear_bit(CONNECTED, &vif->flags);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001765
1766 return 0;
1767}
1768
Jouni Malinen23875132011-08-30 21:57:53 +03001769static int ath6kl_change_station(struct wiphy *wiphy, struct net_device *dev,
1770 u8 *mac, struct station_parameters *params)
1771{
1772 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301773 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen23875132011-08-30 21:57:53 +03001774
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301775 if (vif->nw_type != AP_NETWORK)
Jouni Malinen23875132011-08-30 21:57:53 +03001776 return -EOPNOTSUPP;
1777
1778 /* Use this only for authorizing/unauthorizing a station */
1779 if (!(params->sta_flags_mask & BIT(NL80211_STA_FLAG_AUTHORIZED)))
1780 return -EOPNOTSUPP;
1781
1782 if (params->sta_flags_set & BIT(NL80211_STA_FLAG_AUTHORIZED))
1783 return ath6kl_wmi_ap_set_mlme(ar->wmi, WMI_AP_MLME_AUTHORIZE,
1784 mac, 0);
1785 return ath6kl_wmi_ap_set_mlme(ar->wmi, WMI_AP_MLME_UNAUTHORIZE, mac,
1786 0);
1787}
1788
Jouni Malinen63fa1e02011-08-30 21:57:55 +03001789static int ath6kl_remain_on_channel(struct wiphy *wiphy,
1790 struct net_device *dev,
1791 struct ieee80211_channel *chan,
1792 enum nl80211_channel_type channel_type,
1793 unsigned int duration,
1794 u64 *cookie)
1795{
1796 struct ath6kl *ar = ath6kl_priv(dev);
1797
1798 /* TODO: if already pending or ongoing remain-on-channel,
1799 * return -EBUSY */
1800 *cookie = 1; /* only a single pending request is supported */
1801
1802 return ath6kl_wmi_remain_on_chnl_cmd(ar->wmi, chan->center_freq,
1803 duration);
1804}
1805
1806static int ath6kl_cancel_remain_on_channel(struct wiphy *wiphy,
1807 struct net_device *dev,
1808 u64 cookie)
1809{
1810 struct ath6kl *ar = ath6kl_priv(dev);
1811
1812 if (cookie != 1)
1813 return -ENOENT;
1814
1815 return ath6kl_wmi_cancel_remain_on_chnl_cmd(ar->wmi);
1816}
1817
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001818static int ath6kl_send_go_probe_resp(struct ath6kl *ar, const u8 *buf,
1819 size_t len, unsigned int freq)
1820{
1821 const u8 *pos;
1822 u8 *p2p;
1823 int p2p_len;
1824 int ret;
1825 const struct ieee80211_mgmt *mgmt;
1826
1827 mgmt = (const struct ieee80211_mgmt *) buf;
1828
1829 /* Include P2P IE(s) from the frame generated in user space. */
1830
1831 p2p = kmalloc(len, GFP_KERNEL);
1832 if (p2p == NULL)
1833 return -ENOMEM;
1834 p2p_len = 0;
1835
1836 pos = mgmt->u.probe_resp.variable;
1837 while (pos + 1 < buf + len) {
1838 if (pos + 2 + pos[1] > buf + len)
1839 break;
1840 if (ath6kl_is_p2p_ie(pos)) {
1841 memcpy(p2p + p2p_len, pos, 2 + pos[1]);
1842 p2p_len += 2 + pos[1];
1843 }
1844 pos += 2 + pos[1];
1845 }
1846
1847 ret = ath6kl_wmi_send_probe_response_cmd(ar->wmi, freq, mgmt->da,
1848 p2p, p2p_len);
1849 kfree(p2p);
1850 return ret;
1851}
1852
Jouni Malinen8a6c80602011-08-30 21:57:56 +03001853static int ath6kl_mgmt_tx(struct wiphy *wiphy, struct net_device *dev,
1854 struct ieee80211_channel *chan, bool offchan,
1855 enum nl80211_channel_type channel_type,
1856 bool channel_type_valid, unsigned int wait,
Johannes Berge247bd902011-11-04 11:18:21 +01001857 const u8 *buf, size_t len, bool no_cck,
1858 bool dont_wait_for_ack, u64 *cookie)
Jouni Malinen8a6c80602011-08-30 21:57:56 +03001859{
1860 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301861 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03001862 u32 id;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001863 const struct ieee80211_mgmt *mgmt;
1864
1865 mgmt = (const struct ieee80211_mgmt *) buf;
1866 if (buf + len >= mgmt->u.probe_resp.variable &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301867 vif->nw_type == AP_NETWORK && test_bit(CONNECTED, &vif->flags) &&
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001868 ieee80211_is_probe_resp(mgmt->frame_control)) {
1869 /*
1870 * Send Probe Response frame in AP mode using a separate WMI
1871 * command to allow the target to fill in the generic IEs.
1872 */
1873 *cookie = 0; /* TX status not supported */
1874 return ath6kl_send_go_probe_resp(ar, buf, len,
1875 chan->center_freq);
1876 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03001877
1878 id = ar->send_action_id++;
1879 if (id == 0) {
1880 /*
1881 * 0 is a reserved value in the WMI command and shall not be
1882 * used for the command.
1883 */
1884 id = ar->send_action_id++;
1885 }
1886
1887 *cookie = id;
1888 return ath6kl_wmi_send_action_cmd(ar->wmi, id, chan->center_freq, wait,
1889 buf, len);
1890}
1891
Jouni Malinenae32c302011-08-30 21:58:01 +03001892static void ath6kl_mgmt_frame_register(struct wiphy *wiphy,
1893 struct net_device *dev,
1894 u16 frame_type, bool reg)
1895{
1896 struct ath6kl *ar = ath6kl_priv(dev);
1897
1898 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: frame_type=0x%x reg=%d\n",
1899 __func__, frame_type, reg);
1900 if (frame_type == IEEE80211_STYPE_PROBE_REQ) {
1901 /*
1902 * Note: This notification callback is not allowed to sleep, so
1903 * we cannot send WMI_PROBE_REQ_REPORT_CMD here. Instead, we
1904 * hardcode target to report Probe Request frames all the time.
1905 */
1906 ar->probe_req_report = reg;
1907 }
1908}
1909
Jouni Malinenf80574a2011-08-30 21:58:04 +03001910static const struct ieee80211_txrx_stypes
1911ath6kl_mgmt_stypes[NUM_NL80211_IFTYPES] = {
1912 [NL80211_IFTYPE_STATION] = {
1913 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1914 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
1915 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1916 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
1917 },
1918 [NL80211_IFTYPE_P2P_CLIENT] = {
1919 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1920 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
1921 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1922 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
1923 },
1924 [NL80211_IFTYPE_P2P_GO] = {
1925 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1926 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
1927 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
1928 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
1929 },
1930};
1931
Kalle Valobdcd8172011-07-18 00:22:30 +03001932static struct cfg80211_ops ath6kl_cfg80211_ops = {
1933 .change_virtual_intf = ath6kl_cfg80211_change_iface,
1934 .scan = ath6kl_cfg80211_scan,
1935 .connect = ath6kl_cfg80211_connect,
1936 .disconnect = ath6kl_cfg80211_disconnect,
1937 .add_key = ath6kl_cfg80211_add_key,
1938 .get_key = ath6kl_cfg80211_get_key,
1939 .del_key = ath6kl_cfg80211_del_key,
1940 .set_default_key = ath6kl_cfg80211_set_default_key,
1941 .set_wiphy_params = ath6kl_cfg80211_set_wiphy_params,
1942 .set_tx_power = ath6kl_cfg80211_set_txpower,
1943 .get_tx_power = ath6kl_cfg80211_get_txpower,
1944 .set_power_mgmt = ath6kl_cfg80211_set_power_mgmt,
1945 .join_ibss = ath6kl_cfg80211_join_ibss,
1946 .leave_ibss = ath6kl_cfg80211_leave_ibss,
1947 .get_station = ath6kl_get_station,
1948 .set_pmksa = ath6kl_set_pmksa,
1949 .del_pmksa = ath6kl_del_pmksa,
1950 .flush_pmksa = ath6kl_flush_pmksa,
Kalle Valo003353b0d2011-09-01 10:14:21 +03001951 CFG80211_TESTMODE_CMD(ath6kl_tm_cmd)
Kalle Valoabcb3442011-07-22 08:26:20 +03001952#ifdef CONFIG_PM
1953 .suspend = ar6k_cfg80211_suspend,
Chilam Ngaa6cffc2011-10-05 10:12:52 +03001954 .resume = ar6k_cfg80211_resume,
Kalle Valoabcb3442011-07-22 08:26:20 +03001955#endif
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001956 .set_channel = ath6kl_set_channel,
1957 .add_beacon = ath6kl_add_beacon,
1958 .set_beacon = ath6kl_set_beacon,
1959 .del_beacon = ath6kl_del_beacon,
Jouni Malinen23875132011-08-30 21:57:53 +03001960 .change_station = ath6kl_change_station,
Jouni Malinen63fa1e02011-08-30 21:57:55 +03001961 .remain_on_channel = ath6kl_remain_on_channel,
1962 .cancel_remain_on_channel = ath6kl_cancel_remain_on_channel,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03001963 .mgmt_tx = ath6kl_mgmt_tx,
Jouni Malinenae32c302011-08-30 21:58:01 +03001964 .mgmt_frame_register = ath6kl_mgmt_frame_register,
Kalle Valobdcd8172011-07-18 00:22:30 +03001965};
1966
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05301967struct ath6kl *ath6kl_core_alloc(struct device *dev)
Kalle Valobdcd8172011-07-18 00:22:30 +03001968{
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03001969 struct ath6kl *ar;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05301970 struct wiphy *wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05301971 u8 ctr;
Kalle Valobdcd8172011-07-18 00:22:30 +03001972
1973 /* create a new wiphy for use with cfg80211 */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05301974 wiphy = wiphy_new(&ath6kl_cfg80211_ops, sizeof(struct ath6kl));
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05301975
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05301976 if (!wiphy) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001977 ath6kl_err("couldn't allocate wiphy device\n");
Kalle Valobdcd8172011-07-18 00:22:30 +03001978 return NULL;
1979 }
1980
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05301981 ar = wiphy_priv(wiphy);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03001982 ar->p2p = !!ath6kl_p2p;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05301983 ar->wiphy = wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05301984 ar->dev = dev;
1985
1986 spin_lock_init(&ar->lock);
1987 spin_lock_init(&ar->mcastpsq_lock);
1988
1989 init_waitqueue_head(&ar->event_wq);
1990 sema_init(&ar->sem, 1);
1991
1992 INIT_LIST_HEAD(&ar->amsdu_rx_buffer_queue);
1993
1994 clear_bit(WMI_ENABLED, &ar->flag);
1995 clear_bit(SKIP_SCAN, &ar->flag);
1996 clear_bit(DESTROY_IN_PROGRESS, &ar->flag);
1997
1998 ar->listen_intvl_t = A_DEFAULT_LISTEN_INTERVAL;
1999 ar->listen_intvl_b = 0;
2000 ar->tx_pwr = 0;
2001
2002 ar->intra_bss = 1;
2003 memset(&ar->sc_params, 0, sizeof(ar->sc_params));
2004 ar->sc_params.short_scan_ratio = WMI_SHORTSCANRATIO_DEFAULT;
2005 ar->sc_params.scan_ctrl_flags = DEFAULT_SCAN_CTRL_FLAGS;
2006 ar->lrssi_roam_threshold = DEF_LRSSI_ROAM_THRESHOLD;
2007
2008 memset((u8 *)ar->sta_list, 0,
2009 AP_MAX_NUM_STA * sizeof(struct ath6kl_sta));
2010
2011 /* Init the PS queues */
2012 for (ctr = 0; ctr < AP_MAX_NUM_STA; ctr++) {
2013 spin_lock_init(&ar->sta_list[ctr].psq_lock);
2014 skb_queue_head_init(&ar->sta_list[ctr].psq);
2015 }
2016
2017 skb_queue_head_init(&ar->mcastpsq);
2018
2019 memcpy(ar->ap_country_code, DEF_AP_COUNTRY_CODE, 3);
2020
2021 return ar;
2022}
2023
2024int ath6kl_register_ieee80211_hw(struct ath6kl *ar)
2025{
2026 struct wiphy *wiphy = ar->wiphy;
2027 int ret;
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002028
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302029 wiphy->mgmt_stypes = ath6kl_mgmt_stypes;
Jouni Malinenf80574a2011-08-30 21:58:04 +03002030
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302031 wiphy->max_remain_on_channel_duration = 5000;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002032
Kalle Valobdcd8172011-07-18 00:22:30 +03002033 /* set device pointer for wiphy */
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302034 set_wiphy_dev(wiphy, ar->dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03002035
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302036 wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302037 BIT(NL80211_IFTYPE_ADHOC) |
2038 BIT(NL80211_IFTYPE_AP);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002039 if (ar->p2p) {
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302040 wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_GO) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302041 BIT(NL80211_IFTYPE_P2P_CLIENT);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002042 }
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302043
Kalle Valobdcd8172011-07-18 00:22:30 +03002044 /* max num of ssids that can be probed during scanning */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302045 wiphy->max_scan_ssids = MAX_PROBED_SSID_INDEX;
2046 wiphy->max_scan_ie_len = 1000; /* FIX: what is correct limit? */
2047 wiphy->bands[IEEE80211_BAND_2GHZ] = &ath6kl_band_2ghz;
2048 wiphy->bands[IEEE80211_BAND_5GHZ] = &ath6kl_band_5ghz;
2049 wiphy->signal_type = CFG80211_SIGNAL_TYPE_MBM;
Kalle Valobdcd8172011-07-18 00:22:30 +03002050
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302051 wiphy->cipher_suites = cipher_suites;
2052 wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
Kalle Valobdcd8172011-07-18 00:22:30 +03002053
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302054 ret = wiphy_register(wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002055 if (ret < 0) {
2056 ath6kl_err("couldn't register wiphy device\n");
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302057 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +03002058 }
2059
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302060 return 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03002061}
2062
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302063static int ath6kl_init_if_data(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +03002064{
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302065 struct ath6kl *ar = vif->ar;
2066
2067 ar->aggr_cntxt = aggr_init(vif->ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302068 if (!ar->aggr_cntxt) {
2069 ath6kl_err("failed to initialize aggr\n");
2070 return -ENOMEM;
2071 }
Kalle Valobdcd8172011-07-18 00:22:30 +03002072
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302073 setup_timer(&ar->disconnect_timer, disconnect_timer_handler,
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302074 (unsigned long) vif->ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302075
2076 return 0;
2077}
2078
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302079void ath6kl_deinit_if_data(struct ath6kl_vif *vif)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302080{
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302081 struct ath6kl *ar = vif->ar;
2082
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302083 aggr_module_destroy(ar->aggr_cntxt);
2084
2085 ar->aggr_cntxt = NULL;
2086
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302087 if (test_bit(NETDEV_REGISTERED, &vif->flags)) {
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302088 unregister_netdev(vif->ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302089 clear_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302090 }
2091
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302092 free_netdev(vif->ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302093}
2094
2095struct net_device *ath6kl_interface_add(struct ath6kl *ar, char *name,
2096 enum nl80211_iftype type)
2097{
2098 struct net_device *ndev;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302099 struct ath6kl_vif *vif;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302100
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302101 ndev = alloc_netdev(sizeof(*vif), "wlan%d", ether_setup);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302102 if (!ndev)
2103 return NULL;
2104
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302105 vif = netdev_priv(ndev);
2106 ndev->ieee80211_ptr = &vif->wdev;
2107 vif->wdev.wiphy = ar->wiphy;
2108 vif->ar = ar;
2109 ar->vif = vif;
2110 vif->ndev = ndev;
2111 SET_NETDEV_DEV(ndev, wiphy_dev(vif->wdev.wiphy));
2112 vif->wdev.netdev = ndev;
2113 vif->wdev.iftype = type;
2114 ar->wdev = &vif->wdev;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302115 ar->net_dev = ndev;
2116
2117 init_netdev(ndev);
2118
2119 ath6kl_init_control_info(ar);
2120
2121 /* TODO: Pass interface specific pointer instead of ar */
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302122 if (ath6kl_init_if_data(vif))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302123 goto err;
2124
2125 if (register_netdev(ndev))
2126 goto err;
2127
2128 ar->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302129 set_bit(WLAN_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302130 ar->wlan_pwr_state = WLAN_POWER_STATE_ON;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302131 set_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302132
2133 return ndev;
2134
2135err:
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302136 ath6kl_deinit_if_data(vif);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302137
2138 return NULL;
2139}
2140
2141void ath6kl_deinit_ieee80211_hw(struct ath6kl *ar)
2142{
Kalle Valobdcd8172011-07-18 00:22:30 +03002143 if (ar->scan_req) {
2144 cfg80211_scan_done(ar->scan_req, true);
2145 ar->scan_req = NULL;
2146 }
2147
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302148 wiphy_unregister(ar->wiphy);
2149 wiphy_free(ar->wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002150}