blob: 364f7884a3d37dd72396bcee286a993ec8e963ea [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
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530126static int ath6kl_set_wpa_version(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300127 enum nl80211_wpa_versions wpa_version)
128{
129 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: %u\n", __func__, wpa_version);
130
131 if (!wpa_version) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530132 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300133 } else if (wpa_version & NL80211_WPA_VERSION_2) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530134 vif->auth_mode = WPA2_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300135 } else if (wpa_version & NL80211_WPA_VERSION_1) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530136 vif->auth_mode = WPA_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300137 } else {
138 ath6kl_err("%s: %u not supported\n", __func__, wpa_version);
139 return -ENOTSUPP;
140 }
141
142 return 0;
143}
144
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530145static int ath6kl_set_auth_type(struct ath6kl_vif *vif,
Kalle Valobdcd8172011-07-18 00:22:30 +0300146 enum nl80211_auth_type auth_type)
147{
Kalle Valobdcd8172011-07-18 00:22:30 +0300148 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, auth_type);
149
150 switch (auth_type) {
151 case NL80211_AUTHTYPE_OPEN_SYSTEM:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530152 vif->dot11_auth_mode = OPEN_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300153 break;
154 case NL80211_AUTHTYPE_SHARED_KEY:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530155 vif->dot11_auth_mode = SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300156 break;
157 case NL80211_AUTHTYPE_NETWORK_EAP:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530158 vif->dot11_auth_mode = LEAP_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300159 break;
160
161 case NL80211_AUTHTYPE_AUTOMATIC:
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530162 vif->dot11_auth_mode = OPEN_AUTH | SHARED_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300163 break;
164
165 default:
166 ath6kl_err("%s: 0x%x not spported\n", __func__, auth_type);
167 return -ENOTSUPP;
168 }
169
170 return 0;
171}
172
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530173static int ath6kl_set_cipher(struct ath6kl_vif *vif, u32 cipher, bool ucast)
Kalle Valobdcd8172011-07-18 00:22:30 +0300174{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530175 u8 *ar_cipher = ucast ? &vif->prwise_crypto : &vif->grp_crypto;
176 u8 *ar_cipher_len = ucast ? &vif->prwise_crypto_len :
177 &vif->grp_crypto_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300178
179 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: cipher 0x%x, ucast %u\n",
180 __func__, cipher, ucast);
181
182 switch (cipher) {
183 case 0:
184 /* our own hack to use value 0 as no crypto used */
185 *ar_cipher = NONE_CRYPT;
186 *ar_cipher_len = 0;
187 break;
188 case WLAN_CIPHER_SUITE_WEP40:
189 *ar_cipher = WEP_CRYPT;
190 *ar_cipher_len = 5;
191 break;
192 case WLAN_CIPHER_SUITE_WEP104:
193 *ar_cipher = WEP_CRYPT;
194 *ar_cipher_len = 13;
195 break;
196 case WLAN_CIPHER_SUITE_TKIP:
197 *ar_cipher = TKIP_CRYPT;
198 *ar_cipher_len = 0;
199 break;
200 case WLAN_CIPHER_SUITE_CCMP:
201 *ar_cipher = AES_CRYPT;
202 *ar_cipher_len = 0;
203 break;
Dai Shuibing5e070212011-11-03 11:39:37 +0200204 case WLAN_CIPHER_SUITE_SMS4:
205 *ar_cipher = WAPI_CRYPT;
206 *ar_cipher_len = 0;
207 break;
Kalle Valobdcd8172011-07-18 00:22:30 +0300208 default:
209 ath6kl_err("cipher 0x%x not supported\n", cipher);
210 return -ENOTSUPP;
211 }
212
213 return 0;
214}
215
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530216static void ath6kl_set_key_mgmt(struct ath6kl_vif *vif, u32 key_mgmt)
Kalle Valobdcd8172011-07-18 00:22:30 +0300217{
218 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: 0x%x\n", __func__, key_mgmt);
219
220 if (key_mgmt == WLAN_AKM_SUITE_PSK) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530221 if (vif->auth_mode == WPA_AUTH)
222 vif->auth_mode = WPA_PSK_AUTH;
223 else if (vif->auth_mode == WPA2_AUTH)
224 vif->auth_mode = WPA2_PSK_AUTH;
Jouni Malinen837cb972011-10-11 17:31:57 +0300225 } else if (key_mgmt == 0x00409600) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530226 if (vif->auth_mode == WPA_AUTH)
227 vif->auth_mode = WPA_AUTH_CCKM;
228 else if (vif->auth_mode == WPA2_AUTH)
229 vif->auth_mode = WPA2_AUTH_CCKM;
Kalle Valobdcd8172011-07-18 00:22:30 +0300230 } else if (key_mgmt != WLAN_AKM_SUITE_8021X) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530231 vif->auth_mode = NONE_AUTH;
Kalle Valobdcd8172011-07-18 00:22:30 +0300232 }
233}
234
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530235static bool ath6kl_cfg80211_ready(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +0300236{
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530237 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530238
Kalle Valobdcd8172011-07-18 00:22:30 +0300239 if (!test_bit(WMI_READY, &ar->flag)) {
240 ath6kl_err("wmi is not ready\n");
241 return false;
242 }
243
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530244 if (!test_bit(WLAN_ENABLED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300245 ath6kl_err("wlan disabled\n");
246 return false;
247 }
248
249 return true;
250}
251
Kevin Fang6981ffd2011-10-07 08:51:19 +0800252static bool ath6kl_is_wpa_ie(const u8 *pos)
253{
254 return pos[0] == WLAN_EID_WPA && pos[1] >= 4 &&
255 pos[2] == 0x00 && pos[3] == 0x50 &&
256 pos[4] == 0xf2 && pos[5] == 0x01;
257}
258
259static bool ath6kl_is_rsn_ie(const u8 *pos)
260{
261 return pos[0] == WLAN_EID_RSN;
262}
263
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700264static bool ath6kl_is_wps_ie(const u8 *pos)
265{
266 return (pos[0] == WLAN_EID_VENDOR_SPECIFIC &&
267 pos[1] >= 4 &&
268 pos[2] == 0x00 && pos[3] == 0x50 && pos[4] == 0xf2 &&
269 pos[5] == 0x04);
270}
271
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530272static int ath6kl_set_assoc_req_ies(struct ath6kl_vif *vif, const u8 *ies,
273 size_t ies_len)
Kevin Fang6981ffd2011-10-07 08:51:19 +0800274{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530275 struct ath6kl *ar = vif->ar;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800276 const u8 *pos;
277 u8 *buf = NULL;
278 size_t len = 0;
279 int ret;
280
281 /*
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700282 * Clear previously set flag
283 */
284
285 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
286
287 /*
Kevin Fang6981ffd2011-10-07 08:51:19 +0800288 * Filter out RSN/WPA IE(s)
289 */
290
291 if (ies && ies_len) {
292 buf = kmalloc(ies_len, GFP_KERNEL);
293 if (buf == NULL)
294 return -ENOMEM;
295 pos = ies;
296
297 while (pos + 1 < ies + ies_len) {
298 if (pos + 2 + pos[1] > ies + ies_len)
299 break;
300 if (!(ath6kl_is_wpa_ie(pos) || ath6kl_is_rsn_ie(pos))) {
301 memcpy(buf + len, pos, 2 + pos[1]);
302 len += 2 + pos[1];
303 }
Aarthi Thiruvengadam63541212011-10-25 11:25:52 -0700304
305 if (ath6kl_is_wps_ie(pos))
306 ar->connect_ctrl_flags |= CONNECT_WPS_FLAG;
307
Kevin Fang6981ffd2011-10-07 08:51:19 +0800308 pos += 2 + pos[1];
309 }
310 }
311
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530312 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
313 WMI_FRAME_ASSOC_REQ, buf, len);
Kevin Fang6981ffd2011-10-07 08:51:19 +0800314 kfree(buf);
315 return ret;
316}
317
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530318static int ath6kl_nliftype_to_drv_iftype(enum nl80211_iftype type, u8 *nw_type)
319{
320 switch (type) {
321 case NL80211_IFTYPE_STATION:
322 *nw_type = INFRA_NETWORK;
323 break;
324 case NL80211_IFTYPE_ADHOC:
325 *nw_type = ADHOC_NETWORK;
326 break;
327 case NL80211_IFTYPE_AP:
328 *nw_type = AP_NETWORK;
329 break;
330 case NL80211_IFTYPE_P2P_CLIENT:
331 *nw_type = INFRA_NETWORK;
332 break;
333 case NL80211_IFTYPE_P2P_GO:
334 *nw_type = AP_NETWORK;
335 break;
336 default:
337 ath6kl_err("invalid interface type %u\n", type);
338 return -ENOTSUPP;
339 }
340
341 return 0;
342}
343
344static bool ath6kl_is_valid_iftype(struct ath6kl *ar, enum nl80211_iftype type,
345 u8 *if_idx, u8 *nw_type)
346{
347 int i;
348
349 if (ath6kl_nliftype_to_drv_iftype(type, nw_type))
350 return false;
351
352 if (ar->ibss_if_active || ((type == NL80211_IFTYPE_ADHOC) &&
353 ar->num_vif))
354 return false;
355
356 if (type == NL80211_IFTYPE_STATION ||
357 type == NL80211_IFTYPE_AP || type == NL80211_IFTYPE_ADHOC) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200358 for (i = 0; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530359 if ((ar->avail_idx_map >> i) & BIT(0)) {
360 *if_idx = i;
361 return true;
362 }
363 }
364 }
365
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530366 if (type == NL80211_IFTYPE_P2P_CLIENT ||
367 type == NL80211_IFTYPE_P2P_GO) {
Kalle Valo71f96ee2011-11-14 19:31:30 +0200368 for (i = ar->max_norm_iface; i < ar->vif_max; i++) {
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +0530369 if ((ar->avail_idx_map >> i) & BIT(0)) {
370 *if_idx = i;
371 return true;
372 }
373 }
374 }
375
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +0530376 return false;
377}
378
Kalle Valobdcd8172011-07-18 00:22:30 +0300379static int ath6kl_cfg80211_connect(struct wiphy *wiphy, struct net_device *dev,
380 struct cfg80211_connect_params *sme)
381{
382 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530383 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300384 int status;
385
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530386 vif->sme_state = SME_CONNECTING;
Kalle Valobdcd8172011-07-18 00:22:30 +0300387
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530388 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300389 return -EIO;
390
391 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
392 ath6kl_err("destroy in progress\n");
393 return -EBUSY;
394 }
395
396 if (test_bit(SKIP_SCAN, &ar->flag) &&
397 ((sme->channel && sme->channel->center_freq == 0) ||
398 (sme->bssid && is_zero_ether_addr(sme->bssid)))) {
399 ath6kl_err("SkipScan: channel or bssid invalid\n");
400 return -EINVAL;
401 }
402
403 if (down_interruptible(&ar->sem)) {
404 ath6kl_err("busy, couldn't get access\n");
405 return -ERESTARTSYS;
406 }
407
408 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
409 ath6kl_err("busy, destroy in progress\n");
410 up(&ar->sem);
411 return -EBUSY;
412 }
413
414 if (ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)]) {
415 /*
416 * sleep until the command queue drains
417 */
418 wait_event_interruptible_timeout(ar->event_wq,
419 ar->tx_pending[ath6kl_wmi_get_control_ep(ar->wmi)] == 0,
420 WMI_TIMEOUT);
421 if (signal_pending(current)) {
422 ath6kl_err("cmd queue drain timeout\n");
423 up(&ar->sem);
424 return -EINTR;
425 }
426 }
427
Kevin Fang6981ffd2011-10-07 08:51:19 +0800428 if (sme->ie && (sme->ie_len > 0)) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530429 status = ath6kl_set_assoc_req_ies(vif, sme->ie, sme->ie_len);
Dan Carpenter743b4512011-11-18 17:09:32 +0300430 if (status) {
431 up(&ar->sem);
Kevin Fang6981ffd2011-10-07 08:51:19 +0800432 return status;
Dan Carpenter743b4512011-11-18 17:09:32 +0300433 }
Raja Mani542c5192011-11-15 14:14:56 +0530434 } else
435 ar->connect_ctrl_flags &= ~CONNECT_WPS_FLAG;
Kevin Fang6981ffd2011-10-07 08:51:19 +0800436
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530437 if (test_bit(CONNECTED, &vif->flags) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530438 vif->ssid_len == sme->ssid_len &&
439 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530440 vif->reconnect_flag = true;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530441 status = ath6kl_wmi_reconnect_cmd(ar->wmi, vif->fw_vif_idx,
442 vif->req_bssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530443 vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300444
445 up(&ar->sem);
446 if (status) {
447 ath6kl_err("wmi_reconnect_cmd failed\n");
448 return -EIO;
449 }
450 return 0;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530451 } else if (vif->ssid_len == sme->ssid_len &&
452 !memcmp(vif->ssid, sme->ssid, vif->ssid_len)) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530453 ath6kl_disconnect(vif);
Kalle Valobdcd8172011-07-18 00:22:30 +0300454 }
455
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530456 memset(vif->ssid, 0, sizeof(vif->ssid));
457 vif->ssid_len = sme->ssid_len;
458 memcpy(vif->ssid, sme->ssid, sme->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +0300459
460 if (sme->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530461 vif->ch_hint = sme->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +0300462
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530463 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300464 if (sme->bssid && !is_broadcast_ether_addr(sme->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530465 memcpy(vif->req_bssid, sme->bssid, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300466
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530467 ath6kl_set_wpa_version(vif, sme->crypto.wpa_versions);
Kalle Valobdcd8172011-07-18 00:22:30 +0300468
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530469 status = ath6kl_set_auth_type(vif, sme->auth_type);
Kalle Valobdcd8172011-07-18 00:22:30 +0300470 if (status) {
471 up(&ar->sem);
472 return status;
473 }
474
475 if (sme->crypto.n_ciphers_pairwise)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530476 ath6kl_set_cipher(vif, sme->crypto.ciphers_pairwise[0], true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300477 else
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530478 ath6kl_set_cipher(vif, 0, true);
Kalle Valobdcd8172011-07-18 00:22:30 +0300479
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530480 ath6kl_set_cipher(vif, sme->crypto.cipher_group, false);
Kalle Valobdcd8172011-07-18 00:22:30 +0300481
482 if (sme->crypto.n_akm_suites)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530483 ath6kl_set_key_mgmt(vif, sme->crypto.akm_suites[0]);
Kalle Valobdcd8172011-07-18 00:22:30 +0300484
485 if ((sme->key_len) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530486 (vif->auth_mode == NONE_AUTH) &&
487 (vif->prwise_crypto == WEP_CRYPT)) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300488 struct ath6kl_key *key = NULL;
489
490 if (sme->key_idx < WMI_MIN_KEY_INDEX ||
491 sme->key_idx > WMI_MAX_KEY_INDEX) {
492 ath6kl_err("key index %d out of bounds\n",
493 sme->key_idx);
494 up(&ar->sem);
495 return -ENOENT;
496 }
497
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530498 key = &vif->keys[sme->key_idx];
Kalle Valobdcd8172011-07-18 00:22:30 +0300499 key->key_len = sme->key_len;
500 memcpy(key->key, sme->key, key->key_len);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530501 key->cipher = vif->prwise_crypto;
502 vif->def_txkey_index = sme->key_idx;
Kalle Valobdcd8172011-07-18 00:22:30 +0300503
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530504 ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, sme->key_idx,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530505 vif->prwise_crypto,
Kalle Valobdcd8172011-07-18 00:22:30 +0300506 GROUP_USAGE | TX_USAGE,
507 key->key_len,
Jouni Malinenf4bb9a62011-11-02 23:45:55 +0200508 NULL, 0,
Kalle Valobdcd8172011-07-18 00:22:30 +0300509 key->key, KEY_OP_INIT_VAL, NULL,
510 NO_SYNC_WMIFLAG);
511 }
512
513 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530514 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530515 if (ath6kl_wmi_bssfilter_cmd(ar->wmi, vif->fw_vif_idx,
516 ALL_BSS_FILTER, 0) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300517 ath6kl_err("couldn't set bss filtering\n");
518 up(&ar->sem);
519 return -EIO;
520 }
521 }
522
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530523 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +0300524
525 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
526 "%s: connect called with authmode %d dot11 auth %d"
527 " PW crypto %d PW crypto len %d GRP crypto %d"
528 " GRP crypto len %d channel hint %u\n",
529 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530530 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
531 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530532 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +0300533
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530534 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530535 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530536 vif->dot11_auth_mode, vif->auth_mode,
537 vif->prwise_crypto,
538 vif->prwise_crypto_len,
539 vif->grp_crypto, vif->grp_crypto_len,
540 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +0530541 vif->req_bssid, vif->ch_hint,
Kalle Valobdcd8172011-07-18 00:22:30 +0300542 ar->connect_ctrl_flags);
543
544 up(&ar->sem);
545
546 if (status == -EINVAL) {
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530547 memset(vif->ssid, 0, sizeof(vif->ssid));
548 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300549 ath6kl_err("invalid request\n");
550 return -ENOENT;
551 } else if (status) {
552 ath6kl_err("ath6kl_wmi_connect_cmd failed\n");
553 return -EIO;
554 }
555
556 if ((!(ar->connect_ctrl_flags & CONNECT_DO_WPA_OFFLOAD)) &&
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530557 ((vif->auth_mode == WPA_PSK_AUTH)
558 || (vif->auth_mode == WPA2_PSK_AUTH))) {
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +0530559 mod_timer(&vif->disconnect_timer,
Kalle Valobdcd8172011-07-18 00:22:30 +0300560 jiffies + msecs_to_jiffies(DISCON_TIMER_INTVAL));
561 }
562
563 ar->connect_ctrl_flags &= ~CONNECT_DO_WPA_OFFLOAD;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530564 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300565
566 return 0;
567}
568
Raja Mani4eab6f42011-11-09 17:02:23 +0530569static int ath6kl_add_bss_if_needed(struct ath6kl_vif *vif,
570 enum network_type nw_type,
571 const u8 *bssid,
Jouni Malinen01cac472011-09-19 19:14:59 +0300572 struct ieee80211_channel *chan,
573 const u8 *beacon_ie, size_t beacon_ie_len)
574{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530575 struct ath6kl *ar = vif->ar;
Jouni Malinen01cac472011-09-19 19:14:59 +0300576 struct cfg80211_bss *bss;
Raja Mani4eab6f42011-11-09 17:02:23 +0530577 u16 cap_mask, cap_val;
Jouni Malinen01cac472011-09-19 19:14:59 +0300578 u8 *ie;
579
Raja Mani4eab6f42011-11-09 17:02:23 +0530580 if (nw_type & ADHOC_NETWORK) {
581 cap_mask = WLAN_CAPABILITY_IBSS;
582 cap_val = WLAN_CAPABILITY_IBSS;
583 } else {
584 cap_mask = WLAN_CAPABILITY_ESS;
585 cap_val = WLAN_CAPABILITY_ESS;
586 }
587
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530588 bss = cfg80211_get_bss(ar->wiphy, chan, bssid,
Raja Mani4eab6f42011-11-09 17:02:23 +0530589 vif->ssid, vif->ssid_len,
590 cap_mask, cap_val);
Jouni Malinen01cac472011-09-19 19:14:59 +0300591 if (bss == NULL) {
592 /*
593 * Since cfg80211 may not yet know about the BSS,
594 * generate a partial entry until the first BSS info
595 * event becomes available.
596 *
597 * Prepend SSID element since it is not included in the Beacon
598 * IEs from the target.
599 */
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530600 ie = kmalloc(2 + vif->ssid_len + beacon_ie_len, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300601 if (ie == NULL)
602 return -ENOMEM;
603 ie[0] = WLAN_EID_SSID;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530604 ie[1] = vif->ssid_len;
605 memcpy(ie + 2, vif->ssid, vif->ssid_len);
606 memcpy(ie + 2 + vif->ssid_len, beacon_ie, beacon_ie_len);
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530607 bss = cfg80211_inform_bss(ar->wiphy, chan,
Raja Mani4eab6f42011-11-09 17:02:23 +0530608 bssid, 0, cap_val, 100,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530609 ie, 2 + vif->ssid_len + beacon_ie_len,
Jouni Malinen01cac472011-09-19 19:14:59 +0300610 0, GFP_KERNEL);
611 if (bss)
Raja Mani4eab6f42011-11-09 17:02:23 +0530612 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "added bss %pM to "
613 "cfg80211\n", bssid);
Jouni Malinen01cac472011-09-19 19:14:59 +0300614 kfree(ie);
615 } else
616 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "cfg80211 already has a bss "
617 "entry\n");
618
619 if (bss == NULL)
620 return -ENOMEM;
621
622 cfg80211_put_bss(bss);
623
624 return 0;
625}
626
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530627void ath6kl_cfg80211_connect_event(struct ath6kl_vif *vif, u16 channel,
Kalle Valobdcd8172011-07-18 00:22:30 +0300628 u8 *bssid, u16 listen_intvl,
629 u16 beacon_intvl,
630 enum network_type nw_type,
631 u8 beacon_ie_len, u8 assoc_req_len,
632 u8 assoc_resp_len, u8 *assoc_info)
633{
Jouni Malinen01cac472011-09-19 19:14:59 +0300634 struct ieee80211_channel *chan;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530635 struct ath6kl *ar = vif->ar;
Kalle Valobdcd8172011-07-18 00:22:30 +0300636
637 /* capinfo + listen interval */
638 u8 assoc_req_ie_offset = sizeof(u16) + sizeof(u16);
639
640 /* capinfo + status code + associd */
641 u8 assoc_resp_ie_offset = sizeof(u16) + sizeof(u16) + sizeof(u16);
642
643 u8 *assoc_req_ie = assoc_info + beacon_ie_len + assoc_req_ie_offset;
644 u8 *assoc_resp_ie = assoc_info + beacon_ie_len + assoc_req_len +
645 assoc_resp_ie_offset;
646
647 assoc_req_len -= assoc_req_ie_offset;
648 assoc_resp_len -= assoc_resp_ie_offset;
649
Jouni Malinen32c10872011-09-19 19:15:07 +0300650 /*
651 * Store Beacon interval here; DTIM period will be available only once
652 * a Beacon frame from the AP is seen.
653 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530654 vif->assoc_bss_beacon_int = beacon_intvl;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530655 clear_bit(DTIM_PERIOD_AVAIL, &vif->flags);
Jouni Malinen32c10872011-09-19 19:15:07 +0300656
Kalle Valobdcd8172011-07-18 00:22:30 +0300657 if (nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530658 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300659 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
660 "%s: ath6k not in ibss mode\n", __func__);
661 return;
662 }
663 }
664
665 if (nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530666 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
667 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300668 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
669 "%s: ath6k not in station mode\n", __func__);
670 return;
671 }
672 }
673
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +0530674 chan = ieee80211_get_channel(ar->wiphy, (int) channel);
Kalle Valobdcd8172011-07-18 00:22:30 +0300675
Raja Mani4eab6f42011-11-09 17:02:23 +0530676 if (ath6kl_add_bss_if_needed(vif, nw_type, bssid, chan, assoc_info,
677 beacon_ie_len) < 0) {
678 ath6kl_err("could not add cfg80211 bss entry\n");
Kalle Valobdcd8172011-07-18 00:22:30 +0300679 return;
680 }
681
Raja Mani4eab6f42011-11-09 17:02:23 +0530682 if (nw_type & ADHOC_NETWORK) {
683 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "ad-hoc %s selected\n",
684 nw_type & ADHOC_CREATOR ? "creator" : "joiner");
685 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Jouni Malinen01cac472011-09-19 19:14:59 +0300686 return;
687 }
688
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530689 if (vif->sme_state == SME_CONNECTING) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300690 /* inform connect result to cfg80211 */
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530691 vif->sme_state = SME_CONNECTED;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530692 cfg80211_connect_result(vif->ndev, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300693 assoc_req_ie, assoc_req_len,
694 assoc_resp_ie, assoc_resp_len,
695 WLAN_STATUS_SUCCESS, GFP_KERNEL);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530696 } else if (vif->sme_state == SME_CONNECTED) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300697 /* inform roam event to cfg80211 */
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530698 cfg80211_roamed(vif->ndev, chan, bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +0300699 assoc_req_ie, assoc_req_len,
700 assoc_resp_ie, assoc_resp_len, GFP_KERNEL);
701 }
702}
703
704static int ath6kl_cfg80211_disconnect(struct wiphy *wiphy,
705 struct net_device *dev, u16 reason_code)
706{
707 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530708 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300709
710 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: reason=%u\n", __func__,
711 reason_code);
712
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530713 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300714 return -EIO;
715
716 if (test_bit(DESTROY_IN_PROGRESS, &ar->flag)) {
717 ath6kl_err("busy, destroy in progress\n");
718 return -EBUSY;
719 }
720
721 if (down_interruptible(&ar->sem)) {
722 ath6kl_err("busy, couldn't get access\n");
723 return -ERESTARTSYS;
724 }
725
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +0530726 vif->reconnect_flag = 0;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530727 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530728 memset(vif->ssid, 0, sizeof(vif->ssid));
729 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300730
731 if (!test_bit(SKIP_SCAN, &ar->flag))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +0530732 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +0300733
734 up(&ar->sem);
735
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530736 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan170826d2011-09-10 15:26:35 +0530737
Kalle Valobdcd8172011-07-18 00:22:30 +0300738 return 0;
739}
740
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530741void ath6kl_cfg80211_disconnect_event(struct ath6kl_vif *vif, u8 reason,
Kalle Valobdcd8172011-07-18 00:22:30 +0300742 u8 *bssid, u8 assoc_resp_len,
743 u8 *assoc_info, u16 proto_reason)
744{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530745 struct ath6kl *ar = vif->ar;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530746
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530747 if (vif->scan_req) {
748 cfg80211_scan_done(vif->scan_req, true);
749 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300750 }
751
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530752 if (vif->nw_type & ADHOC_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530753 if (vif->wdev.iftype != NL80211_IFTYPE_ADHOC) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300754 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
755 "%s: ath6k not in ibss mode\n", __func__);
756 return;
757 }
758 memset(bssid, 0, ETH_ALEN);
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530759 cfg80211_ibss_joined(vif->ndev, bssid, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300760 return;
761 }
762
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +0530763 if (vif->nw_type & INFRA_NETWORK) {
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +0530764 if (vif->wdev.iftype != NL80211_IFTYPE_STATION &&
765 vif->wdev.iftype != NL80211_IFTYPE_P2P_CLIENT) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300766 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
767 "%s: ath6k not in station mode\n", __func__);
768 return;
769 }
770 }
771
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530772 /*
773 * Send a disconnect command to target when a disconnect event is
774 * received with reason code other than 3 (DISCONNECT_CMD - disconnect
775 * request from host) to make the firmware stop trying to connect even
776 * after giving disconnect event. There will be one more disconnect
777 * event for this disconnect command with reason code DISCONNECT_CMD
778 * which will be notified to cfg80211.
779 */
Kalle Valobdcd8172011-07-18 00:22:30 +0300780
Vasanthakumar Thiagarajan1de547d2011-09-23 10:57:50 +0530781 if (reason != DISCONNECT_CMD) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530782 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +0300783 return;
784 }
785
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530786 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +0300787
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530788 if (vif->sme_state == SME_CONNECTING) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530789 cfg80211_connect_result(vif->ndev,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530790 bssid, NULL, 0,
791 NULL, 0,
792 WLAN_STATUS_UNSPECIFIED_FAILURE,
793 GFP_KERNEL);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530794 } else if (vif->sme_state == SME_CONNECTED) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530795 cfg80211_disconnected(vif->ndev, reason,
Vasanthakumar Thiagarajanac59a2b2011-09-10 15:26:34 +0530796 NULL, 0, GFP_KERNEL);
Kalle Valobdcd8172011-07-18 00:22:30 +0300797 }
798
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530799 vif->sme_state = SME_DISCONNECTED;
Kalle Valobdcd8172011-07-18 00:22:30 +0300800}
801
Kalle Valobdcd8172011-07-18 00:22:30 +0300802static int ath6kl_cfg80211_scan(struct wiphy *wiphy, struct net_device *ndev,
803 struct cfg80211_scan_request *request)
804{
805 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530806 struct ath6kl_vif *vif = netdev_priv(ndev);
Edward Lu1276c9e2011-08-30 21:58:00 +0300807 s8 n_channels = 0;
808 u16 *channels = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300809 int ret = 0;
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530810 u32 force_fg_scan = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +0300811
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530812 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300813 return -EIO;
814
815 if (!ar->usr_bss_filter) {
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530816 clear_bit(CLEAR_BSSFILTER_ON_BEACON, &vif->flags);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300817 ret = ath6kl_wmi_bssfilter_cmd(
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530818 ar->wmi, vif->fw_vif_idx,
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530819 (test_bit(CONNECTED, &vif->flags) ?
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300820 ALL_BUT_BSS_FILTER : ALL_BSS_FILTER), 0);
821 if (ret) {
Kalle Valobdcd8172011-07-18 00:22:30 +0300822 ath6kl_err("couldn't set bss filtering\n");
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300823 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +0300824 }
825 }
826
827 if (request->n_ssids && request->ssids[0].ssid_len) {
828 u8 i;
829
830 if (request->n_ssids > (MAX_PROBED_SSID_INDEX - 1))
831 request->n_ssids = MAX_PROBED_SSID_INDEX - 1;
832
833 for (i = 0; i < request->n_ssids; i++)
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530834 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
835 i + 1, SPECIFIC_SSID_FLAG,
Kalle Valobdcd8172011-07-18 00:22:30 +0300836 request->ssids[i].ssid_len,
837 request->ssids[i].ssid);
838 }
839
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300840 if (request->ie) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530841 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
842 WMI_FRAME_PROBE_REQ,
Jouni Malinenb84da8c2011-08-30 21:57:59 +0300843 request->ie, request->ie_len);
844 if (ret) {
845 ath6kl_err("failed to set Probe Request appie for "
846 "scan");
847 return ret;
848 }
849 }
850
Jouni Malinen11869be2011-09-02 20:07:06 +0300851 /*
852 * Scan only the requested channels if the request specifies a set of
853 * channels. If the list is longer than the target supports, do not
854 * configure the list and instead, scan all available channels.
855 */
856 if (request->n_channels > 0 &&
857 request->n_channels <= WMI_MAX_CHANNELS) {
Edward Lu1276c9e2011-08-30 21:58:00 +0300858 u8 i;
859
Jouni Malinen11869be2011-09-02 20:07:06 +0300860 n_channels = request->n_channels;
Edward Lu1276c9e2011-08-30 21:58:00 +0300861
862 channels = kzalloc(n_channels * sizeof(u16), GFP_KERNEL);
863 if (channels == NULL) {
864 ath6kl_warn("failed to set scan channels, "
865 "scan all channels");
866 n_channels = 0;
867 }
868
869 for (i = 0; i < n_channels; i++)
870 channels[i] = request->channels[i]->center_freq;
871 }
872
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530873 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajanf1f92172011-10-01 16:12:36 +0530874 force_fg_scan = 1;
875
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530876 ret = ath6kl_wmi_startscan_cmd(ar->wmi, vif->fw_vif_idx, WMI_LONG_SCAN,
877 force_fg_scan, false, 0, 0, n_channels,
878 channels);
Jouni Malinen1b1e6ee2011-08-30 21:58:10 +0300879 if (ret)
Kalle Valobdcd8172011-07-18 00:22:30 +0300880 ath6kl_err("wmi_startscan_cmd failed\n");
Jouni Malinen11869be2011-09-02 20:07:06 +0300881 else
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530882 vif->scan_req = request;
Kalle Valobdcd8172011-07-18 00:22:30 +0300883
Edward Lu1276c9e2011-08-30 21:58:00 +0300884 kfree(channels);
885
Kalle Valobdcd8172011-07-18 00:22:30 +0300886 return ret;
887}
888
Kalle Valo1c17d312011-11-01 08:43:56 +0200889void ath6kl_cfg80211_scan_complete_event(struct ath6kl_vif *vif, bool aborted)
Kalle Valobdcd8172011-07-18 00:22:30 +0300890{
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530891 struct ath6kl *ar = vif->ar;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300892 int i;
Kalle Valobdcd8172011-07-18 00:22:30 +0300893
Kalle Valo1c17d312011-11-01 08:43:56 +0200894 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: status%s\n", __func__,
895 aborted ? " aborted" : "");
Kalle Valobdcd8172011-07-18 00:22:30 +0300896
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530897 if (!vif->scan_req)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300898 return;
Kalle Valobdcd8172011-07-18 00:22:30 +0300899
Kalle Valo1c17d312011-11-01 08:43:56 +0200900 if (aborted)
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300901 goto out;
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300902
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530903 if (vif->scan_req->n_ssids && vif->scan_req->ssids[0].ssid_len) {
904 for (i = 0; i < vif->scan_req->n_ssids; i++) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +0530905 ath6kl_wmi_probedssid_cmd(ar->wmi, vif->fw_vif_idx,
906 i + 1, DISABLE_SSID_FLAG,
Kalle Valo6fd1eac2011-07-21 10:22:50 +0300907 0, NULL);
908 }
909 }
910
911out:
Kalle Valocb938212011-10-27 18:47:46 +0300912 cfg80211_scan_done(vif->scan_req, aborted);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +0530913 vif->scan_req = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +0300914}
915
916static int ath6kl_cfg80211_add_key(struct wiphy *wiphy, struct net_device *ndev,
917 u8 key_index, bool pairwise,
918 const u8 *mac_addr,
919 struct key_params *params)
920{
921 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +0530922 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +0300923 struct ath6kl_key *key = NULL;
924 u8 key_usage;
925 u8 key_type;
Kalle Valobdcd8172011-07-18 00:22:30 +0300926
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +0530927 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +0300928 return -EIO;
929
Jouni Malinen837cb972011-10-11 17:31:57 +0300930 if (params->cipher == CCKM_KRK_CIPHER_SUITE) {
931 if (params->key_len != WMI_KRK_LEN)
932 return -EINVAL;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +0530933 return ath6kl_wmi_add_krk_cmd(ar->wmi, vif->fw_vif_idx,
934 params->key);
Jouni Malinen837cb972011-10-11 17:31:57 +0300935 }
936
Kalle Valobdcd8172011-07-18 00:22:30 +0300937 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
938 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
939 "%s: key index %d out of bounds\n", __func__,
940 key_index);
941 return -ENOENT;
942 }
943
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +0530944 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +0300945 memset(key, 0, sizeof(struct ath6kl_key));
946
947 if (pairwise)
948 key_usage = PAIRWISE_USAGE;
949 else
950 key_usage = GROUP_USAGE;
951
952 if (params) {
Dai Shuibing5e070212011-11-03 11:39:37 +0200953 int seq_len = params->seq_len;
954 if (params->cipher == WLAN_CIPHER_SUITE_SMS4 &&
955 seq_len > ATH6KL_KEY_SEQ_LEN) {
956 /* Only first half of the WPI PN is configured */
957 seq_len = ATH6KL_KEY_SEQ_LEN;
958 }
Kalle Valobdcd8172011-07-18 00:22:30 +0300959 if (params->key_len > WLAN_MAX_KEY_LEN ||
Dai Shuibing5e070212011-11-03 11:39:37 +0200960 seq_len > sizeof(key->seq))
Kalle Valobdcd8172011-07-18 00:22:30 +0300961 return -EINVAL;
962
963 key->key_len = params->key_len;
964 memcpy(key->key, params->key, key->key_len);
Dai Shuibing5e070212011-11-03 11:39:37 +0200965 key->seq_len = seq_len;
Kalle Valobdcd8172011-07-18 00:22:30 +0300966 memcpy(key->seq, params->seq, key->seq_len);
967 key->cipher = params->cipher;
968 }
969
970 switch (key->cipher) {
971 case WLAN_CIPHER_SUITE_WEP40:
972 case WLAN_CIPHER_SUITE_WEP104:
973 key_type = WEP_CRYPT;
974 break;
975
976 case WLAN_CIPHER_SUITE_TKIP:
977 key_type = TKIP_CRYPT;
978 break;
979
980 case WLAN_CIPHER_SUITE_CCMP:
981 key_type = AES_CRYPT;
982 break;
Dai Shuibing5e070212011-11-03 11:39:37 +0200983 case WLAN_CIPHER_SUITE_SMS4:
984 key_type = WAPI_CRYPT;
985 break;
Kalle Valobdcd8172011-07-18 00:22:30 +0300986
987 default:
988 return -ENOTSUPP;
989 }
990
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +0530991 if (((vif->auth_mode == WPA_PSK_AUTH)
992 || (vif->auth_mode == WPA2_PSK_AUTH))
Kalle Valobdcd8172011-07-18 00:22:30 +0300993 && (key_usage & GROUP_USAGE))
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +0530994 del_timer(&vif->disconnect_timer);
Kalle Valobdcd8172011-07-18 00:22:30 +0300995
996 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
997 "%s: index %d, key_len %d, key_type 0x%x, key_usage 0x%x, seq_len %d\n",
998 __func__, key_index, key->key_len, key_type,
999 key_usage, key->seq_len);
1000
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301001 if (vif->nw_type == AP_NETWORK && !pairwise &&
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001002 (key_type == TKIP_CRYPT || key_type == AES_CRYPT) && params) {
1003 ar->ap_mode_bkey.valid = true;
1004 ar->ap_mode_bkey.key_index = key_index;
1005 ar->ap_mode_bkey.key_type = key_type;
1006 ar->ap_mode_bkey.key_len = key->key_len;
1007 memcpy(ar->ap_mode_bkey.key, key->key, key->key_len);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301008 if (!test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001009 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay initial group "
1010 "key configuration until AP mode has been "
1011 "started\n");
1012 /*
1013 * The key will be set in ath6kl_connect_ap_mode() once
1014 * the connected event is received from the target.
1015 */
1016 return 0;
1017 }
1018 }
1019
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301020 if (vif->next_mode == AP_NETWORK && key_type == WEP_CRYPT &&
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301021 !test_bit(CONNECTED, &vif->flags)) {
Jouni Malinen151411e2011-09-15 15:10:16 +03001022 /*
1023 * Store the key locally so that it can be re-configured after
1024 * the AP mode has properly started
1025 * (ath6kl_install_statioc_wep_keys).
1026 */
1027 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "Delay WEP key configuration "
1028 "until AP mode has been started\n");
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301029 vif->wep_key_list[key_index].key_len = key->key_len;
1030 memcpy(vif->wep_key_list[key_index].key, key->key,
1031 key->key_len);
Jouni Malinen151411e2011-09-15 15:10:16 +03001032 return 0;
1033 }
1034
Vasanthakumar Thiagarajan7cefa442011-11-11 20:33:00 +05301035 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx, key_index,
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001036 key_type, key_usage, key->key_len,
1037 key->seq, key->seq_len, key->key,
1038 KEY_OP_INIT_VAL,
1039 (u8 *) mac_addr, SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001040}
1041
1042static int ath6kl_cfg80211_del_key(struct wiphy *wiphy, struct net_device *ndev,
1043 u8 key_index, bool pairwise,
1044 const u8 *mac_addr)
1045{
1046 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301047 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001048
1049 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1050
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301051 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001052 return -EIO;
1053
1054 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1055 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1056 "%s: key index %d out of bounds\n", __func__,
1057 key_index);
1058 return -ENOENT;
1059 }
1060
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301061 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001062 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1063 "%s: index %d is empty\n", __func__, key_index);
1064 return 0;
1065 }
1066
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301067 vif->keys[key_index].key_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001068
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301069 return ath6kl_wmi_deletekey_cmd(ar->wmi, vif->fw_vif_idx, key_index);
Kalle Valobdcd8172011-07-18 00:22:30 +03001070}
1071
1072static int ath6kl_cfg80211_get_key(struct wiphy *wiphy, struct net_device *ndev,
1073 u8 key_index, bool pairwise,
1074 const u8 *mac_addr, void *cookie,
1075 void (*callback) (void *cookie,
1076 struct key_params *))
1077{
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301078 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001079 struct ath6kl_key *key = NULL;
1080 struct key_params params;
1081
1082 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1083
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301084 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001085 return -EIO;
1086
1087 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1088 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1089 "%s: key index %d out of bounds\n", __func__,
1090 key_index);
1091 return -ENOENT;
1092 }
1093
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301094 key = &vif->keys[key_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001095 memset(&params, 0, sizeof(params));
1096 params.cipher = key->cipher;
1097 params.key_len = key->key_len;
1098 params.seq_len = key->seq_len;
1099 params.seq = key->seq;
1100 params.key = key->key;
1101
1102 callback(cookie, &params);
1103
1104 return key->key_len ? 0 : -ENOENT;
1105}
1106
1107static int ath6kl_cfg80211_set_default_key(struct wiphy *wiphy,
1108 struct net_device *ndev,
1109 u8 key_index, bool unicast,
1110 bool multicast)
1111{
1112 struct ath6kl *ar = (struct ath6kl *)ath6kl_priv(ndev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301113 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001114 struct ath6kl_key *key = NULL;
Kalle Valobdcd8172011-07-18 00:22:30 +03001115 u8 key_usage;
Edward Lu229ed6b2011-08-30 21:58:07 +03001116 enum crypto_type key_type = NONE_CRYPT;
Kalle Valobdcd8172011-07-18 00:22:30 +03001117
1118 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: index %d\n", __func__, key_index);
1119
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301120 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001121 return -EIO;
1122
1123 if (key_index < WMI_MIN_KEY_INDEX || key_index > WMI_MAX_KEY_INDEX) {
1124 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1125 "%s: key index %d out of bounds\n",
1126 __func__, key_index);
1127 return -ENOENT;
1128 }
1129
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301130 if (!vif->keys[key_index].key_len) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001131 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: invalid key index %d\n",
1132 __func__, key_index);
1133 return -EINVAL;
1134 }
1135
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301136 vif->def_txkey_index = key_index;
Vasanthakumar Thiagarajan6f2a73f2011-10-25 19:34:06 +05301137 key = &vif->keys[vif->def_txkey_index];
Kalle Valobdcd8172011-07-18 00:22:30 +03001138 key_usage = GROUP_USAGE;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301139 if (vif->prwise_crypto == WEP_CRYPT)
Kalle Valobdcd8172011-07-18 00:22:30 +03001140 key_usage |= TX_USAGE;
Edward Lu229ed6b2011-08-30 21:58:07 +03001141 if (unicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301142 key_type = vif->prwise_crypto;
Edward Lu229ed6b2011-08-30 21:58:07 +03001143 if (multicast)
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301144 key_type = vif->grp_crypto;
Kalle Valobdcd8172011-07-18 00:22:30 +03001145
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301146 if (vif->next_mode == AP_NETWORK && !test_bit(CONNECTED, &vif->flags))
Jouni Malinen9a5b1312011-08-30 21:57:52 +03001147 return 0; /* Delay until AP mode has been started */
1148
Jouni Malinenf3e61ec2011-11-02 23:46:47 +02001149 return ath6kl_wmi_addkey_cmd(ar->wmi, vif->fw_vif_idx,
1150 vif->def_txkey_index,
1151 key_type, key_usage,
1152 key->key_len, key->seq, key->seq_len,
1153 key->key,
1154 KEY_OP_INIT_VAL, NULL,
1155 SYNC_BOTH_WMIFLAG);
Kalle Valobdcd8172011-07-18 00:22:30 +03001156}
1157
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301158void ath6kl_cfg80211_tkip_micerr_event(struct ath6kl_vif *vif, u8 keyid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001159 bool ismcast)
1160{
1161 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1162 "%s: keyid %d, ismcast %d\n", __func__, keyid, ismcast);
1163
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301164 cfg80211_michael_mic_failure(vif->ndev, vif->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001165 (ismcast ? NL80211_KEYTYPE_GROUP :
1166 NL80211_KEYTYPE_PAIRWISE), keyid, NULL,
1167 GFP_KERNEL);
1168}
1169
1170static int ath6kl_cfg80211_set_wiphy_params(struct wiphy *wiphy, u32 changed)
1171{
1172 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301173 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001174 int ret;
1175
1176 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: changed 0x%x\n", __func__,
1177 changed);
1178
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301179 vif = ath6kl_vif_first(ar);
1180 if (!vif)
1181 return -EIO;
1182
1183 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001184 return -EIO;
1185
1186 if (changed & WIPHY_PARAM_RTS_THRESHOLD) {
1187 ret = ath6kl_wmi_set_rts_cmd(ar->wmi, wiphy->rts_threshold);
1188 if (ret != 0) {
1189 ath6kl_err("ath6kl_wmi_set_rts_cmd failed\n");
1190 return -EIO;
1191 }
1192 }
1193
1194 return 0;
1195}
1196
1197/*
1198 * The type nl80211_tx_power_setting replaces the following
1199 * data type from 2.6.36 onwards
1200*/
1201static int ath6kl_cfg80211_set_txpower(struct wiphy *wiphy,
1202 enum nl80211_tx_power_setting type,
1203 int dbm)
1204{
1205 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301206 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001207 u8 ath6kl_dbm;
1208
1209 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x, dbm %d\n", __func__,
1210 type, dbm);
1211
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301212 vif = ath6kl_vif_first(ar);
1213 if (!vif)
1214 return -EIO;
1215
1216 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001217 return -EIO;
1218
1219 switch (type) {
1220 case NL80211_TX_POWER_AUTOMATIC:
1221 return 0;
1222 case NL80211_TX_POWER_LIMITED:
1223 ar->tx_pwr = ath6kl_dbm = dbm;
1224 break;
1225 default:
1226 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type 0x%x not supported\n",
1227 __func__, type);
1228 return -EOPNOTSUPP;
1229 }
1230
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301231 ath6kl_wmi_set_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx, ath6kl_dbm);
Kalle Valobdcd8172011-07-18 00:22:30 +03001232
1233 return 0;
1234}
1235
1236static int ath6kl_cfg80211_get_txpower(struct wiphy *wiphy, int *dbm)
1237{
1238 struct ath6kl *ar = (struct ath6kl *)wiphy_priv(wiphy);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301239 struct ath6kl_vif *vif;
Kalle Valobdcd8172011-07-18 00:22:30 +03001240
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301241 vif = ath6kl_vif_first(ar);
1242 if (!vif)
1243 return -EIO;
1244
1245 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001246 return -EIO;
1247
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301248 if (test_bit(CONNECTED, &vif->flags)) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001249 ar->tx_pwr = 0;
1250
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301251 if (ath6kl_wmi_get_tx_pwr_cmd(ar->wmi, vif->fw_vif_idx) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001252 ath6kl_err("ath6kl_wmi_get_tx_pwr_cmd failed\n");
1253 return -EIO;
1254 }
1255
1256 wait_event_interruptible_timeout(ar->event_wq, ar->tx_pwr != 0,
1257 5 * HZ);
1258
1259 if (signal_pending(current)) {
1260 ath6kl_err("target did not respond\n");
1261 return -EINTR;
1262 }
1263 }
1264
1265 *dbm = ar->tx_pwr;
1266 return 0;
1267}
1268
1269static int ath6kl_cfg80211_set_power_mgmt(struct wiphy *wiphy,
1270 struct net_device *dev,
1271 bool pmgmt, int timeout)
1272{
1273 struct ath6kl *ar = ath6kl_priv(dev);
1274 struct wmi_power_mode_cmd mode;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301275 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001276
1277 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: pmgmt %d, timeout %d\n",
1278 __func__, pmgmt, timeout);
1279
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301280 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001281 return -EIO;
1282
1283 if (pmgmt) {
1284 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: max perf\n", __func__);
1285 mode.pwr_mode = REC_POWER;
1286 } else {
1287 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: rec power\n", __func__);
1288 mode.pwr_mode = MAX_PERF_POWER;
1289 }
1290
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301291 if (ath6kl_wmi_powermode_cmd(ar->wmi, vif->fw_vif_idx,
1292 mode.pwr_mode) != 0) {
Kalle Valobdcd8172011-07-18 00:22:30 +03001293 ath6kl_err("wmi_powermode_cmd failed\n");
1294 return -EIO;
1295 }
1296
1297 return 0;
1298}
1299
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301300static struct net_device *ath6kl_cfg80211_add_iface(struct wiphy *wiphy,
1301 char *name,
1302 enum nl80211_iftype type,
1303 u32 *flags,
1304 struct vif_params *params)
1305{
1306 struct ath6kl *ar = wiphy_priv(wiphy);
1307 struct net_device *ndev;
1308 u8 if_idx, nw_type;
1309
Kalle Valo71f96ee2011-11-14 19:31:30 +02001310 if (ar->num_vif == ar->vif_max) {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301311 ath6kl_err("Reached maximum number of supported vif\n");
1312 return ERR_PTR(-EINVAL);
1313 }
1314
1315 if (!ath6kl_is_valid_iftype(ar, type, &if_idx, &nw_type)) {
1316 ath6kl_err("Not a supported interface type\n");
1317 return ERR_PTR(-EINVAL);
1318 }
1319
1320 ndev = ath6kl_interface_add(ar, name, type, if_idx, nw_type);
1321 if (!ndev)
1322 return ERR_PTR(-ENOMEM);
1323
1324 ar->num_vif++;
1325
1326 return ndev;
1327}
1328
1329static int ath6kl_cfg80211_del_iface(struct wiphy *wiphy,
1330 struct net_device *ndev)
1331{
1332 struct ath6kl *ar = wiphy_priv(wiphy);
1333 struct ath6kl_vif *vif = netdev_priv(ndev);
1334
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301335 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301336 list_del(&vif->list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05301337 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05301338
1339 ath6kl_cleanup_vif(vif, test_bit(WMI_READY, &ar->flag));
1340
1341 ath6kl_deinit_if_data(vif);
1342
1343 return 0;
1344}
1345
Kalle Valobdcd8172011-07-18 00:22:30 +03001346static int ath6kl_cfg80211_change_iface(struct wiphy *wiphy,
1347 struct net_device *ndev,
1348 enum nl80211_iftype type, u32 *flags,
1349 struct vif_params *params)
1350{
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301351 struct ath6kl_vif *vif = netdev_priv(ndev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001352
1353 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: type %u\n", __func__, type);
1354
Kalle Valobdcd8172011-07-18 00:22:30 +03001355 switch (type) {
1356 case NL80211_IFTYPE_STATION:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301357 vif->next_mode = INFRA_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001358 break;
1359 case NL80211_IFTYPE_ADHOC:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301360 vif->next_mode = ADHOC_NETWORK;
Kalle Valobdcd8172011-07-18 00:22:30 +03001361 break;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001362 case NL80211_IFTYPE_AP:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301363 vif->next_mode = AP_NETWORK;
Jouni Malinen6e4604c2011-09-05 17:38:46 +03001364 break;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001365 case NL80211_IFTYPE_P2P_CLIENT:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301366 vif->next_mode = INFRA_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001367 break;
1368 case NL80211_IFTYPE_P2P_GO:
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301369 vif->next_mode = AP_NETWORK;
Jouni Malinen6b5e5d22011-08-30 21:58:05 +03001370 break;
Kalle Valobdcd8172011-07-18 00:22:30 +03001371 default:
1372 ath6kl_err("invalid interface type %u\n", type);
1373 return -EOPNOTSUPP;
1374 }
1375
Vasanthakumar Thiagarajan551959d2011-10-25 19:34:26 +05301376 vif->wdev.iftype = type;
Kalle Valobdcd8172011-07-18 00:22:30 +03001377
1378 return 0;
1379}
1380
1381static int ath6kl_cfg80211_join_ibss(struct wiphy *wiphy,
1382 struct net_device *dev,
1383 struct cfg80211_ibss_params *ibss_param)
1384{
1385 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301386 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001387 int status;
1388
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301389 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001390 return -EIO;
1391
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301392 vif->ssid_len = ibss_param->ssid_len;
1393 memcpy(vif->ssid, ibss_param->ssid, vif->ssid_len);
Kalle Valobdcd8172011-07-18 00:22:30 +03001394
1395 if (ibss_param->channel)
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301396 vif->ch_hint = ibss_param->channel->center_freq;
Kalle Valobdcd8172011-07-18 00:22:30 +03001397
1398 if (ibss_param->channel_fixed) {
1399 /*
1400 * TODO: channel_fixed: The channel should be fixed, do not
1401 * search for IBSSs to join on other channels. Target
1402 * firmware does not support this feature, needs to be
1403 * updated.
1404 */
1405 return -EOPNOTSUPP;
1406 }
1407
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301408 memset(vif->req_bssid, 0, sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001409 if (ibss_param->bssid && !is_broadcast_ether_addr(ibss_param->bssid))
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301410 memcpy(vif->req_bssid, ibss_param->bssid,
1411 sizeof(vif->req_bssid));
Kalle Valobdcd8172011-07-18 00:22:30 +03001412
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301413 ath6kl_set_wpa_version(vif, 0);
Kalle Valobdcd8172011-07-18 00:22:30 +03001414
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301415 status = ath6kl_set_auth_type(vif, NL80211_AUTHTYPE_OPEN_SYSTEM);
Kalle Valobdcd8172011-07-18 00:22:30 +03001416 if (status)
1417 return status;
1418
1419 if (ibss_param->privacy) {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301420 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, true);
1421 ath6kl_set_cipher(vif, WLAN_CIPHER_SUITE_WEP40, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001422 } else {
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301423 ath6kl_set_cipher(vif, 0, true);
1424 ath6kl_set_cipher(vif, 0, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001425 }
1426
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301427 vif->nw_type = vif->next_mode;
Kalle Valobdcd8172011-07-18 00:22:30 +03001428
1429 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1430 "%s: connect called with authmode %d dot11 auth %d"
1431 " PW crypto %d PW crypto len %d GRP crypto %d"
1432 " GRP crypto len %d channel hint %u\n",
1433 __func__,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301434 vif->auth_mode, vif->dot11_auth_mode, vif->prwise_crypto,
1435 vif->prwise_crypto_len, vif->grp_crypto,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301436 vif->grp_crypto_len, vif->ch_hint);
Kalle Valobdcd8172011-07-18 00:22:30 +03001437
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301438 status = ath6kl_wmi_connect_cmd(ar->wmi, vif->fw_vif_idx, vif->nw_type,
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301439 vif->dot11_auth_mode, vif->auth_mode,
1440 vif->prwise_crypto,
1441 vif->prwise_crypto_len,
1442 vif->grp_crypto, vif->grp_crypto_len,
1443 vif->ssid_len, vif->ssid,
Vasanthakumar Thiagarajanf74bac52011-10-25 19:34:05 +05301444 vif->req_bssid, vif->ch_hint,
Kalle Valobdcd8172011-07-18 00:22:30 +03001445 ar->connect_ctrl_flags);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301446 set_bit(CONNECT_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001447
1448 return 0;
1449}
1450
1451static int ath6kl_cfg80211_leave_ibss(struct wiphy *wiphy,
1452 struct net_device *dev)
1453{
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301454 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001455
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301456 if (!ath6kl_cfg80211_ready(vif))
Kalle Valobdcd8172011-07-18 00:22:30 +03001457 return -EIO;
1458
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05301459 ath6kl_disconnect(vif);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05301460 memset(vif->ssid, 0, sizeof(vif->ssid));
1461 vif->ssid_len = 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03001462
1463 return 0;
1464}
1465
1466static const u32 cipher_suites[] = {
1467 WLAN_CIPHER_SUITE_WEP40,
1468 WLAN_CIPHER_SUITE_WEP104,
1469 WLAN_CIPHER_SUITE_TKIP,
1470 WLAN_CIPHER_SUITE_CCMP,
Jouni Malinen837cb972011-10-11 17:31:57 +03001471 CCKM_KRK_CIPHER_SUITE,
Dai Shuibing5e070212011-11-03 11:39:37 +02001472 WLAN_CIPHER_SUITE_SMS4,
Kalle Valobdcd8172011-07-18 00:22:30 +03001473};
1474
1475static bool is_rate_legacy(s32 rate)
1476{
1477 static const s32 legacy[] = { 1000, 2000, 5500, 11000,
1478 6000, 9000, 12000, 18000, 24000,
1479 36000, 48000, 54000
1480 };
1481 u8 i;
1482
1483 for (i = 0; i < ARRAY_SIZE(legacy); i++)
1484 if (rate == legacy[i])
1485 return true;
1486
1487 return false;
1488}
1489
1490static bool is_rate_ht20(s32 rate, u8 *mcs, bool *sgi)
1491{
1492 static const s32 ht20[] = { 6500, 13000, 19500, 26000, 39000,
1493 52000, 58500, 65000, 72200
1494 };
1495 u8 i;
1496
1497 for (i = 0; i < ARRAY_SIZE(ht20); i++) {
1498 if (rate == ht20[i]) {
1499 if (i == ARRAY_SIZE(ht20) - 1)
1500 /* last rate uses sgi */
1501 *sgi = true;
1502 else
1503 *sgi = false;
1504
1505 *mcs = i;
1506 return true;
1507 }
1508 }
1509 return false;
1510}
1511
1512static bool is_rate_ht40(s32 rate, u8 *mcs, bool *sgi)
1513{
1514 static const s32 ht40[] = { 13500, 27000, 40500, 54000,
1515 81000, 108000, 121500, 135000,
1516 150000
1517 };
1518 u8 i;
1519
1520 for (i = 0; i < ARRAY_SIZE(ht40); i++) {
1521 if (rate == ht40[i]) {
1522 if (i == ARRAY_SIZE(ht40) - 1)
1523 /* last rate uses sgi */
1524 *sgi = true;
1525 else
1526 *sgi = false;
1527
1528 *mcs = i;
1529 return true;
1530 }
1531 }
1532
1533 return false;
1534}
1535
1536static int ath6kl_get_station(struct wiphy *wiphy, struct net_device *dev,
1537 u8 *mac, struct station_info *sinfo)
1538{
1539 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301540 struct ath6kl_vif *vif = netdev_priv(dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03001541 long left;
1542 bool sgi;
1543 s32 rate;
1544 int ret;
1545 u8 mcs;
1546
Vasanthakumar Thiagarajan8c8b65e2011-10-25 19:34:04 +05301547 if (memcmp(mac, vif->bssid, ETH_ALEN) != 0)
Kalle Valobdcd8172011-07-18 00:22:30 +03001548 return -ENOENT;
1549
1550 if (down_interruptible(&ar->sem))
1551 return -EBUSY;
1552
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301553 set_bit(STATS_UPDATE_PEND, &vif->flags);
Kalle Valobdcd8172011-07-18 00:22:30 +03001554
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301555 ret = ath6kl_wmi_get_stats_cmd(ar->wmi, vif->fw_vif_idx);
Kalle Valobdcd8172011-07-18 00:22:30 +03001556
1557 if (ret != 0) {
1558 up(&ar->sem);
1559 return -EIO;
1560 }
1561
1562 left = wait_event_interruptible_timeout(ar->event_wq,
1563 !test_bit(STATS_UPDATE_PEND,
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301564 &vif->flags),
Kalle Valobdcd8172011-07-18 00:22:30 +03001565 WMI_TIMEOUT);
1566
1567 up(&ar->sem);
1568
1569 if (left == 0)
1570 return -ETIMEDOUT;
1571 else if (left < 0)
1572 return left;
1573
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301574 if (vif->target_stats.rx_byte) {
1575 sinfo->rx_bytes = vif->target_stats.rx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001576 sinfo->filled |= STATION_INFO_RX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301577 sinfo->rx_packets = vif->target_stats.rx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001578 sinfo->filled |= STATION_INFO_RX_PACKETS;
1579 }
1580
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301581 if (vif->target_stats.tx_byte) {
1582 sinfo->tx_bytes = vif->target_stats.tx_byte;
Kalle Valobdcd8172011-07-18 00:22:30 +03001583 sinfo->filled |= STATION_INFO_TX_BYTES;
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301584 sinfo->tx_packets = vif->target_stats.tx_pkt;
Kalle Valobdcd8172011-07-18 00:22:30 +03001585 sinfo->filled |= STATION_INFO_TX_PACKETS;
1586 }
1587
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301588 sinfo->signal = vif->target_stats.cs_rssi;
Kalle Valobdcd8172011-07-18 00:22:30 +03001589 sinfo->filled |= STATION_INFO_SIGNAL;
1590
Vasanthakumar Thiagarajanb95907a2011-10-25 19:34:11 +05301591 rate = vif->target_stats.tx_ucast_rate;
Kalle Valobdcd8172011-07-18 00:22:30 +03001592
1593 if (is_rate_legacy(rate)) {
1594 sinfo->txrate.legacy = rate / 100;
1595 } else if (is_rate_ht20(rate, &mcs, &sgi)) {
1596 if (sgi) {
1597 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1598 sinfo->txrate.mcs = mcs - 1;
1599 } else {
1600 sinfo->txrate.mcs = mcs;
1601 }
1602
1603 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1604 } else if (is_rate_ht40(rate, &mcs, &sgi)) {
1605 if (sgi) {
1606 sinfo->txrate.flags |= RATE_INFO_FLAGS_SHORT_GI;
1607 sinfo->txrate.mcs = mcs - 1;
1608 } else {
1609 sinfo->txrate.mcs = mcs;
1610 }
1611
1612 sinfo->txrate.flags |= RATE_INFO_FLAGS_40_MHZ_WIDTH;
1613 sinfo->txrate.flags |= RATE_INFO_FLAGS_MCS;
1614 } else {
Kalle Valo9a730832011-09-27 23:33:28 +03001615 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG,
1616 "invalid rate from stats: %d\n", rate);
1617 ath6kl_debug_war(ar, ATH6KL_WAR_INVALID_RATE);
Kalle Valobdcd8172011-07-18 00:22:30 +03001618 return 0;
1619 }
1620
1621 sinfo->filled |= STATION_INFO_TX_BITRATE;
1622
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301623 if (test_bit(CONNECTED, &vif->flags) &&
1624 test_bit(DTIM_PERIOD_AVAIL, &vif->flags) &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05301625 vif->nw_type == INFRA_NETWORK) {
Jouni Malinen32c10872011-09-19 19:15:07 +03001626 sinfo->filled |= STATION_INFO_BSS_PARAM;
1627 sinfo->bss_param.flags = 0;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301628 sinfo->bss_param.dtim_period = vif->assoc_bss_dtim_period;
1629 sinfo->bss_param.beacon_interval = vif->assoc_bss_beacon_int;
Jouni Malinen32c10872011-09-19 19:15:07 +03001630 }
1631
Kalle Valobdcd8172011-07-18 00:22:30 +03001632 return 0;
1633}
1634
1635static int ath6kl_set_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1636 struct cfg80211_pmksa *pmksa)
1637{
1638 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301639 struct ath6kl_vif *vif = netdev_priv(netdev);
1640
1641 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001642 pmksa->pmkid, true);
1643}
1644
1645static int ath6kl_del_pmksa(struct wiphy *wiphy, struct net_device *netdev,
1646 struct cfg80211_pmksa *pmksa)
1647{
1648 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301649 struct ath6kl_vif *vif = netdev_priv(netdev);
1650
1651 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx, pmksa->bssid,
Kalle Valobdcd8172011-07-18 00:22:30 +03001652 pmksa->pmkid, false);
1653}
1654
1655static int ath6kl_flush_pmksa(struct wiphy *wiphy, struct net_device *netdev)
1656{
1657 struct ath6kl *ar = ath6kl_priv(netdev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05301658 struct ath6kl_vif *vif = netdev_priv(netdev);
1659
1660 if (test_bit(CONNECTED, &vif->flags))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301661 return ath6kl_wmi_setpmkid_cmd(ar->wmi, vif->fw_vif_idx,
1662 vif->bssid, NULL, false);
Kalle Valobdcd8172011-07-18 00:22:30 +03001663 return 0;
1664}
1665
Raja Mani6cb3c712011-11-07 22:52:45 +02001666static int ath6kl_wow_suspend(struct ath6kl *ar, struct cfg80211_wowlan *wow)
1667{
1668 struct ath6kl_vif *vif;
1669 int ret, pos, left;
1670 u32 filter = 0;
1671 u16 i;
1672 u8 mask[WOW_MASK_SIZE];
1673
1674 vif = ath6kl_vif_first(ar);
1675 if (!vif)
1676 return -EIO;
1677
1678 if (!ath6kl_cfg80211_ready(vif))
1679 return -EIO;
1680
1681 if (!test_bit(CONNECTED, &vif->flags))
1682 return -EINVAL;
1683
1684 /* Clear existing WOW patterns */
1685 for (i = 0; i < WOW_MAX_FILTERS_PER_LIST; i++)
1686 ath6kl_wmi_del_wow_pattern_cmd(ar->wmi, vif->fw_vif_idx,
1687 WOW_LIST_ID, i);
1688 /* Configure new WOW patterns */
1689 for (i = 0; i < wow->n_patterns; i++) {
1690
1691 /*
1692 * Convert given nl80211 specific mask value to equivalent
1693 * driver specific mask value and send it to the chip along
1694 * with patterns. For example, If the mask value defined in
1695 * struct cfg80211_wowlan is 0xA (equivalent binary is 1010),
1696 * then equivalent driver specific mask value is
1697 * "0xFF 0x00 0xFF 0x00".
1698 */
1699 memset(&mask, 0, sizeof(mask));
1700 for (pos = 0; pos < wow->patterns[i].pattern_len; pos++) {
1701 if (wow->patterns[i].mask[pos / 8] & (0x1 << (pos % 8)))
1702 mask[pos] = 0xFF;
1703 }
1704 /*
1705 * Note: Pattern's offset is not passed as part of wowlan
1706 * parameter from CFG layer. So it's always passed as ZERO
1707 * to the firmware. It means, given WOW patterns are always
1708 * matched from the first byte of received pkt in the firmware.
1709 */
1710 ret = ath6kl_wmi_add_wow_pattern_cmd(ar->wmi,
1711 vif->fw_vif_idx, WOW_LIST_ID,
1712 wow->patterns[i].pattern_len,
1713 0 /* pattern offset */,
1714 wow->patterns[i].pattern, mask);
1715 if (ret)
1716 return ret;
1717 }
1718
1719 if (wow->disconnect)
1720 filter |= WOW_FILTER_OPTION_NWK_DISASSOC;
1721
1722 if (wow->magic_pkt)
1723 filter |= WOW_FILTER_OPTION_MAGIC_PACKET;
1724
1725 if (wow->gtk_rekey_failure)
1726 filter |= WOW_FILTER_OPTION_GTK_ERROR;
1727
1728 if (wow->eap_identity_req)
1729 filter |= WOW_FILTER_OPTION_EAP_REQ;
1730
1731 if (wow->four_way_handshake)
1732 filter |= WOW_FILTER_OPTION_8021X_4WAYHS;
1733
1734 ret = ath6kl_wmi_set_wow_mode_cmd(ar->wmi, vif->fw_vif_idx,
1735 ATH6KL_WOW_MODE_ENABLE,
1736 filter,
1737 WOW_HOST_REQ_DELAY);
1738 if (ret)
1739 return ret;
1740
1741 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
1742 ATH6KL_HOST_MODE_ASLEEP);
1743 if (ret)
1744 return ret;
1745
1746 if (ar->tx_pending[ar->ctrl_ep]) {
1747 left = wait_event_interruptible_timeout(ar->event_wq,
1748 ar->tx_pending[ar->ctrl_ep] == 0, WMI_TIMEOUT);
1749 if (left == 0) {
1750 ath6kl_warn("clear wmi ctrl data timeout\n");
1751 ret = -ETIMEDOUT;
1752 } else if (left < 0) {
1753 ath6kl_warn("clear wmi ctrl data failed: %d\n", left);
1754 ret = left;
1755 }
1756 }
1757
1758 return ret;
1759}
1760
1761static int ath6kl_wow_resume(struct ath6kl *ar)
1762{
1763 struct ath6kl_vif *vif;
1764 int ret;
1765
1766 vif = ath6kl_vif_first(ar);
1767 if (!vif)
1768 return -EIO;
1769
1770 ret = ath6kl_wmi_set_host_sleep_mode_cmd(ar->wmi, vif->fw_vif_idx,
1771 ATH6KL_HOST_MODE_AWAKE);
1772 return ret;
1773}
1774
Kalle Valo52d81a62011-11-01 08:44:21 +02001775int ath6kl_cfg80211_suspend(struct ath6kl *ar,
Raja Mani0f60e9f2011-11-07 22:52:45 +02001776 enum ath6kl_cfg_suspend_mode mode,
1777 struct cfg80211_wowlan *wow)
Kalle Valo52d81a62011-11-01 08:44:21 +02001778{
1779 int ret;
1780
Kalle Valo52d81a62011-11-01 08:44:21 +02001781 switch (mode) {
Raja Manid7c44e02011-11-07 22:52:46 +02001782 case ATH6KL_CFG_SUSPEND_WOW:
1783
1784 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode suspend\n");
1785
1786 /* Flush all non control pkts in TX path */
1787 ath6kl_tx_data_cleanup(ar);
1788
1789 ret = ath6kl_wow_suspend(ar, wow);
1790 if (ret) {
1791 ath6kl_err("wow suspend failed: %d\n", ret);
1792 return ret;
1793 }
1794 ar->state = ATH6KL_STATE_WOW;
1795 break;
1796
Kalle Valo52d81a62011-11-01 08:44:21 +02001797 case ATH6KL_CFG_SUSPEND_DEEPSLEEP:
Raja Mani524441e2011-11-07 22:52:46 +02001798
1799 ath6kl_cfg80211_stop(ar);
1800
Kalle Valo52d81a62011-11-01 08:44:21 +02001801 /* save the current power mode before enabling power save */
1802 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
1803
1804 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER);
1805 if (ret) {
1806 ath6kl_warn("wmi powermode command failed during suspend: %d\n",
1807 ret);
1808 }
1809
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001810 ar->state = ATH6KL_STATE_DEEPSLEEP;
1811
Kalle Valo52d81a62011-11-01 08:44:21 +02001812 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001813
1814 case ATH6KL_CFG_SUSPEND_CUTPOWER:
Raja Mani524441e2011-11-07 22:52:46 +02001815
1816 ath6kl_cfg80211_stop(ar);
1817
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001818 if (ar->state == ATH6KL_STATE_OFF) {
1819 ath6kl_dbg(ATH6KL_DBG_SUSPEND,
1820 "suspend hw off, no action for cutpower\n");
1821 break;
1822 }
1823
1824 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "suspend cutting power\n");
1825
1826 ret = ath6kl_init_hw_stop(ar);
1827 if (ret) {
1828 ath6kl_warn("failed to stop hw during suspend: %d\n",
1829 ret);
1830 }
1831
1832 ar->state = ATH6KL_STATE_CUTPOWER;
1833
1834 break;
1835
1836 default:
1837 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02001838 }
1839
1840 return 0;
1841}
1842
1843int ath6kl_cfg80211_resume(struct ath6kl *ar)
1844{
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001845 int ret;
1846
1847 switch (ar->state) {
Raja Manid7c44e02011-11-07 22:52:46 +02001848 case ATH6KL_STATE_WOW:
1849 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "wow mode resume\n");
1850
1851 ret = ath6kl_wow_resume(ar);
1852 if (ret) {
1853 ath6kl_warn("wow mode resume failed: %d\n", ret);
1854 return ret;
1855 }
1856
1857 ar->state = ATH6KL_STATE_ON;
1858 break;
1859
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001860 case ATH6KL_STATE_DEEPSLEEP:
1861 if (ar->wmi->pwr_mode != ar->wmi->saved_pwr_mode) {
1862 ret = ath6kl_wmi_powermode_cmd(ar->wmi, 0,
1863 ar->wmi->saved_pwr_mode);
1864 if (ret) {
1865 ath6kl_warn("wmi powermode command failed during resume: %d\n",
1866 ret);
1867 }
1868 }
1869
1870 ar->state = ATH6KL_STATE_ON;
1871
1872 break;
1873
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001874 case ATH6KL_STATE_CUTPOWER:
1875 ath6kl_dbg(ATH6KL_DBG_SUSPEND, "resume restoring power\n");
1876
1877 ret = ath6kl_init_hw_start(ar);
1878 if (ret) {
1879 ath6kl_warn("Failed to boot hw in resume: %d\n", ret);
1880 return ret;
1881 }
Raja Manid7c44e02011-11-07 22:52:46 +02001882 break;
Kalle Valob4b2a0b2011-11-01 08:44:44 +02001883
Kalle Valo76a9fbe2011-11-01 08:44:28 +02001884 default:
1885 break;
Kalle Valo52d81a62011-11-01 08:44:21 +02001886 }
1887
1888 return 0;
1889}
1890
Kalle Valoabcb3442011-07-22 08:26:20 +03001891#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02001892
1893/* hif layer decides what suspend mode to use */
1894static int __ath6kl_cfg80211_suspend(struct wiphy *wiphy,
Kalle Valoabcb3442011-07-22 08:26:20 +03001895 struct cfg80211_wowlan *wow)
1896{
1897 struct ath6kl *ar = wiphy_priv(wiphy);
1898
Raja Mani0f60e9f2011-11-07 22:52:45 +02001899 return ath6kl_hif_suspend(ar, wow);
Kalle Valoabcb3442011-07-22 08:26:20 +03001900}
Chilam Ngaa6cffc2011-10-05 10:12:52 +03001901
Kalle Valo52d81a62011-11-01 08:44:21 +02001902static int __ath6kl_cfg80211_resume(struct wiphy *wiphy)
Chilam Ngaa6cffc2011-10-05 10:12:52 +03001903{
1904 struct ath6kl *ar = wiphy_priv(wiphy);
1905
1906 return ath6kl_hif_resume(ar);
1907}
Raja Mania918fb32011-11-07 22:52:46 +02001908
1909/*
1910 * FIXME: WOW suspend mode is selected if the host sdio controller supports
1911 * both sdio irq wake up and keep power. The target pulls sdio data line to
1912 * wake up the host when WOW pattern matches. This causes sdio irq handler
1913 * is being called in the host side which internally hits ath6kl's RX path.
1914 *
1915 * Since sdio interrupt is not disabled, RX path executes even before
1916 * the host executes the actual resume operation from PM module.
1917 *
1918 * In the current scenario, WOW resume should happen before start processing
1919 * any data from the target. So It's required to perform WOW resume in RX path.
1920 * Ideally we should perform WOW resume only in the actual platform
1921 * resume path. This area needs bit rework to avoid WOW resume in RX path.
1922 *
1923 * ath6kl_check_wow_status() is called from ath6kl_rx().
1924 */
1925void ath6kl_check_wow_status(struct ath6kl *ar)
1926{
1927 if (ar->state == ATH6KL_STATE_WOW)
1928 ath6kl_cfg80211_resume(ar);
1929}
1930
1931#else
1932
1933void ath6kl_check_wow_status(struct ath6kl *ar)
1934{
1935}
Kalle Valoabcb3442011-07-22 08:26:20 +03001936#endif
1937
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001938static int ath6kl_set_channel(struct wiphy *wiphy, struct net_device *dev,
1939 struct ieee80211_channel *chan,
1940 enum nl80211_channel_type channel_type)
1941{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301942 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001943
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05301944 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001945 return -EIO;
1946
1947 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: center_freq=%u hw_value=%u\n",
1948 __func__, chan->center_freq, chan->hw_value);
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05301949 vif->next_chan = chan->center_freq;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001950
1951 return 0;
1952}
1953
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001954static bool ath6kl_is_p2p_ie(const u8 *pos)
1955{
1956 return pos[0] == WLAN_EID_VENDOR_SPECIFIC && pos[1] >= 4 &&
1957 pos[2] == 0x50 && pos[3] == 0x6f &&
1958 pos[4] == 0x9a && pos[5] == 0x09;
1959}
1960
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301961static int ath6kl_set_ap_probe_resp_ies(struct ath6kl_vif *vif,
1962 const u8 *ies, size_t ies_len)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001963{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301964 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001965 const u8 *pos;
1966 u8 *buf = NULL;
1967 size_t len = 0;
1968 int ret;
1969
1970 /*
1971 * Filter out P2P IE(s) since they will be included depending on
1972 * the Probe Request frame in ath6kl_send_go_probe_resp().
1973 */
1974
1975 if (ies && ies_len) {
1976 buf = kmalloc(ies_len, GFP_KERNEL);
1977 if (buf == NULL)
1978 return -ENOMEM;
1979 pos = ies;
1980 while (pos + 1 < ies + ies_len) {
1981 if (pos + 2 + pos[1] > ies + ies_len)
1982 break;
1983 if (!ath6kl_is_p2p_ie(pos)) {
1984 memcpy(buf + len, pos, 2 + pos[1]);
1985 len += 2 + pos[1];
1986 }
1987 pos += 2 + pos[1];
1988 }
1989 }
1990
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05301991 ret = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
1992 WMI_FRAME_PROBE_RESP, buf, len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03001993 kfree(buf);
1994 return ret;
1995}
1996
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03001997static int ath6kl_ap_beacon(struct wiphy *wiphy, struct net_device *dev,
1998 struct beacon_parameters *info, bool add)
1999{
2000 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302001 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002002 struct ieee80211_mgmt *mgmt;
2003 u8 *ies;
2004 int ies_len;
2005 struct wmi_connect_cmd p;
2006 int res;
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302007 int i, ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002008
2009 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: add=%d\n", __func__, add);
2010
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302011 if (!ath6kl_cfg80211_ready(vif))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002012 return -EIO;
2013
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302014 if (vif->next_mode != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002015 return -EOPNOTSUPP;
2016
2017 if (info->beacon_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302018 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2019 WMI_FRAME_BEACON,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002020 info->beacon_ies,
2021 info->beacon_ies_len);
2022 if (res)
2023 return res;
2024 }
2025 if (info->proberesp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302026 res = ath6kl_set_ap_probe_resp_ies(vif, info->proberesp_ies,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002027 info->proberesp_ies_len);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002028 if (res)
2029 return res;
2030 }
2031 if (info->assocresp_ies) {
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302032 res = ath6kl_wmi_set_appie_cmd(ar->wmi, vif->fw_vif_idx,
2033 WMI_FRAME_ASSOC_RESP,
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002034 info->assocresp_ies,
2035 info->assocresp_ies_len);
2036 if (res)
2037 return res;
2038 }
2039
2040 if (!add)
2041 return 0;
2042
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002043 ar->ap_mode_bkey.valid = false;
2044
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002045 /* TODO:
2046 * info->interval
2047 * info->dtim_period
2048 */
2049
2050 if (info->head == NULL)
2051 return -EINVAL;
2052 mgmt = (struct ieee80211_mgmt *) info->head;
2053 ies = mgmt->u.beacon.variable;
2054 if (ies > info->head + info->head_len)
2055 return -EINVAL;
2056 ies_len = info->head + info->head_len - ies;
2057
2058 if (info->ssid == NULL)
2059 return -EINVAL;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302060 memcpy(vif->ssid, info->ssid, info->ssid_len);
2061 vif->ssid_len = info->ssid_len;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002062 if (info->hidden_ssid != NL80211_HIDDEN_SSID_NOT_IN_USE)
2063 return -EOPNOTSUPP; /* TODO */
2064
Vasanthakumar Thiagarajanbe5abaa2011-11-11 20:33:01 +05302065 ret = ath6kl_set_auth_type(vif, info->auth_type);
2066 if (ret)
2067 return ret;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002068
2069 memset(&p, 0, sizeof(p));
2070
2071 for (i = 0; i < info->crypto.n_akm_suites; i++) {
2072 switch (info->crypto.akm_suites[i]) {
2073 case WLAN_AKM_SUITE_8021X:
2074 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2075 p.auth_mode |= WPA_AUTH;
2076 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2077 p.auth_mode |= WPA2_AUTH;
2078 break;
2079 case WLAN_AKM_SUITE_PSK:
2080 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_1)
2081 p.auth_mode |= WPA_PSK_AUTH;
2082 if (info->crypto.wpa_versions & NL80211_WPA_VERSION_2)
2083 p.auth_mode |= WPA2_PSK_AUTH;
2084 break;
2085 }
2086 }
2087 if (p.auth_mode == 0)
2088 p.auth_mode = NONE_AUTH;
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302089 vif->auth_mode = p.auth_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002090
2091 for (i = 0; i < info->crypto.n_ciphers_pairwise; i++) {
2092 switch (info->crypto.ciphers_pairwise[i]) {
2093 case WLAN_CIPHER_SUITE_WEP40:
2094 case WLAN_CIPHER_SUITE_WEP104:
2095 p.prwise_crypto_type |= WEP_CRYPT;
2096 break;
2097 case WLAN_CIPHER_SUITE_TKIP:
2098 p.prwise_crypto_type |= TKIP_CRYPT;
2099 break;
2100 case WLAN_CIPHER_SUITE_CCMP:
2101 p.prwise_crypto_type |= AES_CRYPT;
2102 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002103 case WLAN_CIPHER_SUITE_SMS4:
2104 p.prwise_crypto_type |= WAPI_CRYPT;
2105 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002106 }
2107 }
Edward Lu229ed6b2011-08-30 21:58:07 +03002108 if (p.prwise_crypto_type == 0) {
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002109 p.prwise_crypto_type = NONE_CRYPT;
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302110 ath6kl_set_cipher(vif, 0, true);
Edward Lu229ed6b2011-08-30 21:58:07 +03002111 } else if (info->crypto.n_ciphers_pairwise == 1)
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302112 ath6kl_set_cipher(vif, info->crypto.ciphers_pairwise[0], true);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002113
2114 switch (info->crypto.cipher_group) {
2115 case WLAN_CIPHER_SUITE_WEP40:
2116 case WLAN_CIPHER_SUITE_WEP104:
2117 p.grp_crypto_type = WEP_CRYPT;
2118 break;
2119 case WLAN_CIPHER_SUITE_TKIP:
2120 p.grp_crypto_type = TKIP_CRYPT;
2121 break;
2122 case WLAN_CIPHER_SUITE_CCMP:
2123 p.grp_crypto_type = AES_CRYPT;
2124 break;
Dai Shuibingb8214df2011-11-03 11:39:38 +02002125 case WLAN_CIPHER_SUITE_SMS4:
2126 p.grp_crypto_type = WAPI_CRYPT;
2127 break;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002128 default:
2129 p.grp_crypto_type = NONE_CRYPT;
2130 break;
2131 }
Vasanthakumar Thiagarajan240d2792011-10-25 19:34:13 +05302132 ath6kl_set_cipher(vif, info->crypto.cipher_group, false);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002133
2134 p.nw_type = AP_NETWORK;
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302135 vif->nw_type = vif->next_mode;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002136
Vasanthakumar Thiagarajan34503342011-10-25 19:34:02 +05302137 p.ssid_len = vif->ssid_len;
2138 memcpy(p.ssid, vif->ssid, vif->ssid_len);
2139 p.dot11_auth_mode = vif->dot11_auth_mode;
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302140 p.ch = cpu_to_le16(vif->next_chan);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002141
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302142 res = ath6kl_wmi_ap_profile_commit(ar->wmi, vif->fw_vif_idx, &p);
Jouni Malinen9a5b1312011-08-30 21:57:52 +03002143 if (res < 0)
2144 return res;
2145
2146 return 0;
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002147}
2148
2149static int ath6kl_add_beacon(struct wiphy *wiphy, struct net_device *dev,
2150 struct beacon_parameters *info)
2151{
2152 return ath6kl_ap_beacon(wiphy, dev, info, true);
2153}
2154
2155static int ath6kl_set_beacon(struct wiphy *wiphy, struct net_device *dev,
2156 struct beacon_parameters *info)
2157{
2158 return ath6kl_ap_beacon(wiphy, dev, info, false);
2159}
2160
2161static int ath6kl_del_beacon(struct wiphy *wiphy, struct net_device *dev)
2162{
2163 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302164 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002165
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302166 if (vif->nw_type != AP_NETWORK)
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002167 return -EOPNOTSUPP;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302168 if (!test_bit(CONNECTED, &vif->flags))
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002169 return -ENOTCONN;
2170
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302171 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302172 clear_bit(CONNECTED, &vif->flags);
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002173
2174 return 0;
2175}
2176
Jouni Malinen23875132011-08-30 21:57:53 +03002177static int ath6kl_change_station(struct wiphy *wiphy, struct net_device *dev,
2178 u8 *mac, struct station_parameters *params)
2179{
2180 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302181 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen23875132011-08-30 21:57:53 +03002182
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302183 if (vif->nw_type != AP_NETWORK)
Jouni Malinen23875132011-08-30 21:57:53 +03002184 return -EOPNOTSUPP;
2185
2186 /* Use this only for authorizing/unauthorizing a station */
2187 if (!(params->sta_flags_mask & BIT(NL80211_STA_FLAG_AUTHORIZED)))
2188 return -EOPNOTSUPP;
2189
2190 if (params->sta_flags_set & BIT(NL80211_STA_FLAG_AUTHORIZED))
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302191 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2192 WMI_AP_MLME_AUTHORIZE, mac, 0);
2193 return ath6kl_wmi_ap_set_mlme(ar->wmi, vif->fw_vif_idx,
2194 WMI_AP_MLME_UNAUTHORIZE, mac, 0);
Jouni Malinen23875132011-08-30 21:57:53 +03002195}
2196
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002197static int ath6kl_remain_on_channel(struct wiphy *wiphy,
2198 struct net_device *dev,
2199 struct ieee80211_channel *chan,
2200 enum nl80211_channel_type channel_type,
2201 unsigned int duration,
2202 u64 *cookie)
2203{
2204 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302205 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen10522612011-10-27 16:00:13 +03002206 u32 id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002207
2208 /* TODO: if already pending or ongoing remain-on-channel,
2209 * return -EBUSY */
Jouni Malinen10522612011-10-27 16:00:13 +03002210 id = ++vif->last_roc_id;
2211 if (id == 0) {
2212 /* Do not use 0 as the cookie value */
2213 id = ++vif->last_roc_id;
2214 }
2215 *cookie = id;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002216
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302217 return ath6kl_wmi_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx,
2218 chan->center_freq, duration);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002219}
2220
2221static int ath6kl_cancel_remain_on_channel(struct wiphy *wiphy,
2222 struct net_device *dev,
2223 u64 cookie)
2224{
2225 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302226 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002227
Jouni Malinen10522612011-10-27 16:00:13 +03002228 if (cookie != vif->last_roc_id)
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002229 return -ENOENT;
Jouni Malinen10522612011-10-27 16:00:13 +03002230 vif->last_cancel_roc_id = cookie;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002231
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302232 return ath6kl_wmi_cancel_remain_on_chnl_cmd(ar->wmi, vif->fw_vif_idx);
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002233}
2234
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302235static int ath6kl_send_go_probe_resp(struct ath6kl_vif *vif,
2236 const u8 *buf, size_t len,
2237 unsigned int freq)
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002238{
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302239 struct ath6kl *ar = vif->ar;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002240 const u8 *pos;
2241 u8 *p2p;
2242 int p2p_len;
2243 int ret;
2244 const struct ieee80211_mgmt *mgmt;
2245
2246 mgmt = (const struct ieee80211_mgmt *) buf;
2247
2248 /* Include P2P IE(s) from the frame generated in user space. */
2249
2250 p2p = kmalloc(len, GFP_KERNEL);
2251 if (p2p == NULL)
2252 return -ENOMEM;
2253 p2p_len = 0;
2254
2255 pos = mgmt->u.probe_resp.variable;
2256 while (pos + 1 < buf + len) {
2257 if (pos + 2 + pos[1] > buf + len)
2258 break;
2259 if (ath6kl_is_p2p_ie(pos)) {
2260 memcpy(p2p + p2p_len, pos, 2 + pos[1]);
2261 p2p_len += 2 + pos[1];
2262 }
2263 pos += 2 + pos[1];
2264 }
2265
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302266 ret = ath6kl_wmi_send_probe_response_cmd(ar->wmi, vif->fw_vif_idx, freq,
2267 mgmt->da, p2p, p2p_len);
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002268 kfree(p2p);
2269 return ret;
2270}
2271
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002272static int ath6kl_mgmt_tx(struct wiphy *wiphy, struct net_device *dev,
2273 struct ieee80211_channel *chan, bool offchan,
2274 enum nl80211_channel_type channel_type,
2275 bool channel_type_valid, unsigned int wait,
Johannes Berge247bd902011-11-04 11:18:21 +01002276 const u8 *buf, size_t len, bool no_cck,
2277 bool dont_wait_for_ack, u64 *cookie)
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002278{
2279 struct ath6kl *ar = ath6kl_priv(dev);
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302280 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002281 u32 id;
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002282 const struct ieee80211_mgmt *mgmt;
2283
2284 mgmt = (const struct ieee80211_mgmt *) buf;
2285 if (buf + len >= mgmt->u.probe_resp.variable &&
Vasanthakumar Thiagarajanf5938f22011-10-25 19:34:03 +05302286 vif->nw_type == AP_NETWORK && test_bit(CONNECTED, &vif->flags) &&
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002287 ieee80211_is_probe_resp(mgmt->frame_control)) {
2288 /*
2289 * Send Probe Response frame in AP mode using a separate WMI
2290 * command to allow the target to fill in the generic IEs.
2291 */
2292 *cookie = 0; /* TX status not supported */
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302293 return ath6kl_send_go_probe_resp(vif, buf, len,
Jouni Malinen8bdfbf42011-08-30 21:58:09 +03002294 chan->center_freq);
2295 }
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002296
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302297 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002298 if (id == 0) {
2299 /*
2300 * 0 is a reserved value in the WMI command and shall not be
2301 * used for the command.
2302 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302303 id = vif->send_action_id++;
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002304 }
2305
2306 *cookie = id;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302307 return ath6kl_wmi_send_action_cmd(ar->wmi, vif->fw_vif_idx, id,
2308 chan->center_freq, wait,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002309 buf, len);
2310}
2311
Jouni Malinenae32c302011-08-30 21:58:01 +03002312static void ath6kl_mgmt_frame_register(struct wiphy *wiphy,
2313 struct net_device *dev,
2314 u16 frame_type, bool reg)
2315{
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302316 struct ath6kl_vif *vif = netdev_priv(dev);
Jouni Malinenae32c302011-08-30 21:58:01 +03002317
2318 ath6kl_dbg(ATH6KL_DBG_WLAN_CFG, "%s: frame_type=0x%x reg=%d\n",
2319 __func__, frame_type, reg);
2320 if (frame_type == IEEE80211_STYPE_PROBE_REQ) {
2321 /*
2322 * Note: This notification callback is not allowed to sleep, so
2323 * we cannot send WMI_PROBE_REQ_REPORT_CMD here. Instead, we
2324 * hardcode target to report Probe Request frames all the time.
2325 */
Vasanthakumar Thiagarajancf5333d2011-10-25 19:34:10 +05302326 vif->probe_req_report = reg;
Jouni Malinenae32c302011-08-30 21:58:01 +03002327 }
2328}
2329
Jouni Malinenf80574a2011-08-30 21:58:04 +03002330static const struct ieee80211_txrx_stypes
2331ath6kl_mgmt_stypes[NUM_NL80211_IFTYPES] = {
2332 [NL80211_IFTYPE_STATION] = {
2333 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2334 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2335 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2336 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2337 },
2338 [NL80211_IFTYPE_P2P_CLIENT] = {
2339 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2340 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2341 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2342 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2343 },
2344 [NL80211_IFTYPE_P2P_GO] = {
2345 .tx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2346 BIT(IEEE80211_STYPE_PROBE_RESP >> 4),
2347 .rx = BIT(IEEE80211_STYPE_ACTION >> 4) |
2348 BIT(IEEE80211_STYPE_PROBE_REQ >> 4)
2349 },
2350};
2351
Kalle Valobdcd8172011-07-18 00:22:30 +03002352static struct cfg80211_ops ath6kl_cfg80211_ops = {
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302353 .add_virtual_intf = ath6kl_cfg80211_add_iface,
2354 .del_virtual_intf = ath6kl_cfg80211_del_iface,
Kalle Valobdcd8172011-07-18 00:22:30 +03002355 .change_virtual_intf = ath6kl_cfg80211_change_iface,
2356 .scan = ath6kl_cfg80211_scan,
2357 .connect = ath6kl_cfg80211_connect,
2358 .disconnect = ath6kl_cfg80211_disconnect,
2359 .add_key = ath6kl_cfg80211_add_key,
2360 .get_key = ath6kl_cfg80211_get_key,
2361 .del_key = ath6kl_cfg80211_del_key,
2362 .set_default_key = ath6kl_cfg80211_set_default_key,
2363 .set_wiphy_params = ath6kl_cfg80211_set_wiphy_params,
2364 .set_tx_power = ath6kl_cfg80211_set_txpower,
2365 .get_tx_power = ath6kl_cfg80211_get_txpower,
2366 .set_power_mgmt = ath6kl_cfg80211_set_power_mgmt,
2367 .join_ibss = ath6kl_cfg80211_join_ibss,
2368 .leave_ibss = ath6kl_cfg80211_leave_ibss,
2369 .get_station = ath6kl_get_station,
2370 .set_pmksa = ath6kl_set_pmksa,
2371 .del_pmksa = ath6kl_del_pmksa,
2372 .flush_pmksa = ath6kl_flush_pmksa,
Kalle Valo003353b0d2011-09-01 10:14:21 +03002373 CFG80211_TESTMODE_CMD(ath6kl_tm_cmd)
Kalle Valoabcb3442011-07-22 08:26:20 +03002374#ifdef CONFIG_PM
Kalle Valo52d81a62011-11-01 08:44:21 +02002375 .suspend = __ath6kl_cfg80211_suspend,
2376 .resume = __ath6kl_cfg80211_resume,
Kalle Valoabcb3442011-07-22 08:26:20 +03002377#endif
Jouni Malinen6a7c9ba2011-08-30 21:57:50 +03002378 .set_channel = ath6kl_set_channel,
2379 .add_beacon = ath6kl_add_beacon,
2380 .set_beacon = ath6kl_set_beacon,
2381 .del_beacon = ath6kl_del_beacon,
Jouni Malinen23875132011-08-30 21:57:53 +03002382 .change_station = ath6kl_change_station,
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002383 .remain_on_channel = ath6kl_remain_on_channel,
2384 .cancel_remain_on_channel = ath6kl_cancel_remain_on_channel,
Jouni Malinen8a6c80602011-08-30 21:57:56 +03002385 .mgmt_tx = ath6kl_mgmt_tx,
Jouni Malinenae32c302011-08-30 21:58:01 +03002386 .mgmt_frame_register = ath6kl_mgmt_frame_register,
Kalle Valobdcd8172011-07-18 00:22:30 +03002387};
2388
Kalle Valoec4b7f62011-11-01 08:44:04 +02002389void ath6kl_cfg80211_stop(struct ath6kl *ar)
2390{
2391 struct ath6kl_vif *vif;
2392
2393 /* FIXME: for multi vif */
2394 vif = ath6kl_vif_first(ar);
2395 if (!vif) {
2396 /* save the current power mode before enabling power save */
2397 ar->wmi->saved_pwr_mode = ar->wmi->pwr_mode;
2398
2399 if (ath6kl_wmi_powermode_cmd(ar->wmi, 0, REC_POWER) != 0)
2400 ath6kl_warn("ath6kl_deep_sleep_enable: "
2401 "wmi_powermode_cmd failed\n");
2402 return;
2403 }
2404
2405 switch (vif->sme_state) {
2406 case SME_CONNECTING:
2407 cfg80211_connect_result(vif->ndev, vif->bssid, NULL, 0,
2408 NULL, 0,
2409 WLAN_STATUS_UNSPECIFIED_FAILURE,
2410 GFP_KERNEL);
2411 break;
2412 case SME_CONNECTED:
2413 default:
2414 /*
2415 * FIXME: oddly enough smeState is in DISCONNECTED during
2416 * suspend, why? Need to send disconnected event in that
2417 * state.
2418 */
2419 cfg80211_disconnected(vif->ndev, 0, NULL, 0, GFP_KERNEL);
2420 break;
2421 }
2422
2423 if (test_bit(CONNECTED, &vif->flags) ||
2424 test_bit(CONNECT_PEND, &vif->flags))
2425 ath6kl_wmi_disconnect_cmd(ar->wmi, vif->fw_vif_idx);
2426
2427 vif->sme_state = SME_DISCONNECTED;
Kalle Valo1f40525512011-11-01 08:44:13 +02002428 clear_bit(CONNECTED, &vif->flags);
2429 clear_bit(CONNECT_PEND, &vif->flags);
Kalle Valoec4b7f62011-11-01 08:44:04 +02002430
2431 /* disable scanning */
2432 if (ath6kl_wmi_scanparams_cmd(ar->wmi, vif->fw_vif_idx, 0xFFFF, 0, 0,
2433 0, 0, 0, 0, 0, 0, 0) != 0)
2434 printk(KERN_WARNING "ath6kl: failed to disable scan "
2435 "during suspend\n");
2436
2437 ath6kl_cfg80211_scan_complete_event(vif, true);
2438}
2439
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302440struct ath6kl *ath6kl_core_alloc(struct device *dev)
Kalle Valobdcd8172011-07-18 00:22:30 +03002441{
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002442 struct ath6kl *ar;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302443 struct wiphy *wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302444 u8 ctr;
Kalle Valobdcd8172011-07-18 00:22:30 +03002445
2446 /* create a new wiphy for use with cfg80211 */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302447 wiphy = wiphy_new(&ath6kl_cfg80211_ops, sizeof(struct ath6kl));
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302448
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302449 if (!wiphy) {
Kalle Valobdcd8172011-07-18 00:22:30 +03002450 ath6kl_err("couldn't allocate wiphy device\n");
Kalle Valobdcd8172011-07-18 00:22:30 +03002451 return NULL;
2452 }
2453
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302454 ar = wiphy_priv(wiphy);
Vasanthakumar Thiagarajan774439a2011-11-18 10:05:26 +05302455 ar->p2p = !!ath6kl_p2p;
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302456 ar->wiphy = wiphy;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302457 ar->dev = dev;
2458
Kalle Valo71f96ee2011-11-14 19:31:30 +02002459 ar->vif_max = 1;
2460
Vasanthakumar Thiagarajan3226f68a2011-10-25 19:34:24 +05302461 ar->max_norm_iface = 1;
2462
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302463 spin_lock_init(&ar->lock);
2464 spin_lock_init(&ar->mcastpsq_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302465 spin_lock_init(&ar->list_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302466
2467 init_waitqueue_head(&ar->event_wq);
2468 sema_init(&ar->sem, 1);
2469
2470 INIT_LIST_HEAD(&ar->amsdu_rx_buffer_queue);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302471 INIT_LIST_HEAD(&ar->vif_list);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302472
2473 clear_bit(WMI_ENABLED, &ar->flag);
2474 clear_bit(SKIP_SCAN, &ar->flag);
2475 clear_bit(DESTROY_IN_PROGRESS, &ar->flag);
2476
2477 ar->listen_intvl_t = A_DEFAULT_LISTEN_INTERVAL;
2478 ar->listen_intvl_b = 0;
2479 ar->tx_pwr = 0;
2480
2481 ar->intra_bss = 1;
2482 memset(&ar->sc_params, 0, sizeof(ar->sc_params));
2483 ar->sc_params.short_scan_ratio = WMI_SHORTSCANRATIO_DEFAULT;
2484 ar->sc_params.scan_ctrl_flags = DEFAULT_SCAN_CTRL_FLAGS;
2485 ar->lrssi_roam_threshold = DEF_LRSSI_ROAM_THRESHOLD;
2486
Kalle Valo76a9fbe2011-11-01 08:44:28 +02002487 ar->state = ATH6KL_STATE_OFF;
2488
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302489 memset((u8 *)ar->sta_list, 0,
2490 AP_MAX_NUM_STA * sizeof(struct ath6kl_sta));
2491
2492 /* Init the PS queues */
2493 for (ctr = 0; ctr < AP_MAX_NUM_STA; ctr++) {
2494 spin_lock_init(&ar->sta_list[ctr].psq_lock);
2495 skb_queue_head_init(&ar->sta_list[ctr].psq);
2496 }
2497
2498 skb_queue_head_init(&ar->mcastpsq);
2499
2500 memcpy(ar->ap_country_code, DEF_AP_COUNTRY_CODE, 3);
2501
2502 return ar;
2503}
2504
2505int ath6kl_register_ieee80211_hw(struct ath6kl *ar)
2506{
2507 struct wiphy *wiphy = ar->wiphy;
2508 int ret;
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002509
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302510 wiphy->mgmt_stypes = ath6kl_mgmt_stypes;
Jouni Malinenf80574a2011-08-30 21:58:04 +03002511
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302512 wiphy->max_remain_on_channel_duration = 5000;
Jouni Malinen63fa1e02011-08-30 21:57:55 +03002513
Kalle Valobdcd8172011-07-18 00:22:30 +03002514 /* set device pointer for wiphy */
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302515 set_wiphy_dev(wiphy, ar->dev);
Kalle Valobdcd8172011-07-18 00:22:30 +03002516
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302517 wiphy->interface_modes = BIT(NL80211_IFTYPE_STATION) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302518 BIT(NL80211_IFTYPE_ADHOC) |
2519 BIT(NL80211_IFTYPE_AP);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002520 if (ar->p2p) {
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302521 wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_GO) |
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302522 BIT(NL80211_IFTYPE_P2P_CLIENT);
Jouni Malinen6bbc7c32011-09-05 17:38:47 +03002523 }
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302524
Kalle Valobdcd8172011-07-18 00:22:30 +03002525 /* max num of ssids that can be probed during scanning */
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302526 wiphy->max_scan_ssids = MAX_PROBED_SSID_INDEX;
2527 wiphy->max_scan_ie_len = 1000; /* FIX: what is correct limit? */
2528 wiphy->bands[IEEE80211_BAND_2GHZ] = &ath6kl_band_2ghz;
2529 wiphy->bands[IEEE80211_BAND_5GHZ] = &ath6kl_band_5ghz;
2530 wiphy->signal_type = CFG80211_SIGNAL_TYPE_MBM;
Kalle Valobdcd8172011-07-18 00:22:30 +03002531
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302532 wiphy->cipher_suites = cipher_suites;
2533 wiphy->n_cipher_suites = ARRAY_SIZE(cipher_suites);
Kalle Valobdcd8172011-07-18 00:22:30 +03002534
Raja Manieae9e062011-11-07 22:52:46 +02002535 wiphy->wowlan.flags = WIPHY_WOWLAN_MAGIC_PKT |
2536 WIPHY_WOWLAN_DISCONNECT |
2537 WIPHY_WOWLAN_GTK_REKEY_FAILURE |
2538 WIPHY_WOWLAN_SUPPORTS_GTK_REKEY |
2539 WIPHY_WOWLAN_EAP_IDENTITY_REQ |
2540 WIPHY_WOWLAN_4WAY_HANDSHAKE;
2541 wiphy->wowlan.n_patterns = WOW_MAX_FILTERS_PER_LIST;
2542 wiphy->wowlan.pattern_min_len = 1;
2543 wiphy->wowlan.pattern_max_len = WOW_PATTERN_SIZE;
2544
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302545 ret = wiphy_register(wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002546 if (ret < 0) {
2547 ath6kl_err("couldn't register wiphy device\n");
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302548 return ret;
Kalle Valobdcd8172011-07-18 00:22:30 +03002549 }
2550
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302551 return 0;
Kalle Valobdcd8172011-07-18 00:22:30 +03002552}
2553
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302554static int ath6kl_init_if_data(struct ath6kl_vif *vif)
Kalle Valobdcd8172011-07-18 00:22:30 +03002555{
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302556 vif->aggr_cntxt = aggr_init(vif->ndev);
2557 if (!vif->aggr_cntxt) {
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302558 ath6kl_err("failed to initialize aggr\n");
2559 return -ENOMEM;
2560 }
Kalle Valobdcd8172011-07-18 00:22:30 +03002561
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302562 setup_timer(&vif->disconnect_timer, disconnect_timer_handler,
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302563 (unsigned long) vif->ndev);
Vasanthakumar Thiagarajande3ad712011-10-25 19:34:08 +05302564 set_bit(WMM_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan478ac022011-10-25 19:34:19 +05302565 spin_lock_init(&vif->if_lock);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302566
2567 return 0;
2568}
2569
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302570void ath6kl_deinit_if_data(struct ath6kl_vif *vif)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302571{
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302572 struct ath6kl *ar = vif->ar;
2573
Vasanthakumar Thiagarajan2132c692011-10-25 19:34:07 +05302574 aggr_module_destroy(vif->aggr_cntxt);
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302575
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302576 ar->avail_idx_map |= BIT(vif->fw_vif_idx);
2577
2578 if (vif->nw_type == ADHOC_NETWORK)
2579 ar->ibss_if_active = false;
2580
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302581 unregister_netdevice(vif->ndev);
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302582
2583 ar->num_vif--;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302584}
2585
2586struct net_device *ath6kl_interface_add(struct ath6kl *ar, char *name,
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302587 enum nl80211_iftype type, u8 fw_vif_idx,
2588 u8 nw_type)
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302589{
2590 struct net_device *ndev;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302591 struct ath6kl_vif *vif;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302592
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302593 ndev = alloc_netdev(sizeof(*vif), name, ether_setup);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302594 if (!ndev)
2595 return NULL;
2596
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302597 vif = netdev_priv(ndev);
2598 ndev->ieee80211_ptr = &vif->wdev;
2599 vif->wdev.wiphy = ar->wiphy;
2600 vif->ar = ar;
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302601 vif->ndev = ndev;
2602 SET_NETDEV_DEV(ndev, wiphy_dev(vif->wdev.wiphy));
2603 vif->wdev.netdev = ndev;
2604 vif->wdev.iftype = type;
Vasanthakumar Thiagarajan334234b2011-10-25 19:34:12 +05302605 vif->fw_vif_idx = fw_vif_idx;
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302606 vif->nw_type = vif->next_mode = nw_type;
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302607
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302608 memcpy(ndev->dev_addr, ar->mac_addr, ETH_ALEN);
2609 if (fw_vif_idx != 0)
2610 ndev->dev_addr[0] = (ndev->dev_addr[0] ^ (1 << fw_vif_idx)) |
2611 0x2;
2612
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302613 init_netdev(ndev);
2614
Vasanthakumar Thiagarajane29f25f2011-10-25 19:34:15 +05302615 ath6kl_init_control_info(vif);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302616
2617 /* TODO: Pass interface specific pointer instead of ar */
Vasanthakumar Thiagarajan108438b2011-10-25 19:34:00 +05302618 if (ath6kl_init_if_data(vif))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302619 goto err;
2620
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302621 if (register_netdevice(ndev))
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302622 goto err;
2623
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302624 ar->avail_idx_map &= ~BIT(fw_vif_idx);
Vasanthakumar Thiagarajan14ee6f62011-10-25 19:34:09 +05302625 vif->sme_state = SME_DISCONNECTED;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302626 set_bit(WLAN_ENABLED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302627 ar->wlan_pwr_state = WLAN_POWER_STATE_ON;
Vasanthakumar Thiagarajan59c98442011-10-25 19:34:01 +05302628 set_bit(NETDEV_REGISTERED, &vif->flags);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302629
Vasanthakumar Thiagarajan55055972011-10-25 19:34:23 +05302630 if (type == NL80211_IFTYPE_ADHOC)
2631 ar->ibss_if_active = true;
2632
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05302633 spin_lock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302634 list_add_tail(&vif->list, &ar->vif_list);
Vasanthakumar Thiagarajan11f6e402011-11-01 16:38:50 +05302635 spin_unlock_bh(&ar->list_lock);
Vasanthakumar Thiagarajan990bd912011-10-25 19:34:20 +05302636
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302637 return ndev;
2638
2639err:
Vasanthakumar Thiagarajan27929722011-10-25 19:34:21 +05302640 aggr_module_destroy(vif->aggr_cntxt);
2641 free_netdev(ndev);
Vasanthakumar Thiagarajan8dafb702011-10-25 19:33:58 +05302642 return NULL;
2643}
2644
2645void ath6kl_deinit_ieee80211_hw(struct ath6kl *ar)
2646{
Vasanthakumar Thiagarajanbe98e3a2011-10-25 19:33:57 +05302647 wiphy_unregister(ar->wiphy);
2648 wiphy_free(ar->wiphy);
Kalle Valobdcd8172011-07-18 00:22:30 +03002649}